Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1677 lines
50 KiB

  1. /************************************************************/
  2. /* Windows Write, Copyright 1985-1992 Microsoft Corporation */
  3. /************************************************************/
  4. /* This file contains the routines for creating, displaying, and manipulating
  5. the ruler for Memo. */
  6. #define NOGDICAPMASKS
  7. #define NOVIRTUALKEYCODES
  8. #define NOWINMESSAGES
  9. #define NOMENUS
  10. #define NOICON
  11. #define NOKEYSTATE
  12. #define NOSYSCOMMANDS
  13. #define NOATOM
  14. #define NOBRUSH
  15. #define NOCLIPBOARD
  16. #define NOCOLOR
  17. #define NOCREATESTRUCT
  18. #define NOCTLMGR
  19. #define NODRAWTEXT
  20. #define NOMB
  21. #define NOMEMMGR
  22. #define NOMENUS
  23. #define NOMETAFILE
  24. #define NOMSG
  25. #define NOOPENFILE
  26. #define NOREGION
  27. #define NOSCROLL
  28. #define NOSOUND
  29. #define NOWH
  30. #define NOWINOFFSETS
  31. #define NOWNDCLASS
  32. #define NOCOMM
  33. #include <windows.h>
  34. #include "mw.h"
  35. #include "cmddefs.h"
  36. #include "wwdefs.h"
  37. #include "rulerdef.h"
  38. #include "propdefs.h"
  39. #include "prmdefs.h"
  40. #include "docdefs.h"
  41. #include "bitmaps.h"
  42. #define MERGEMARK 0x00990066
  43. extern HWND hParentWw;
  44. extern HANDLE hMmwModInstance;
  45. extern HCURSOR vhcIBeam;
  46. extern struct DOD (**hpdocdod)[];
  47. extern struct WWD *pwwdCur;
  48. extern struct PAP vpapAbs;
  49. extern struct SEP vsepAbs;
  50. extern struct SEL selCur;
  51. extern typeCP cpMacCur;
  52. extern int docCur;
  53. extern int vdocParaCache;
  54. extern int dypRuler;
  55. extern int dxpLogInch;
  56. extern int dypLogInch;
  57. extern int dxpLogCm;
  58. extern int dypLogCm;
  59. extern int xpSelBar;
  60. extern HWND vhWndRuler;
  61. extern int vdxaTextRuler;
  62. extern int mprmkdxa[rmkMARGMAX];
  63. extern int vfTabsChanged;
  64. extern int vfMargChanged;
  65. extern struct WWD rgwwd[];
  66. extern long rgbBkgrnd;
  67. extern long rgbText;
  68. extern HBRUSH hbrBkgrnd;
  69. extern long ropErase;
  70. extern BOOL vfMonochrome;
  71. extern BOOL vfEraseWw;
  72. extern int vfIconic;
  73. #ifdef RULERALSO
  74. extern HWND vhDlgIndent;
  75. #endif /* RULERALSO */
  76. HDC vhDCRuler = NULL;
  77. HDC hMDCBitmap = NULL;
  78. HDC hMDCScreen = NULL;
  79. HBITMAP hbmBtn = NULL;
  80. HBITMAP hbmMark = NULL;
  81. HBITMAP hbmNullRuler = NULL;
  82. int dxpRuler;
  83. int viBmRuler = -1; /* Index into [CGA/EGA/VGA/8514] bitmaps (see
  84. WRITE.RC). Set appropriately in FCreateRuler(). */
  85. static RECT rgrcRulerBtn[btnMaxUsed];
  86. static int mprlcbtnDown[rlcBTNMAX] = {btnNIL, btnNIL, btnNIL};
  87. static struct TBD rgtbdRuler[itbdMax];
  88. static int xpMinCur;
  89. static int dxpMark;
  90. static int dypMark;
  91. static int btnTabSave = btnLTAB;
  92. near UpdateRulerBtn(int, int);
  93. BOOL near FCreateRuler(void);
  94. int near DestroyRuler(void);
  95. int near RulerStateFromPt(POINT, int *, int *);
  96. int near MergeRulerMark(int, int, BOOL);
  97. BOOL near FPointNear(unsigned, unsigned);
  98. unsigned near XaQuantize(int);
  99. int near DeleteRulerTab(struct TBD *);
  100. int near InsertRulerTab(struct TBD *);
  101. BOOL near FCloseXa(unsigned, unsigned);
  102. #ifdef KINTL
  103. unsigned near XaKickBackXa(unsigned);
  104. near XpKickBackXp(int);
  105. unsigned near XaQuantizeXa(unsigned);
  106. #endif /* KINTL */
  107. fnShowRuler()
  108. {
  109. /* This routine toggles the creation and the destruction of the ruler
  110. window. */
  111. StartLongOp();
  112. if (pwwdCur->fRuler)
  113. {
  114. /* Take down the existing ruler. */
  115. DestroyRuler();
  116. SetRulerMenu(TRUE);
  117. }
  118. else
  119. {
  120. /* There is no ruler, bring one up. */
  121. if (FCreateRuler())
  122. {
  123. SetRulerMenu(FALSE);
  124. }
  125. }
  126. EndLongOp(vhcIBeam);
  127. }
  128. BOOL near FCreateRuler()
  129. {
  130. /* This routine creates the ruler child window and positions it on the
  131. screen. */
  132. extern CHAR szRulerClass[];
  133. int xpMac = pwwdCur->xpMac;
  134. int ypMac = pwwdCur->ypMac;
  135. LOGFONT lf;
  136. HFONT hf;
  137. int dyp;
  138. HPEN hpen;
  139. RECT rc;
  140. TEXTMETRIC tmSys;
  141. HDC hdcSys;
  142. /* Create the ruler window. */
  143. if ((vhWndRuler = CreateWindow((LPSTR)szRulerClass, (LPSTR)NULL,
  144. WS_CHILD | WS_CLIPSIBLINGS, 0, 0, 0, 0, hParentWw, NULL, hMmwModInstance,
  145. (LPSTR)NULL)) == NULL)
  146. {
  147. goto Error2;
  148. }
  149. /* Save the DC and the memory DC. */
  150. if ((vhDCRuler = GetDC(vhWndRuler)) == NULL || (hMDCBitmap =
  151. CreateCompatibleDC(vhDCRuler)) == NULL || (hMDCScreen =
  152. CreateCompatibleDC(vhDCRuler)) == NULL)
  153. {
  154. goto Error1;
  155. }
  156. /* Create a null bitmap for the ruler. */
  157. if ((hbmNullRuler = CreateBitmap(1, 1, 1, 1, (LPSTR)NULL)) == NULL)
  158. {
  159. goto Error1;
  160. }
  161. /* New for Write 3.0: we have a variety of bitmaps for the ruler buttons
  162. and marks -- loaded depending on the resolution of the user's display.
  163. All we really want to do here is set viBmRuler, which indexes into the
  164. appropriate bitmaps (see bitmaps.h) ..pault 7/13/89 */
  165. if (viBmRuler < 0)
  166. {
  167. /* This idea of passing NULL to GetDC borrowed from WinWord ..pt */
  168. if ((hdcSys = GetDC(NULL)) == NULL)
  169. goto Error1;
  170. else
  171. {
  172. int tmHeight;
  173. GetTextMetrics(hdcSys, (LPTEXTMETRIC) &tmSys);
  174. tmHeight = tmSys.tmHeight;
  175. ReleaseDC(NULL, hdcSys);
  176. viBmRuler = 0;
  177. if (tmHeight > 8)
  178. viBmRuler++;
  179. if (tmHeight > 12)
  180. viBmRuler++;
  181. if (tmHeight > 16)
  182. viBmRuler++;
  183. }
  184. Diag(CommSzNum("FCreateRuler: index into [CGA/EGA/VGA/8514] bitmaps==", viBmRuler));
  185. Assert(idBmBtns + viBmRuler < idBmBtnsMax);
  186. Assert(idBmMarks + viBmRuler < idBmMarksMax);
  187. }
  188. /* Get the bitmaps for the ruler buttons and the ruler marks. */
  189. if (hbmBtn == NULL || SelectObject(hMDCBitmap, hbmBtn) == NULL)
  190. {
  191. if (NULL == (hbmBtn = LoadBitmap(hMmwModInstance,
  192. MAKEINTRESOURCE(idBmBtns+viBmRuler))))
  193. {
  194. goto Error1;
  195. }
  196. }
  197. if (hbmMark == NULL || SelectObject(hMDCBitmap, hbmMark) == NULL)
  198. {
  199. if (NULL == (hbmMark = LoadBitmap(hMmwModInstance,
  200. MAKEINTRESOURCE(idBmMarks+viBmRuler))))
  201. {
  202. goto Error1;
  203. }
  204. }
  205. /* Get the font for labelling the ruler ticks. */
  206. bltbc(&lf, 0, sizeof(LOGFONT));
  207. lf.lfHeight = -MultDiv(czaPoint * 8, dypLogInch, czaInch);
  208. if ((hf = CreateFontIndirect(&lf)) != NULL)
  209. {
  210. if (SelectObject(vhDCRuler, hf) == NULL)
  211. {
  212. DeleteObject(hf);
  213. }
  214. }
  215. /* If this is the first time the ruler is created, then initialize the
  216. static variables. */
  217. if (dypRuler == 0)
  218. {
  219. int dxpMajor;
  220. int dxpMinor;
  221. BITMAP bm;
  222. int xp;
  223. int dxpBtn;
  224. int btn;
  225. PRECT prc;
  226. TEXTMETRIC tm;
  227. /* Initialize the starting position of the buttons. */
  228. dxpMinor = (dxpMajor = dxpLogInch >> 1) >> 2;
  229. xp = xpSelBar + dxpMajor + (dxpMajor >> 1);
  230. /* Get the width and height of the buttons. */
  231. GetObject(hbmBtn, sizeof(BITMAP), (LPSTR)&bm);
  232. /* Factor of 2 since we have positive and negative images
  233. of each button embedded in the bitmap now ..pault */
  234. dxpBtn = bm.bmWidth / (btnMaxReal*2);
  235. dypRuler = bm.bmHeight;
  236. /* Position the buttons. */
  237. for (prc = &rgrcRulerBtn[btn = btnMIN]; btn < btnMaxUsed; btn++, prc++)
  238. {
  239. prc->left = xp;
  240. prc->top = 1;
  241. prc->right = (xp += dxpBtn);
  242. prc->bottom = bm.bmHeight + 1;
  243. xp += (btn == btnTABMAX || btn == btnSPACEMAX) ? dxpMajor :
  244. dxpMinor;
  245. }
  246. /* Get the width and height of the tab marks. */
  247. GetObject(hbmMark, sizeof(BITMAP), (LPSTR)&bm);
  248. dxpMark = bm.bmWidth / rmkMAX;
  249. dypMark = bm.bmHeight;
  250. /* Lastly, initialize the height of the ruler. (Four is for the two
  251. lines at the bottom of the ruler plus two blank lines.) */
  252. GetTextMetrics(vhDCRuler, (LPTEXTMETRIC)&tm);
  253. dypRuler += dypMark + (tm.tmAscent - tm.tmInternalLeading) + 4;
  254. }
  255. /* Move the document window to make room for the ruler. */
  256. pwwdCur->fRuler = TRUE;
  257. dyp = dypRuler - (pwwdCur->ypMin - 1);
  258. MoveWindow(wwdCurrentDoc.wwptr, 0, dyp, xpMac, ypMac - dyp, FALSE);
  259. /* Erase the top of the document window. */
  260. PatBlt(wwdCurrentDoc.hDC, 0, 0, xpMac, wwdCurrentDoc.ypMin, ropErase);
  261. rc.left = rc.top = 0;
  262. rc.right = xpMac;
  263. rc.bottom = wwdCurrentDoc.ypMin;
  264. ValidateRect(wwdCurrentDoc.wwptr, (LPRECT)&rc);
  265. UpdateWindow(wwdCurrentDoc.wwptr);
  266. /* Move the ruler into position. */
  267. MoveWindow(vhWndRuler, 0, 0, xpMac, dypRuler, FALSE);
  268. BringWindowToTop(vhWndRuler);
  269. /* Set the DC to transparent mode. */
  270. SetBkMode(vhDCRuler, TRANSPARENT);
  271. /* Set the background and foreground colors for the ruler. */
  272. SetBkColor(vhDCRuler, rgbBkgrnd);
  273. SetTextColor(vhDCRuler, rgbText);
  274. /* Set the brush and the pen for the ruler. */
  275. SelectObject(vhDCRuler, hbrBkgrnd);
  276. if ((hpen = CreatePen(0, 0, rgbText)) == NULL)
  277. {
  278. hpen = GetStockObject(BLACK_PEN);
  279. }
  280. SelectObject(vhDCRuler, hpen);
  281. /* Lastly, ensure that the ruler is painted. */
  282. ShowWindow(vhWndRuler, SHOW_OPENWINDOW);
  283. UpdateWindow(vhWndRuler);
  284. return (TRUE);
  285. Error1:
  286. DestroyWindow(vhWndRuler);
  287. vhWndRuler = NULL;
  288. Error2:
  289. WinFailure();
  290. return (FALSE);
  291. }
  292. near DestroyRuler()
  293. {
  294. /* This routine destroys the ruler window and refreshes the screen. */
  295. /* First, erase the ruler. */
  296. PatBlt(vhDCRuler, 0, 0, dxpRuler, dypRuler, ropErase);
  297. /* Clean up the ruler window. */
  298. DestroyWindow(vhWndRuler);
  299. vhWndRuler = NULL;
  300. ResetRuler();
  301. /* Move the document window back to the top of the window. */
  302. pwwdCur->fRuler = FALSE;
  303. vfEraseWw = TRUE;
  304. MoveWindow(wwdCurrentDoc.wwptr, 0, 0, dxpRuler, wwdCurrentDoc.ypMac +
  305. dypRuler - (wwdCurrentDoc.ypMin - 1), FALSE);
  306. vfEraseWw = FALSE;
  307. /* Validate the area in the document window above the text. */
  308. PatBlt(wwdCurrentDoc.hDC, 0, 0, dxpRuler, wwdCurrentDoc.ypMin, ropErase);
  309. ValidateRect(hParentWw, (LPRECT)NULL);
  310. }
  311. UpdateRuler()
  312. {
  313. /* This routine will redraw as much of the ruler as necessary to reflect the
  314. current selection. */
  315. /* Only repaint the ruler if it exists and it is not currently being
  316. changed. */
  317. if (vhWndRuler != NULL)
  318. {
  319. RulerPaint(FALSE, FALSE, FALSE);
  320. }
  321. }
  322. ReframeRuler()
  323. {
  324. /* This routine will cause the ruler window to be redrawn,
  325. when units change - leave update out, since dialog box
  326. will repaint */
  327. /* Only repaint the ruler if it exists . */
  328. if (vhWndRuler != NULL)
  329. {
  330. InvalidateRect(vhWndRuler, (LPRECT)NULL, FALSE);
  331. }
  332. }
  333. ResetRuler()
  334. {
  335. /* Reset the values of the ruler buttons and the ruler margins and tabs so
  336. they redrawn during the next paint message. */
  337. if ((btnTabSave = mprlcbtnDown[rlcTAB]) == btnNIL)
  338. {
  339. btnTabSave = btnLTAB;
  340. }
  341. /* Reset the buttons. */
  342. if (vfIconic)
  343. {
  344. /* All we have to do is reset our internal state. */
  345. bltc(mprlcbtnDown, btnNIL, rlcBTNMAX);
  346. }
  347. else
  348. {
  349. /* We had best reset the buttons on the screen as well. */
  350. UpdateRulerBtn(rlcTAB, btnNIL);
  351. UpdateRulerBtn(rlcSPACE, btnNIL);
  352. UpdateRulerBtn(rlcJUST, btnNIL);
  353. }
  354. /* Reset the margins and the tabs. */
  355. bltc(mprmkdxa, -1, rmkMARGMAX);
  356. bltc(rgtbdRuler, 0, cwTBD * itbdMax);
  357. }
  358. ResetTabBtn()
  359. {
  360. /* This routine resets the tab button on the ruler to the left tab button.
  361. */
  362. if (mprlcbtnDown[rlcTAB] != btnLTAB)
  363. {
  364. UpdateRulerBtn(rlcTAB, btnLTAB);
  365. }
  366. }
  367. RulerPaint(fContentsOnly, fFrameOnly, fInit)
  368. BOOL fContentsOnly;
  369. BOOL fInit;
  370. {
  371. /* This routine draws the ruler in the ruler window. If fContentsOnly is
  372. set, then only the tabs as they currently exist in rgtbdRuler, and the
  373. button settings are drawn. If fFrameOnly is set, then only the ruler frame
  374. is redrawn. If fInit is set, then the portion of the ruler to be redrawn
  375. (tabs, frame or all) is redrawn from scratch. */
  376. int xpMin = pwwdCur->xpMin;
  377. HBITMAP hbm;
  378. /* If fContentsOnly is set, then skip most of this stuff and draw only the
  379. tabs and the button settings. */
  380. if (!fContentsOnly)
  381. {
  382. /* We only need to draw the physical ruler itself when the window has
  383. scrolled horizontally. */
  384. if (fInit || xpMinCur != xpMin)
  385. {
  386. register int xp;
  387. TEXTMETRIC tm;
  388. int dypTick;
  389. int ypTickEnd;
  390. int ypTickStart;
  391. int ypTick;
  392. int iLevel;
  393. CHAR rgchInch[3];
  394. int dxpLogUnitInc;
  395. int dcNextTick;
  396. int dxpLine;
  397. extern int utCur;
  398. #define cDivisionMax 8 /* max divisions per ruler unit. e.g. 8 per inch */
  399. int rgypTick[cDivisionMax];
  400. int cxpExtra;
  401. int cDivision;
  402. int dxpLogUnit;
  403. int dxpMeas;
  404. int ypT;
  405. /* Initialize the y-coordinate of the ticks. */
  406. GetTextMetrics(vhDCRuler, (LPTEXTMETRIC)&tm);
  407. ypTickEnd = dypRuler - dypMark - 2;
  408. ypTickStart = ypTick = ypTickEnd - (dypTick = tm.tmAscent -
  409. tm.tmInternalLeading);
  410. /* set up measurements for the ruler based on current unit -
  411. note that only inch and cm are handled in this version */
  412. if (utCur == utInch)
  413. {
  414. dxpLogUnit = dxpLogUnitInc = dxpLogInch;
  415. cDivision = 8; /* # of divisions */
  416. dxpMeas = dxpLogUnit >> 3; /* 1/8" units */
  417. /* get extra pixels to distribute if not even multiple */
  418. /* note - mod done by hand */
  419. cxpExtra = dxpLogUnit - (dxpMeas << 3);
  420. dcNextTick = 1;
  421. /* fill table of tick lengths */
  422. rgypTick[0] = ypT = ypTick;
  423. rgypTick[4] = ypT += (dypTick >> 2);
  424. rgypTick[2] = rgypTick[6] = ypT += (dypTick >> 2);
  425. rgypTick[1] = rgypTick[3] = rgypTick[5] = rgypTick[7] =
  426. ypT += (dypTick >> 2);
  427. }
  428. else
  429. /* default to cm */
  430. {
  431. dxpLogUnit = dxpLogUnitInc = dxpLogCm;
  432. cDivision = 2; /* # of divisions */
  433. dxpMeas = dxpLogUnit >> 1; /* 1/2 cm units */
  434. /* get extra pixels to distribute if not even multiple */
  435. cxpExtra = dxpLogUnit - (dxpMeas << 1);
  436. dcNextTick = 1;
  437. /* fill table of tick lengths */
  438. rgypTick[0] = ypTick;
  439. rgypTick[1] = ypTick + (dypTick >> 1);
  440. }
  441. if (fInit)
  442. {
  443. /* Erase the area where the ruler will be drawn. */
  444. PatBlt(vhDCRuler, 0, 0, dxpRuler, dypRuler, ropErase);
  445. /* Draw a line across the bottom of the ruler. */
  446. MoveTo(vhDCRuler, xpSelBar, dypRuler - 1);
  447. LineTo(vhDCRuler, dxpRuler, dypRuler - 1);
  448. /* Draw the base of the ruler. */
  449. MoveTo(vhDCRuler, xpSelBar, ypTickEnd);
  450. LineTo(vhDCRuler, dxpRuler, ypTickEnd);
  451. }
  452. else
  453. {
  454. /* Erase the old tick marks. */
  455. PatBlt(vhDCRuler, 0, ypTickStart, dxpRuler, ypTickEnd -
  456. ypTickStart, ropErase);
  457. }
  458. /* Set the clip region to be only the ruler. */
  459. iLevel = SaveDC(vhDCRuler);
  460. IntersectClipRect(vhDCRuler, xpSelBar, 0, dxpRuler, dypRuler);
  461. /* Draw the ticks at the each division mark. */
  462. /* iDivision is the current division with in a unit. It is
  463. used to determine when extra pixels are distributed and
  464. which tick mark to use */
  465. {
  466. register int iDivision = 0;
  467. for (xp = (xpSelBar - xpMin); xp < dxpRuler; xp +=
  468. dxpMeas)
  469. {
  470. /* distribute extra pixels at front */
  471. if (iDivision < cxpExtra)
  472. xp++;
  473. MoveTo(vhDCRuler, xp, rgypTick[iDivision]);
  474. LineTo(vhDCRuler, xp, ypTickEnd);
  475. if (++iDivision == cDivision)
  476. iDivision = 0;
  477. }
  478. }
  479. /* Label the tick marks. */
  480. dxpLine = GetSystemMetrics(SM_CXBORDER);
  481. rgchInch[0] = rgchInch[1] = rgchInch[2] = '0';
  482. for (xp = xpSelBar - xpMin;
  483. xp < dxpRuler;
  484. xp += dxpLogUnitInc, rgchInch[2] += dcNextTick)
  485. {
  486. int isz;
  487. int dxpsz;
  488. if (rgchInch[2] > '9')
  489. {
  490. rgchInch[1]++;
  491. rgchInch[2] = '0' + (rgchInch[2] - (CHAR) ('9' + 1));
  492. }
  493. if (rgchInch[1] > '9')
  494. {
  495. rgchInch[0]++;
  496. rgchInch[1] = '0' + (rgchInch[1] - (CHAR) ('9' + 1));
  497. }
  498. isz = rgchInch[0] == '0' ?
  499. (rgchInch[1] == '0' ? 2 : 1):
  500. 0;
  501. dxpsz = LOWORD(GetTextExtent(vhDCRuler,
  502. (LPSTR)&rgchInch[isz],
  503. 3 - isz));
  504. if (dxpsz + dxpLine >= dxpMeas)
  505. {
  506. PatBlt(vhDCRuler, xp + dxpLine, ypTickStart,
  507. dxpsz, ypTickEnd - ypTickStart, ropErase);
  508. }
  509. TextOut(vhDCRuler, xp + dxpLine, ypTickStart -
  510. tm.tmInternalLeading, (LPSTR)&rgchInch[isz],
  511. 3 - isz);
  512. }
  513. /* Set the clip region back. */
  514. RestoreDC(vhDCRuler, iLevel);
  515. }
  516. /* Draw the buttons on the ruler. */
  517. if (fInit)
  518. {
  519. register PRECT prc = &rgrcRulerBtn[btnMIN];
  520. int btn;
  521. /* Ensure that we have the bitmap for the buttons. */
  522. if (SelectObject(hMDCBitmap, hbmBtn) == NULL)
  523. {
  524. if (NULL == (hbmBtn = LoadBitmap(hMmwModInstance,
  525. MAKEINTRESOURCE(idBmBtns+viBmRuler)))
  526. || SelectObject(hMDCBitmap, hbmBtn) == NULL)
  527. {
  528. WinFailure();
  529. goto NoBtns;
  530. }
  531. }
  532. /* Now, draw the buttons. */
  533. for (btn = btnMIN; btn < btnMaxUsed; btn++)
  534. {
  535. int dxpBtn = prc->right - prc->left;
  536. BitBlt(vhDCRuler, prc->left, prc->top, dxpBtn, prc->bottom -
  537. prc->top, hMDCBitmap, (btn - btnMIN) * dxpBtn, 0, vfMonochrome
  538. ? MERGEMARK : SRCCOPY);
  539. prc++;
  540. }
  541. SelectObject(hMDCBitmap, hbmNullRuler);
  542. NoBtns:;
  543. }
  544. }
  545. /* If fFrame only is set, then we're finished. */
  546. if (!fFrameOnly)
  547. {
  548. /* Lastly, draw the button settings, the margins and the tabs. */
  549. TSV rgtsv[itsvparaMax];
  550. register struct TBD *ptbd1;
  551. int rmk;
  552. int xpMarkMin = xpSelBar - (dxpMark >> 1);
  553. int dxpMarkMax = dxpRuler - xpSelBar - (dxpMark >> 1);
  554. unsigned dxa;
  555. if (mprlcbtnDown[rlcTAB] == btnNIL)
  556. {
  557. /* Initalize the tab button to be left tab. */
  558. UpdateRulerBtn(rlcTAB, btnTabSave);
  559. }
  560. /* Now for the spacing and justification. */
  561. GetRgtsvPapSel(rgtsv);
  562. UpdateRulerBtn(rlcSPACE, (rgtsv[itsvSpacing].fGray != 0) ? btnNIL :
  563. (rgtsv[itsvSpacing].wTsv - czaLine) / (czaLine / 2) + btnSINGLE);
  564. UpdateRulerBtn(rlcJUST, (rgtsv[itsvJust].fGray != 0) ? btnNIL :
  565. (rgtsv[itsvJust].wTsv - jcLeft) + btnLEFT);
  566. /* The margins and the tabs are based off of the first cp of the
  567. selection. */
  568. CacheSect(docCur, selCur.cpFirst);
  569. CachePara(docCur, selCur.cpFirst);
  570. /* If the window has scrolled horizontally or become wider, we must
  571. redraw the margins and the tabs. */
  572. if (!fInit && xpMinCur == xpMin)
  573. {
  574. /* Compare to see if the margins have changed. */
  575. if (mprmkdxa[rmkINDENT] != vpapAbs.dxaLeft + vpapAbs.dxaLeft1)
  576. {
  577. goto DrawMargins;
  578. }
  579. if (mprmkdxa[rmkLMARG] != vpapAbs.dxaLeft)
  580. {
  581. goto DrawMargins;
  582. }
  583. if (mprmkdxa[rmkRMARG] != vsepAbs.dxaText - vpapAbs.dxaRight)
  584. {
  585. goto DrawMargins;
  586. }
  587. /* Compare to see if the tabs has changed. */
  588. {
  589. register struct TBD *ptbd2;
  590. for (ptbd1 = &rgtbdRuler[0], ptbd2 = &vpapAbs.rgtbd[0];
  591. ptbd1->dxa == ptbd2->dxa; ptbd1++, ptbd2++)
  592. {
  593. /* If the end of the list of tabs, then the lists are equal.
  594. */
  595. if (ptbd1->dxa == 0)
  596. {
  597. goto SkipTabs;
  598. }
  599. /* The justification codes must match if they are decimal
  600. tabs (everything else collaspes to left tabs). */
  601. if (ptbd1->jc != ptbd2->jc && (ptbd1->jc == (jcTabDecimal
  602. - jcTabMin) || (ptbd2->jc == (jcTabDecimal - jcTabMin))))
  603. {
  604. goto DrawMargins;
  605. }
  606. }
  607. }
  608. }
  609. DrawMargins:
  610. #ifdef KINTL
  611. /* This is really an extra. xpMinCur will get updated later on.
  612. But, we need this variable set up right for the MergeRulerMark()
  613. to draw a mark at the right place.... Oh well. */
  614. xpMinCur = xpMin;
  615. #endif /* ifdef KINTL */
  616. /* Redraw the margins from scratch. Set up the bitmap for hMDCScreen,
  617. the ruler bar in monochrome format. */
  618. if ((hbm = CreateBitmap(dxpRuler + dxpMark, dypMark, 1, 1,
  619. (LPSTR)NULL)) == NULL)
  620. {
  621. WinFailure();
  622. goto SkipTabs;
  623. }
  624. DeleteObject(SelectObject(hMDCScreen, hbm));
  625. PatBlt(hMDCScreen, 0, 0, dxpRuler + dxpMark, dypMark, vfMonochrome ?
  626. ropErase : WHITENESS);
  627. PatBlt(vhDCRuler, 0, dypRuler - dypMark - 1, dxpRuler + dxpMark,
  628. dypMark, ropErase);
  629. /* Determine the margin positions. */
  630. mprmkdxa[rmkINDENT] = vpapAbs.dxaLeft + vpapAbs.dxaLeft1;
  631. mprmkdxa[rmkLMARG] = vpapAbs.dxaLeft;
  632. mprmkdxa[rmkRMARG] = (vdxaTextRuler = vsepAbs.dxaText) -
  633. vpapAbs.dxaRight;
  634. /* Draw the margins marks. */
  635. for (rmk = rmkMARGMIN; rmk < rmkMARGMAX; rmk++)
  636. {
  637. register int dxp = MultDiv(mprmkdxa[rmk], dxpLogInch, czaInch) -
  638. xpMin;
  639. /* If the margin mark would not appear on the ruler, scrolled off to
  640. either end, then don't try to draw it. */
  641. if (dxp >= 0 && dxp < dxpMarkMax)
  642. {
  643. MergeRulerMark(rmk, xpMarkMin + dxp, FALSE);
  644. }
  645. }
  646. /* Redraw the tabs. */
  647. ptbd1 = &rgtbdRuler[0];
  648. if (!fInit)
  649. {
  650. /* If fInit is set, then rgtbdRuler is not changed. */
  651. blt(vpapAbs.rgtbd, ptbd1, cwTBD * itbdMax);
  652. }
  653. while ((dxa = ptbd1->dxa) != 0)
  654. {
  655. register int dxp = MultDiv(dxa, dxpLogInch, czaInch) - xpMin;
  656. /* If the tab mark would not appear on the ruler, scrolled off to
  657. either end, then don't try to draw it. */
  658. if (dxp >= 0 && dxp < dxpMarkMax)
  659. {
  660. MergeRulerMark(ptbd1->jc == (jcTabDecimal - jcTabMin) ? rmkDTAB
  661. : rmkLTAB, xpMarkMin + dxp, FALSE);
  662. }
  663. ptbd1++;
  664. }
  665. SkipTabs:;
  666. }
  667. /* Record the edges of the current window. */
  668. xpMinCur = xpMin;
  669. }
  670. RulerMouse(pt)
  671. POINT pt;
  672. {
  673. /* Process all mouse messages from a down-click at point pt until the
  674. corresponding mouse up-click. */
  675. int btn;
  676. int rlc;
  677. int rlcCur;
  678. int rmkCur;
  679. int xp;
  680. int xpCur;
  681. unsigned xa;
  682. struct TBD *ptbd;
  683. struct TBD tbd;
  684. BOOL fMarkMove = FALSE;
  685. BOOL fDeleteMark = FALSE;
  686. BOOL fBtnChanged = FALSE;
  687. if (!FWriteOk(fwcNil))
  688. {
  689. return;
  690. }
  691. /* Translate the point into a button group and a button. */
  692. RulerStateFromPt(pt, &rlcCur, &btn);
  693. /* Down clicking on the tab rule is a special case. */
  694. if (rlcCur == rlcRULER)
  695. {
  696. unsigned dxa = MultDiv(pt.x - xpSelBar + xpMinCur, czaInch, dxpLogInch);
  697. int rmk;
  698. int itbd;
  699. /* Have we moused down on a margin? */
  700. for (rmk = rmkMARGMIN; rmk < rmkMARGMAX; rmk++)
  701. {
  702. #ifdef KINTL
  703. if (FPointNear(mprmkdxa[rmk], dxa - XaKickBackXa(dxa)))
  704. #else
  705. if (FPointNear(mprmkdxa[rmk], dxa))
  706. #endif /* if-else-def KINTL */
  707. {
  708. int xpT;
  709. /* Remember this mark and its position. */
  710. rmkCur = rmk;
  711. xpCur = xpSelBar + MultDiv(mprmkdxa[rmk], dxpLogInch, czaInch) -
  712. (dxpMark >> 1) - xpMinCur;
  713. InvertMark:
  714. #ifdef KINTL
  715. /* Adjust for the kick-backs. */
  716. /* But don't modify the xpCur. */
  717. xpT = xpCur + XpKickBackXp(xpCur);
  718. #else
  719. xpT = xpCur;
  720. #endif /* if-else-def KINTL */
  721. /* Time to invert the selected mark. */
  722. PatBlt(vhDCRuler, xpT, dypRuler - dypMark - 1, dxpMark,
  723. dypMark, DSTINVERT);
  724. goto GotMark;
  725. }
  726. }
  727. /* Have we moused down on an existing tab? */
  728. for (itbd = 0, ptbd = &rgtbdRuler[0]; ; itbd++, ptbd++)
  729. {
  730. /* The end of the tabs have been found. */
  731. if (ptbd->dxa == 0)
  732. {
  733. break;
  734. }
  735. /* Have we moused down on this tab? */
  736. #ifdef KINTL
  737. if (FPointNear(ptbd->dxa, dxa - XaKickBackXa(dxa)))
  738. #else
  739. if (FPointNear(ptbd->dxa, dxa))
  740. #endif /* if-else-def KANJI */
  741. {
  742. /* Save this tab descriptor and its location. */
  743. tbd = *ptbd;
  744. rmkCur = (tbd.jc + jcTabMin) == jcTabDecimal ? rmkDTAB :
  745. rmkLTAB;
  746. xpCur = xpSelBar + MultDiv(tbd.dxa, dxpLogInch, czaInch) -
  747. (dxpMark >> 1) - xpMinCur;
  748. goto InvertMark;
  749. }
  750. }
  751. /* If one more tab would be too many, then beep and return. */
  752. if (itbd >= itbdMax - 1)
  753. {
  754. _beep();
  755. return;
  756. }
  757. /* Create a tab descriptor for this new tab. */
  758. bltc(&tbd, 0, cwTBD);
  759. tbd.dxa = XaQuantize(pt.x);
  760. tbd.jc = (mprlcbtnDown[rlcTAB] == btnLTAB ? jcTabLeft : jcTabDecimal) -
  761. jcTabMin;
  762. rmkCur = (mprlcbtnDown[rlcTAB] - btnLTAB) + rmkLTAB;
  763. /* A mark for the new tab needs to be drawn. */
  764. MergeRulerMark(rmkCur, xpCur = xpSelBar + MultDiv(tbd.dxa, dxpLogInch,
  765. czaInch) - (dxpMark >> 1) - xpMinCur, TRUE);
  766. /* Inserting a tab is like moving an existing tab. */
  767. fMarkMove = TRUE;
  768. GotMark:;
  769. #ifdef RULERALSO
  770. /* Update dialog box */
  771. if (vhDlgIndent && rmkCur < rmkMARGMAX)
  772. {
  773. SetIndentText(rmkCur, dxa);
  774. }
  775. #endif /* RULERALSO */
  776. }
  777. else if (rlcCur != rlcNIL)
  778. {
  779. /* Otherwise, if a button has been selected, the reflect the change on
  780. the ruler. */
  781. UpdateRulerBtn(rlcCur, btn);
  782. }
  783. else
  784. {
  785. /* The user has moused down on nothing of importance. */
  786. return;
  787. }
  788. /* Get all of the mouse events until further notice. */
  789. SetCapture(vhWndRuler);
  790. /* Process all of the mouse move messages. */
  791. while (FStillDown(&pt))
  792. {
  793. /* Movement on the tab ruler must be handled special. */
  794. if (rlcCur == rlcRULER)
  795. {
  796. #ifdef KINTL
  797. unsigned xaT;
  798. #endif /* ifdef KINTL */
  799. /* Guarantee that xp is in the range xpSelBar <= xp <= dxpRuler. */
  800. if ((xp = pt.x) > dxpRuler)
  801. {
  802. xp = dxpRuler;
  803. }
  804. else if (xp < xpSelBar)
  805. {
  806. xp = xpSelBar;
  807. }
  808. /* Convert the mouse position to twips. */
  809. #ifdef KINTL
  810. if ((xa = XaQuantize(xp)) > (xaT = XaQuantizeXa(vdxaTextRuler))
  811. #else
  812. if ((xa = XaQuantize(xp)) > vdxaTextRuler
  813. #endif /* if-else-def KINTL */
  814. && rmkCur < rmkMARGMAX)
  815. {
  816. /* Margins are confined to the page. */
  817. #ifdef KINTL
  818. xa = xaT;
  819. #else
  820. xa = vdxaTextRuler;
  821. #endif
  822. }
  823. /* If the cursor is on the ruler, then we may move a tab, but we
  824. always move the margins. */
  825. if ((rmkCur < rmkMARGMAX) || (pt.y >= 0 && pt.y < dypRuler + dypMark
  826. && xa != 0))
  827. {
  828. /* If the current mark has not moved, then there is nothing to
  829. do. */
  830. if (fDeleteMark || xa != XaQuantize(xpCur + (dxpMark >> 1)))
  831. {
  832. /* Indicate that the mark has moved. */
  833. fMarkMove = TRUE;
  834. /* Restore the screen under the current mark. */
  835. if (!fDeleteMark)
  836. {
  837. MergeRulerMark(rmkCur, xpCur, FALSE);
  838. }
  839. /* Draw the mark at the new location. */
  840. MergeRulerMark(rmkCur, xpCur = MultDiv(xa, dxpLogInch,
  841. czaInch) + xpSelBar - xpMinCur - (dxpMark >> 1), TRUE);
  842. /* Show this is a valid mark. */
  843. fDeleteMark = FALSE;
  844. #ifdef RULERALSO
  845. /* Update dialog box */
  846. if (vhDlgIndent && rmkCur < rmkMARGMAX)
  847. {
  848. SetIndentText(rmkCur, xa);
  849. }
  850. #endif /* RULERALSO */
  851. }
  852. }
  853. else
  854. {
  855. /* Restore the screen under the current mark. */
  856. if (!fDeleteMark)
  857. {
  858. MergeRulerMark(rmkCur, xpCur, FALSE);
  859. }
  860. /* This mark is being deleted. */
  861. fDeleteMark = TRUE;
  862. }
  863. }
  864. else
  865. {
  866. /* If the mouse is on a button within the same button group, then
  867. reflect the change. */
  868. RulerStateFromPt(pt, &rlc, &btn);
  869. if (rlc == rlcCur)
  870. {
  871. UpdateRulerBtn(rlc, btn);
  872. }
  873. }
  874. }
  875. /* We are capturing all mouse events; we can now release them. */
  876. ReleaseCapture();
  877. /* Up-clicking on the tab ruler is a special case. */
  878. if (rlcCur == rlcRULER)
  879. {
  880. if (!fDeleteMark)
  881. {
  882. /* Restore the screen under the current mark. */
  883. MergeRulerMark(rmkCur, xpCur, FALSE);
  884. }
  885. if (fMarkMove)
  886. {
  887. /* Guarantee that xp is in the range xpSelBar <= xp <= dxpRuler. */
  888. if ((xp = pt.x) > dxpRuler)
  889. {
  890. xp = dxpRuler;
  891. }
  892. else if (xp < xpSelBar)
  893. {
  894. xp = xpSelBar;
  895. }
  896. }
  897. else
  898. {
  899. xp = xpCur + (dxpMark >> 1);
  900. }
  901. /* Convert the mouse position to twips. */
  902. if ((xa = XaQuantize(xp)) > vdxaTextRuler && rmkCur < rmkMARGMAX)
  903. {
  904. /* Margins are confined to the page. */
  905. xa = vdxaTextRuler;
  906. }
  907. /* If the cursor is on the ruler then we may insert/move a tab, but we
  908. always move the margins. */
  909. if ((rmkCur < rmkMARGMAX) || (pt.y >= 0 && pt.y < dypRuler + dypMark &&
  910. xa != 0))
  911. {
  912. /* Draw the mark at the new location. */
  913. MergeRulerMark(rmkCur, MultDiv(xa, dxpLogInch, czaInch) + xpSelBar -
  914. xpMinCur - (dxpMark >> 1), FALSE);
  915. /* We are moving one of the margins. */
  916. if (rmkCur < rmkMARGMAX)
  917. {
  918. if (vfMargChanged = mprmkdxa[rmkCur] != xa)
  919. {
  920. mprmkdxa[rmkCur] = xa;
  921. }
  922. #ifdef RULERALSO
  923. /* Update dialog box */
  924. if (vhDlgIndent)
  925. {
  926. SetIndentText(rmkCur, xa);
  927. }
  928. #endif /* RULERALSO */
  929. }
  930. /* It is a tab we are inserting/deleting. */
  931. else
  932. {
  933. tbd.dxa = xa;
  934. /* Is this a new tab? */
  935. if (ptbd->dxa == 0)
  936. {
  937. /* Insert the new tab. */
  938. InsertRulerTab(&tbd);
  939. }
  940. /* We are moving a tab; if it hasn't really moved, then do
  941. nothing. */
  942. else if (!FCloseXa(ptbd->dxa, xa))
  943. {
  944. DeleteRulerTab(ptbd);
  945. InsertRulerTab(&tbd);
  946. }
  947. }
  948. }
  949. /* We are deleting the tab; if its a new, there's nothing to do. */
  950. else if (ptbd->dxa != 0)
  951. {
  952. DeleteRulerTab(ptbd);
  953. }
  954. }
  955. else
  956. {
  957. /* If the mouse is on a button within the same button group, then
  958. reflect the change. */
  959. int btnT;
  960. RulerStateFromPt(pt, &rlc, &btnT);
  961. if (rlc == rlcCur)
  962. {
  963. UpdateRulerBtn(rlc, btn = btnT);
  964. }
  965. fBtnChanged = btn != mprlcbtnDown[btn];
  966. }
  967. /* Do the format only if a button changed */
  968. if ((fBtnChanged && rlcCur != rlcTAB) || vfMargChanged || vfTabsChanged)
  969. {
  970. struct SEL selSave;
  971. typeCP dcp;
  972. typeCP dcp2;
  973. CHAR rgb[1 + cchINT];
  974. CHAR *pch;
  975. int sprm;
  976. int val;
  977. struct TBD (**hgtbd)[];
  978. /* Set the selection to cover all of the paragraphs selected. */
  979. ExpandCurSel(&selSave);
  980. dcp2 = (dcp = selCur.cpLim - selCur.cpFirst) - (selCur.cpLim > cpMacCur
  981. ? ccpEol : 0);
  982. SetUndo(uacRulerChange, docCur, selCur.cpFirst, (rlcCur != rlcRULER ||
  983. rmkCur < rmkMARGMAX) ? dcp : dcp2, docNil, cpNil, dcp2, 0);
  984. /* Set the sprm and it's value for the ruler change. */
  985. switch (rlcCur)
  986. {
  987. case rlcSPACE:
  988. sprm = sprmPDyaLine;
  989. val = (mprlcbtnDown[rlcSPACE] - btnSINGLE) * (czaLine / 2) +
  990. czaLine;
  991. break;
  992. case rlcJUST:
  993. sprm = sprmPJc;
  994. val = mprlcbtnDown[rlcJUST] - btnLEFT + jcLeft;
  995. break;
  996. case rlcRULER:
  997. switch (rmkCur)
  998. {
  999. case rmkINDENT:
  1000. sprm = sprmPFIndent;
  1001. val = mprmkdxa[rmkINDENT] - mprmkdxa[rmkLMARG];
  1002. break;
  1003. case rmkLMARG:
  1004. /* Changing the left margin changes the first indent as well.
  1005. First, the indent... */
  1006. val = mprmkdxa[rmkINDENT] - mprmkdxa[rmkLMARG];
  1007. pch = &rgb[0];
  1008. *pch++ = sprmPFIndent;
  1009. bltbyte(&val, pch, cchINT);
  1010. AddOneSprm(rgb, FALSE);
  1011. /* Now for the left margin... */
  1012. sprm = sprmPLMarg;
  1013. val = mprmkdxa[rmkLMARG];
  1014. break;
  1015. case rmkRMARG:
  1016. sprm = sprmPRMarg;
  1017. val = vdxaTextRuler - mprmkdxa[rmkRMARG];
  1018. break;
  1019. case rmkLTAB:
  1020. case rmkDTAB:
  1021. /* Tabs are different. The change is made by blting the new tab
  1022. table on top of the old. */
  1023. vfTabsChanged = FALSE;
  1024. if ((hgtbd = (**hpdocdod)[docCur].hgtbd) == NULL)
  1025. {
  1026. if (FNoHeap(hgtbd = (struct TBD (**)[])HAllocate(itbdMax *
  1027. cwTBD)))
  1028. {
  1029. return;
  1030. }
  1031. (**hpdocdod)[docCur].hgtbd = hgtbd;
  1032. }
  1033. blt(rgtbdRuler, *hgtbd, itbdMax * cwTBD);
  1034. /* Changing the tabs makes everything dirty. */
  1035. (**hpdocdod)[docCur].fDirty = TRUE;
  1036. vdocParaCache = docNil;
  1037. TrashAllWws();
  1038. goto ChangeMade;
  1039. }
  1040. /* Indicate that the margins have been set. */
  1041. vfMargChanged = FALSE;
  1042. }
  1043. /* Now, lets set the sprm to the new value. */
  1044. pch = &rgb[0];
  1045. *pch++ = sprm;
  1046. bltbyte(&val, pch, cchINT);
  1047. AddOneSprm(rgb, FALSE);
  1048. ChangeMade:
  1049. /* Reset the selection to it's old value. */
  1050. EndLookSel(&selSave, TRUE);
  1051. }
  1052. }
  1053. near RulerStateFromPt(pt, prlc, pbtn)
  1054. POINT pt;
  1055. int *prlc;
  1056. int *pbtn;
  1057. {
  1058. /* This routine return in *prlc and *pbtn, the button group and the button
  1059. at point pt. The only button in group rlcRULER is btnNIL. */
  1060. int btn;
  1061. /* First check if the point is in a button. */
  1062. for (btn = btnMIN; btn < btnMaxUsed; btn++)
  1063. {
  1064. if (PtInRect((LPRECT)&rgrcRulerBtn[btn], pt))
  1065. {
  1066. goto ButtonFound;
  1067. }
  1068. }
  1069. /* The point is either on the tab ruler or nowhere of any interest. */
  1070. *prlc = (pt.y >= dypRuler - dypMark - 2 && pt.x > xpSelBar - (dxpMark >> 1)
  1071. && pt.x < dxpRuler + (dxpMark >> 1)) ? rlcRULER : rlcNIL;
  1072. *pbtn = btnNIL;
  1073. return;
  1074. ButtonFound:
  1075. /* The point is in a button, we just have to decide which button group. */
  1076. switch (btn)
  1077. {
  1078. case btnLTAB:
  1079. case btnDTAB:
  1080. *prlc = rlcTAB;
  1081. break;
  1082. case btnSINGLE:
  1083. case btnSP15:
  1084. case btnDOUBLE:
  1085. *prlc = rlcSPACE;
  1086. break;
  1087. case btnLEFT:
  1088. case btnCENTER:
  1089. case btnRIGHT:
  1090. case btnJUST:
  1091. *prlc = rlcJUST;
  1092. break;
  1093. }
  1094. *pbtn = btn;
  1095. }
  1096. void near HighlightButton(fOn, btn)
  1097. BOOL fOn; /* true if we should highlight this button, false = unhighlight */
  1098. int btn;
  1099. {
  1100. register PRECT prc = &rgrcRulerBtn[btn];
  1101. int dxpBtn = prc->right - prc->left;
  1102. /* If we're highlighting, then get the black-on-white button from
  1103. the right group; otherwise copy the white-on-black button ..pt */
  1104. int btnFromBM = btn - btnMIN + (fOn ? btnMaxReal : 0);
  1105. /* Ensure that we have the bitmap for the buttons. */
  1106. if (SelectObject(hMDCBitmap, hbmBtn) == NULL)
  1107. {
  1108. if ((hbmBtn = LoadBitmap(hMmwModInstance, MAKEINTRESOURCE(idBmBtns+viBmRuler))) ==
  1109. NULL || SelectObject(hMDCBitmap, hbmBtn) == NULL)
  1110. {
  1111. WinFailure();
  1112. goto NoBtns;
  1113. }
  1114. }
  1115. BitBlt(vhDCRuler, prc->left, prc->top, dxpBtn, prc->bottom - prc->top,
  1116. hMDCBitmap, btnFromBM * dxpBtn, 0, SRCCOPY);
  1117. SelectObject(hMDCBitmap, hbmNullRuler);
  1118. NoBtns:;
  1119. }
  1120. near UpdateRulerBtn(rlc, btn)
  1121. int rlc;
  1122. int btn;
  1123. {
  1124. /* This routine turns off the currently selected button in button group rlc
  1125. and turns on button btn. It is assumed that rlc is neither rlcNIL nor
  1126. rlcRULER, since neither group has buttons to update. */
  1127. int *pbtnOld = &mprlcbtnDown[rlc];
  1128. int btnOld = *pbtnOld;
  1129. Assert(rlc != rlcNIL && rlc != rlcRULER);
  1130. /* If the button hasn't changed, then there is nothing to do. */
  1131. if (btn != btnOld)
  1132. {
  1133. if (vhDCRuler != NULL)
  1134. {
  1135. /* Invert the old button (back to normal), and then invert the new
  1136. button. */
  1137. if (btnOld != btnNIL)
  1138. {
  1139. /* If there is no old button, then, of course, we can't invert
  1140. it. */
  1141. HighlightButton(fFalse, btnOld);
  1142. }
  1143. if (btn != btnNIL)
  1144. {
  1145. /* If the new button is not btnNIL, then invert it. */
  1146. HighlightButton(fTrue, btn);
  1147. }
  1148. }
  1149. /* Record whic button is now set. */
  1150. *pbtnOld = btn;
  1151. }
  1152. }
  1153. #ifdef KINTL
  1154. /* Given xa for a mouse position in a ruler, return the amount of xa for
  1155. a display adjustment. */
  1156. unsigned near XaKickBackXa(xa)
  1157. unsigned xa;
  1158. {
  1159. extern int utCur;
  1160. extern int dxaAdjustPerCm;
  1161. int cCm, cCh;
  1162. switch (utCur) {
  1163. case utCm:
  1164. cCm = xa / czaCm;
  1165. return (dxaAdjustPerCm * cCm);
  1166. case utInch:
  1167. return (0);
  1168. default:
  1169. Assert(FALSE);
  1170. return (0);
  1171. }
  1172. }
  1173. near XpKickBackXp(xp)
  1174. int xp;
  1175. {
  1176. /* Computes the amount of a necessary kick-back in xp, if
  1177. a ruler marker is to be drawn at a given xp. */
  1178. extern int utCur;
  1179. extern int dxaAdjustPerCm;
  1180. int cCm, cCh;
  1181. switch (utCur) {
  1182. case utInch:
  1183. return 0;
  1184. case utCm:
  1185. /* For every cm, we are off by dxaAdjustPerCm twips. */
  1186. cCm = (xp - xpSelBar + xpMinCur + (dxpMark >> 1)) / dxpLogCm;
  1187. return (MultDiv(dxaAdjustPerCm * cCm, dxpLogInch, czaInch));
  1188. default:
  1189. Assert(FALSE);
  1190. return 0;
  1191. }
  1192. }
  1193. #endif /* ifdef KINTL */
  1194. near MergeRulerMark(rmk, xpMark, fHighlight)
  1195. int rmk;
  1196. int xpMark;
  1197. BOOL fHighlight;
  1198. {
  1199. /* This routine merges the ruler mark, rmk, with the contents of the ruler
  1200. bar at xpMark. To accomodate color, the merging of the mark with the
  1201. background must be done first in a monochrome memory bitmap, then converted
  1202. back to color. The mark is highlighed if fHighlight is set. */
  1203. int ypMark = dypRuler - dypMark - 1;
  1204. /* Ensure that we have the bitmap for the ruler marks. */
  1205. if (SelectObject(hMDCBitmap, hbmMark) == NULL)
  1206. {
  1207. if ((hbmMark = LoadBitmap(hMmwModInstance, MAKEINTRESOURCE(idBmMarks+viBmRuler))) == NULL
  1208. || SelectObject(hMDCBitmap, hbmMark) == NULL)
  1209. {
  1210. WinFailure();
  1211. return;
  1212. }
  1213. }
  1214. #ifdef KINTL
  1215. /* Adjust for the kick back */
  1216. xpMark += XpKickBackXp(xpMark);
  1217. #endif /* ifdef KINTL */
  1218. /* Merge the mark into the monochrome bitmap. */
  1219. BitBlt(hMDCScreen, xpMark, 0, dxpMark, dypMark, hMDCBitmap, (rmk - rmkMIN) *
  1220. dxpMark, 0, MERGEMARK);
  1221. /* Display the bitmap on the ruler bar. */
  1222. BitBlt(vhDCRuler, xpMark, ypMark, dxpMark, dypMark, hMDCScreen, xpMark, 0,
  1223. fHighlight ? NOTSRCCOPY : SRCCOPY);
  1224. SelectObject(hMDCBitmap, hbmNullRuler);
  1225. }
  1226. BOOL near FPointNear(xaTarget, xaProbe)
  1227. unsigned xaTarget;
  1228. unsigned xaProbe;
  1229. {
  1230. /* This routine returns TRUE if and only if xaProbe is sufficiently close to
  1231. xaTarget for selection purposes. */
  1232. int dxa;
  1233. if ((dxa = xaTarget - xaProbe) < 0)
  1234. {
  1235. dxa = -dxa;
  1236. }
  1237. return (dxa < MultDiv(dxpMark, czaInch, dxpLogInch) >> 1);
  1238. }
  1239. unsigned near XaQuantize(xp)
  1240. int xp;
  1241. {
  1242. #ifdef KINTL
  1243. /* This routine converts an x-coordinate from the ruler to twips
  1244. rounding it to the nearest sixteenth of an inch if utCur = utInch,
  1245. or to the nearest eighth of a centimeter if utCur = utCm. */
  1246. unsigned xa = MultDiv(xp - xpSelBar + xpMinCur, czaInch, dxpLogInch);
  1247. return (XaQuantizeXa(xa));
  1248. #else
  1249. /* This routine converts an x-coordinate from the ruler to twips rounding it
  1250. to the nearest sixteenth of an inch. */
  1251. unsigned xa = MultDiv(xp - xpSelBar + xpMinCur, czaInch, dxpLogInch);
  1252. /* NOTE: This code has been simplified because we "know" czaInch is a
  1253. multiple of 32. */
  1254. return ((xa + czaInch / 32) / (czaInch / 16) * (czaInch / 16));
  1255. #endif /* not KINTL */
  1256. }
  1257. #ifdef KINTL
  1258. unsigned near XaQuantizeXa(xa)
  1259. unsigned xa;
  1260. {
  1261. extern int utCur;
  1262. long xaL;
  1263. switch (utCur) {
  1264. case utInch:
  1265. /* NOTE: This code has been simplified because we "know" czaInch is a
  1266. multiple of 32. */
  1267. return ((xa + czaInch / 32) / (czaInch / 16) * (czaInch / 16));
  1268. case utCm:
  1269. /* NOTE: Actually, we are calculating:
  1270. (xa + czaCm / 16) / (czaCm / 8) * (czaCm / 8)
  1271. but calculated in 16*twips, so that there will
  1272. be the least rounding error. */
  1273. xaL = ((long) xa) << 4;
  1274. xaL = (xaL + czaCm) / (czaCm << 1) * (czaCm << 1);
  1275. /* Kick back is adjusted in MergeRulerMark. */
  1276. return ((unsigned) (xaL >> 4));
  1277. default:
  1278. Assert(FALSE);
  1279. return (xa); /* Heck, it's better than nothing. */
  1280. }
  1281. }
  1282. #endif /* KINTL */
  1283. near DeleteRulerTab(ptbd)
  1284. struct TBD *ptbd;
  1285. {
  1286. /* This routine removes the tab at ptbd from its table. */
  1287. vfTabsChanged = TRUE;
  1288. do
  1289. {
  1290. *ptbd = *(ptbd + 1);
  1291. }
  1292. while ((ptbd++)->dxa != 0);
  1293. }
  1294. near InsertRulerTab(ptbd)
  1295. struct TBD *ptbd;
  1296. {
  1297. /* This routine inserts the tab *ptbd into rgtbdRuler unless there is one
  1298. close to it already. */
  1299. register struct TBD *ptbdT;
  1300. unsigned dxa = ptbd->dxa;
  1301. unsigned dxaT;
  1302. /* Search the table for a tab that is close to the tab to be inserted. */
  1303. for (ptbdT = &rgtbdRuler[0]; ptbdT->dxa != 0; ptbdT++)
  1304. {
  1305. if (FCloseXa(ptbdT->dxa, dxa))
  1306. {
  1307. /* Overwrite the old tab iff the tab has changed. */
  1308. if (ptbdT->jc != ptbd->jc)
  1309. {
  1310. *ptbdT = *ptbd;
  1311. vfTabsChanged = TRUE;
  1312. }
  1313. /* Clean up the ruler and exit. */
  1314. RulerPaint(TRUE, FALSE, TRUE);
  1315. return;
  1316. }
  1317. }
  1318. vfTabsChanged = TRUE;
  1319. /* Insert the tab at the correctly sorted place. */
  1320. for (ptbdT = &rgtbdRuler[0]; (dxaT = ptbdT->dxa) != 0; ptbdT++)
  1321. {
  1322. if (dxa <= dxaT)
  1323. {
  1324. /* Insert the tab in front of ptbdT and move the remaining tabs up
  1325. one slot. The last tab will be overwritten to avoid table overflow.
  1326. */
  1327. blt(ptbdT, ptbdT + 1, ((&rgtbdRuler[0] - ptbdT) + (itbdMax - 2)) *
  1328. cwTBD);
  1329. *ptbdT = *ptbd;
  1330. return;
  1331. }
  1332. }
  1333. /* Insert the tab at the end of the table unless the table is full. */
  1334. if (ptbdT - &rgtbdRuler[0] < itbdMax - 1)
  1335. {
  1336. *ptbdT = *ptbd;
  1337. (ptbdT + 1)->dxa = 0;
  1338. }
  1339. }
  1340. BOOL near FCloseXa(xa1, xa2)
  1341. unsigned xa1;
  1342. unsigned xa2;
  1343. {
  1344. #ifdef KINTL
  1345. /* This function returns TRUE if xa1 is "close" to xa2;
  1346. FALSE otherwise. Threshold is determined by utCur. */
  1347. int dxa;
  1348. int dxaThreshold;
  1349. extern int utCur;
  1350. if ((dxa = xa1 - xa2) < 0)
  1351. {
  1352. dxa = -dxa;
  1353. }
  1354. switch (utCur) {
  1355. case utInch:
  1356. dxaThreshold = czaInch / 16;
  1357. break;
  1358. case utCm:
  1359. dxaThreshold = czaCm / 8;
  1360. break;
  1361. default:
  1362. Assert(FALSE);
  1363. dxaThreshold = 0; /* Heck. It doesn't matter at this point. */
  1364. break;
  1365. }
  1366. return (dxa < dxaThreshold);
  1367. #else /* not KINTL */
  1368. /* This function returns TRUE if xa1 is "close" to xa2; FALSE otherwise. */
  1369. int dxa;
  1370. if ((dxa = xa1 - xa2) < 0)
  1371. {
  1372. dxa = -dxa;
  1373. }
  1374. return (dxa < czaInch / 16);
  1375. #endif /* not KINTL */
  1376. }
  1377. #ifdef DEBUG
  1378. RulerMarquee()
  1379. {
  1380. /* This routine displays and scrolls the "marquee" message in the ruler mark
  1381. area. */
  1382. static CHAR szMarquee[] = "Dz}w|d`3Dazgv3{r`3qvv}3qa|ft{g3g|3j|f3qj3Q|q?3Q|q?3Qajr}?3P{z>P{fv}?3r}w3Crg";
  1383. LOGFONT lf;
  1384. HFONT hf;
  1385. HFONT hfOld;
  1386. /* Decode the marquee message. */
  1387. if (szMarquee[0] == 'D')
  1388. {
  1389. int ich;
  1390. for (ich = 0; ich < sizeof(szMarquee) - 1; ich++)
  1391. {
  1392. szMarquee[ich] ^= 0x13;
  1393. }
  1394. }
  1395. /* Get a logical font that will fit in the ruler mark area. */
  1396. bltbc(&lf, 0, sizeof(LOGFONT));
  1397. lf.lfHeight = -dypMark;
  1398. lf.lfPitchAndFamily = FIXED_PITCH;
  1399. /* Can we create such a font. */
  1400. if ((hf = CreateFontIndirect(&lf)) != NULL)
  1401. {
  1402. if ((hfOld = SelectObject(vhDCRuler, hf)) != NULL)
  1403. {
  1404. int xp;
  1405. int yp = dypRuler - dypMark - 1;
  1406. int dxp = LOWORD(GetTextExtent(vhDCRuler, (LPSTR)szMarquee,
  1407. sizeof(szMarquee) - 1));
  1408. int dxpScroll = MultDiv(GetSystemMetrics(SM_CXSCREEN), dypMark,
  1409. 2048);
  1410. int iLevel;
  1411. TEXTMETRIC tm;
  1412. /* Erase what is in the ruler mark area. */
  1413. PatBlt(vhDCRuler, 0, yp, dxpRuler, dypMark, ropErase);
  1414. /* Scroll the marquee across the screen. */
  1415. iLevel = SaveDC(vhDCRuler);
  1416. IntersectClipRect(vhDCRuler, xpSelBar, yp, dxpRuler, dypRuler - 1);
  1417. GetTextMetrics(vhDCRuler, (LPTEXTMETRIC)&tm);
  1418. for (xp = dxpRuler; xp > xpSelBar - dxp; xp -= dxpScroll)
  1419. {
  1420. BitBlt(vhDCRuler, xp, yp, min(dxpRuler - (xp + dxpScroll), dxp),
  1421. dypMark, vhDCRuler, xp + dxpScroll, yp, SRCCOPY);
  1422. PatBlt(vhDCRuler, min(dxpRuler - dxpScroll, xp + dxp), yp,
  1423. dxpScroll, dypMark, ropErase);
  1424. if (xp + dxp >= dxpRuler)
  1425. {
  1426. int dxpch = (dxpRuler - xp) % tm.tmAveCharWidth;
  1427. int ich = (dxpRuler - xp) / tm.tmAveCharWidth;
  1428. if (dxpch == 0 && xp < dxpRuler)
  1429. {
  1430. dxpch = tm.tmAveCharWidth;
  1431. ich--;
  1432. }
  1433. TextOut(vhDCRuler, dxpRuler - dxpch, yp -
  1434. tm.tmInternalLeading, (LPSTR)&szMarquee[ich], 1);
  1435. }
  1436. }
  1437. RestoreDC(vhDCRuler, iLevel);
  1438. /* Cleanup the font and the screen. */
  1439. SelectObject(vhDCRuler, hfOld);
  1440. RulerPaint(TRUE, FALSE, TRUE);
  1441. }
  1442. DeleteObject(hf);
  1443. }
  1444. }
  1445. #endif
  1446.