XTPSyntaxEditCtrl.cpp
上传用户:szled88
上传日期:2015-04-09
资源大小:43957k
文件大小:195k
源码类别:

对话框与窗口

开发平台:

Visual C++

  1. CXTPFontDC fontDC(&dc, GetPaintManager()->GetFont());
  2. m_DrawTextProcessor.RecalcRowHeight(&dc, GetPaintManager()->GetFont());
  3. if (m_hWnd && ::IsWindow(m_hWnd))
  4. {
  5. //m_pToolTip->SetFont(GetPaintManager()->GetFontToolTip());
  6. CalculateEditbarLength();
  7. Invalidate(FALSE);
  8. _UpdateIMEStatus();
  9. }
  10. }
  11. void CXTPSyntaxEditCtrl::_UpdateIMEStatus()
  12. {
  13. m_bIMEsupported = m_ImmWrapper.ImmIsIME();
  14. if (!m_bIMEsupported || !m_hWnd || !::IsWindow(m_hWnd))
  15. return;
  16. // IME Support
  17. XTP_HIMC hIMC = m_ImmWrapper.ImmGetContext(m_hWnd);
  18. if (hIMC)
  19. {
  20. LOGFONT lfFont;
  21. ::ZeroMemory(&lfFont, sizeof(lfFont));
  22. BOOL bFont = GetPaintManager()->GetFont()->GetLogFont(&lfFont);
  23. ASSERT(bFont);
  24. if (bFont)
  25. {
  26. VERIFY(m_ImmWrapper.ImmSetCompositionFont(hIMC, &lfFont));
  27. }
  28. VERIFY(m_ImmWrapper.ImmReleaseContext(m_hWnd, hIMC));
  29. }
  30. }
  31. LRESULT CXTPSyntaxEditCtrl::OnGetFont(WPARAM wParam, LPARAM lParam)
  32. {
  33. UNREFERENCED_PARAMETER(wParam);
  34. UNREFERENCED_PARAMETER(lParam);
  35. return (LRESULT)GetPaintManager()->GetFont()->GetSafeHandle();
  36. }
  37. LRESULT CXTPSyntaxEditCtrl::OnSetFont(WPARAM wParam, LPARAM lParam)
  38. {
  39. HFONT hFont = (HFONT)wParam;
  40. LOGFONT lfFont;
  41. LOGFONT *pLF = NULL; // set the default font if hFont == NULL
  42. if (hFont)
  43. {
  44. ::ZeroMemory(&lfFont, sizeof(LOGFONT));
  45. if (!::GetObject(hFont, sizeof(LOGFONT), &lfFont))
  46. {
  47. ASSERT(FALSE);
  48. return 0;
  49. }
  50. pLF = &lfFont;
  51. }
  52. SetFontIndirect(pLF, FALSE);
  53. if (LOWORD(lParam))
  54. UpdateWindow();
  55. //Default();
  56. return 0;
  57. }
  58. LRESULT CXTPSyntaxEditCtrl::OnInputLanguage(WPARAM, LPARAM)
  59. {
  60. _UpdateIMEStatus();
  61. if (m_bIMEsupported)
  62. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  63. Default();
  64. return 1;
  65. }
  66. void CXTPSyntaxEditCtrl::SetDocModified(BOOL bModified)
  67. {
  68. XTP_EDIT_NMHDR_DOCMODIFIED dm;
  69. // NMHDR codes
  70. dm.nmhdr.code = XTP_EDIT_NM_SETDOCMODIFIED;
  71. dm.nmhdr.hwndFrom = m_hWnd;
  72. dm.nmhdr.idFrom = GetDlgCtrlID();
  73. // modified flag
  74. dm.bModified = bModified;
  75. // Notify the parent window
  76. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  77. {
  78. m_pParentWnd->SendMessage(
  79. WM_NOTIFY, (WPARAM)dm.nmhdr.idFrom, (LPARAM)&dm);
  80. }
  81. }
  82. void CXTPSyntaxEditCtrl::InsertString(LPCTSTR szText, int iRow, int iCol, BOOL bDeleteSelection)
  83. {
  84. if (!CanEditDoc())
  85. return;
  86. int nPrevRow = GetCurrentDocumentRow();
  87. if (nPrevRow != iRow)
  88. SetCurrentDocumentRow(iRow);
  89. m_nCurrentCol = iCol;
  90. if (bDeleteSelection)
  91. DeleteSelection();
  92. // At first determine what type of CRLF it has
  93. const int nTextLen = (int)_tcsclen(szText);
  94. if (nTextLen == 0)
  95. return;
  96. int nRowFrom = GetCurrentDocumentRow();
  97. int nColFrom = m_nCurrentCol;
  98. //**----------------------
  99. OnBeforeEditChanged(nRowFrom, nColFrom);
  100. //**----------------------
  101. if (GetAutoIndent() && m_nAutoIndentCol > 0)
  102. {
  103. CString strInsertText(
  104. CString(_T('t'), m_nInsertTabCount) +
  105. CString(_T(' '), m_nInsertSpaceCount));
  106. m_pBuffer->InsertText(strInsertText, GetCurrentDocumentRow(), m_nCurrentCol);
  107. m_nCurrentCol = m_nInsertTabCount + m_nInsertSpaceCount + 1;
  108. m_nAutoIndentCol = 0;
  109. }
  110. XTP_EDIT_LINECOL finalLC;
  111. BOOL bInsRes = m_pBuffer->InsertText(szText, GetCurrentDocumentRow(),
  112. m_nCurrentCol, TRUE, &finalLC);
  113. if (bDeleteSelection)
  114. m_pBuffer->GetUndoRedoManager()->ChainLastCommand();
  115. if (bInsRes)
  116. {
  117. SetCurrentDocumentRow(finalLC.nLine);
  118. m_nCurrentCol = finalLC.nCol;
  119. m_nDispCol = CalcDispCol(GetLineText(finalLC.nLine), m_nCurrentCol);
  120. }
  121. RecalcScrollBars();
  122. UINT nAction = XTP_EDIT_EDITACTION_MODIFYROW;
  123. int nCurDocRow = GetCurrentDocumentRow();
  124. if (nCurDocRow > nPrevRow)
  125. nAction |= XTP_EDIT_EDITACTION_INSERTROW;
  126. //**----------------------
  127. OnEditChanged(nRowFrom, nColFrom, nCurDocRow, m_nCurrentCol, xtpEditActInsert);
  128. //**----------------------
  129. NotifyEditChanged(nPrevRow, nCurDocRow, nAction);
  130. SetCurCaretPos(nCurDocRow, m_nDispCol, FALSE/*, FALSE*/);
  131. }
  132. void CXTPSyntaxEditCtrl::InsertTextBlock(LPCTSTR szText, int iRow, int iCol, BOOL bDeleteSelection)
  133. {
  134. if (!CanEditDoc())
  135. return;
  136. int nPrevRow = GetCurrentDocumentRow();
  137. if (nPrevRow != iRow)
  138. {
  139. SetCurrentDocumentRow(iRow);
  140. }
  141. m_nCurrentCol = iCol;
  142. if (bDeleteSelection)
  143. {
  144. DeleteSelection();
  145. }
  146. // At first determine what type of CRLF it has
  147. const int nTextLen = (int)_tcsclen(szText);
  148. if (nTextLen == 0)
  149. return;
  150. int nRowFrom = GetCurrentDocumentRow();
  151. int nColFrom = m_nCurrentCol;
  152. //**----------------------
  153. OnBeforeEditChanged(nRowFrom, nColFrom);
  154. //**----------------------
  155. XTP_EDIT_LINECOL finalLC;
  156. BOOL bInsRes = m_pBuffer->InsertTextBlock(szText, GetCurrentDocumentRow(),
  157. iCol, TRUE, &finalLC);
  158. UNREFERENCED_PARAMETER(bInsRes);
  159. if (bDeleteSelection)
  160. m_pBuffer->GetUndoRedoManager()->ChainLastCommand();
  161. RecalcScrollBars();
  162. UINT nAction = XTP_EDIT_EDITACTION_MODIFYROW;
  163. int nCurDocRow = GetCurrentDocumentRow();
  164. if (nCurDocRow > nPrevRow)
  165. nAction |= XTP_EDIT_EDITACTION_INSERTROW;
  166. //**----------------------
  167. OnEditChanged(nRowFrom, nColFrom, nCurDocRow, m_nCurrentCol, xtpEditActInsert);
  168. //**----------------------
  169. NotifyEditChanged(nPrevRow, nCurDocRow, nAction);
  170. SetCurCaretPos(nCurDocRow, m_nDispCol, FALSE/*, FALSE*/);
  171. }
  172. void CXTPSyntaxEditCtrl::UpdateScrollPos(DWORD dwUpdate/* = XTP_EDIT_UPDATE_ALL*/)
  173. {
  174. XTP_EDIT_NMHDR_SETSCROLLPOS ssp;
  175. // NMHDR codes
  176. ssp.nmhdr.code = XTP_EDIT_NM_UPDATESCROLLPOS;
  177. ssp.nmhdr.hwndFrom = m_hWnd;
  178. ssp.nmhdr.idFrom = GetDlgCtrlID();
  179. // Update flag
  180. ssp.dwUpdate = dwUpdate;
  181. // Notify the parent window
  182. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  183. {
  184. m_pParentWnd->SendMessage(
  185. WM_NOTIFY, (WPARAM)ssp.nmhdr.idFrom, (LPARAM)&ssp);
  186. }
  187. }
  188. void CXTPSyntaxEditCtrl::_EnableScrollBarNotify(DWORD dwScrollBar, DWORD dwState)
  189. {
  190. ASSERT(dwScrollBar != 0 && (dwScrollBar & (WS_VSCROLL|WS_HSCROLL)) != 0);
  191. XTP_EDIT_NMHDR_ENABLESCROLLBAR nmEScroll;
  192. // NMHDR codes
  193. nmEScroll.nmhdr.code = XTP_EDIT_NM_ENABLESCROLLBAR;
  194. nmEScroll.nmhdr.hwndFrom = m_hWnd;
  195. nmEScroll.nmhdr.idFrom = GetDlgCtrlID();
  196. // Update flag
  197. nmEScroll.dwScrollBar = dwScrollBar;
  198. nmEScroll.dwState = dwState;
  199. // Notify the parent window
  200. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  201. {
  202. m_pParentWnd->SendMessage(
  203. WM_NOTIFY, (WPARAM)nmEScroll.nmhdr.idFrom, (LPARAM)&nmEScroll);
  204. }
  205. }
  206. void CXTPSyntaxEditCtrl::SetOverwriteMode(BOOL bOverwriteMode)
  207. {
  208. if (bOverwriteMode != m_pBuffer->GetOverwriteFlag() )
  209. {
  210. m_pBuffer->SetOverwriteFlag(bOverwriteMode);
  211. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE/*, FALSE*/);
  212. }
  213. }
  214. BOOL CXTPSyntaxEditCtrl::OnEraseBkgnd(CDC* pDC)
  215. {
  216. UNREFERENCED_PARAMETER(pDC);
  217. return TRUE;
  218. }
  219. XTP_EDIT_LINECOL CXTPSyntaxEditCtrl::_SubtractSelSizeFromPos(XTP_EDIT_LINECOL lcDispPos)
  220. {
  221. XTP_EDIT_LINECOL lcStart = m_Selection.GetNormalStart_disp();
  222. XTP_EDIT_LINECOL lcEnd = m_Selection.GetNormalEnd_disp();
  223. if (lcDispPos <= lcStart)
  224. return lcDispPos;
  225. if (lcDispPos < lcEnd && !m_Selection.bBlockSelectionMode)
  226. {
  227. ASSERT(FALSE);
  228. lcDispPos = lcEnd;
  229. }
  230. if (m_Selection.bBlockSelectionMode)
  231. {
  232. if (lcDispPos.nLine >= lcStart.nLine && lcDispPos.nLine <= lcEnd.nLine)
  233. {
  234. if (lcDispPos.nCol >= lcEnd.nCol)
  235. lcDispPos.nCol -= lcEnd.nCol - lcStart.nCol;
  236. }
  237. }
  238. else
  239. {
  240. if (lcEnd.nLine != lcStart.nLine)
  241. lcDispPos.nLine -= lcEnd.nLine - lcStart.nLine;
  242. if (lcEnd.nLine == lcDispPos.nLine)
  243. {
  244. if (lcEnd.nLine == lcStart.nLine)
  245. {
  246. ASSERT(lcDispPos.nCol >= lcEnd.nCol);
  247. lcDispPos.nCol -= lcEnd.nCol - lcStart.nCol;
  248. }
  249. else
  250. {
  251. lcDispPos.nCol -= lcEnd.nCol;
  252. }
  253. }
  254. }
  255. return lcDispPos;
  256. }
  257. void CXTPSyntaxEditCtrl::CopyOrMoveText(BOOL bCopy)
  258. {
  259. if (!m_Selection.IsSelExist() || m_Selection.IsInSel_disp(m_ptDropPos.y, m_ptDropPos.x))
  260. return;
  261. CMemFile memfile(CalcAveDataSize(m_Selection.GetNormalStart_str().nLine,
  262.   m_Selection.GetNormalEnd_str().nLine));
  263. BOOL bRes = m_pBuffer->GetBuffer(m_Selection.GetNormalStart_disp(),
  264.  m_Selection.GetNormalEnd_disp(), memfile,
  265.  m_Selection.bBlockSelectionMode);
  266. if (!bRes)
  267. return;
  268. memfile.Write((const void *)_T(""), sizeof(TCHAR));
  269. BYTE *pBytes = memfile.Detach();
  270. CString strDropText = (LPTSTR)pBytes;
  271. free(pBytes);
  272. // First insert the text
  273. if (m_ptDropPos.y > GetRowCount())
  274. m_ptDropPos.y = GetRowCount();
  275. XTP_EDIT_LINECOL lcDropPos_disp = XTP_EDIT_LINECOL::MakeLineCol(m_ptDropPos.y, m_ptDropPos.x);
  276. BOOL bBlockSelectionMode = m_Selection.bBlockSelectionMode;
  277. if (!bCopy)
  278. {
  279. if (!(lcDropPos_disp < m_Selection.GetNormalStart_disp()))
  280. {
  281. lcDropPos_disp = _SubtractSelSizeFromPos(lcDropPos_disp);
  282. }
  283. DeleteSelection(); // reset current selection
  284. }
  285. CString strText;
  286. m_pBuffer->GetLineText(lcDropPos_disp.nLine, strText);
  287. int nAbsDropCol = CalcAbsCol(strText, lcDropPos_disp.nCol);
  288. //**----------------------
  289. OnBeforeEditChanged(lcDropPos_disp.nLine, nAbsDropCol);
  290. //**----------------------
  291. BOOL bReplace = GetTabWithSpace();
  292. SetTabWithSpace(FALSE);
  293. XTP_EDIT_LINECOL finalLC;
  294. BOOL bInsRes = FALSE;
  295. if (bBlockSelectionMode)
  296. {
  297. bInsRes = m_pBuffer->InsertTextBlock(strDropText, lcDropPos_disp.nLine, nAbsDropCol,
  298. TRUE, &finalLC);
  299. }
  300. else
  301. {
  302. bInsRes = m_pBuffer->InsertText(strDropText, lcDropPos_disp.nLine, nAbsDropCol,
  303. TRUE, &finalLC);
  304. }
  305. SetTabWithSpace(bReplace);
  306. if (!bCopy)
  307. {
  308. m_pBuffer->GetUndoRedoManager()->ChainLastCommand();
  309. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_MOVE);
  310. }
  311. else
  312. {
  313. // It's a drag copy
  314. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_COPY);
  315. }
  316. m_Selection.Reset_str(lcDropPos_disp.nLine, nAbsDropCol - 1);
  317. m_Selection.SetEnd_str(finalLC.nLine, finalLC.nCol - 1);
  318. m_Selection.bBlockSelectionMode = bBlockSelectionMode;
  319. int nRowFrom = m_Selection.GetNormalStart_str().nLine;
  320. int nRowTo = m_Selection.GetNormalEnd_str().nLine;
  321. //**----------------------
  322. OnEditChanged(nRowFrom, m_Selection.GetNormalStart_str().nCol + 1,
  323.   nRowTo, m_Selection.GetNormalEnd_str().nCol + 1, xtpEditActInsert);
  324. //**----------------------
  325. CalculateEditbarLength();
  326. SetCurCaretPos(m_Selection.GetEnd_disp().nLine, m_Selection.GetEnd_disp().nCol);
  327. SetDocModified();
  328. //**----------------------
  329. //  OnEditChanged(nRowFrom, m_Selection.GetNormalStart_str().nCol,
  330. //                nRowTo, m_Selection.GetNormalEnd_str().nCol, xtpEditActInsert);
  331. //**----------------------
  332. UINT nAction = (XTP_EDIT_EDITACTION_MODIFYROW | XTP_EDIT_EDITACTION_INSERTROW);
  333. NotifyEditChanged(nRowFrom, nRowTo, nAction);
  334. }
  335. void CXTPSyntaxEditCtrl::OnContextMenu(CWnd* pWnd, CPoint point)
  336. {
  337. if (GetCapture() == this)
  338. {
  339. ReleaseCapture();
  340. }
  341. if (m_bRightButtonDrag && m_bDragging && m_bDroppable)
  342. {
  343. HandleDrop(FALSE);
  344. }
  345. else
  346. {
  347. CXTPClientRect rcWnd(this);
  348. ClientToScreen(&rcWnd);
  349. m_bDragging = FALSE;
  350. if (m_pParentWnd && rcWnd.PtInRect(point))
  351. {
  352. CWnd::HideCaret();
  353. m_pParentWnd->SendMessage(WM_CONTEXTMENU, (WPARAM)pWnd->GetSafeHwnd(), (LPARAM)MAKELONG(point.x, point.y));
  354. CWnd::ShowCaret();
  355. Invalidate(FALSE);
  356. UpdateWindow();
  357. }
  358. }
  359. m_bRightButtonDrag = FALSE;
  360. }
  361. void CXTPSyntaxEditCtrl::OnDragCopy()
  362. {
  363. CopyOrMoveText(TRUE);
  364. m_bRightButtonDrag = FALSE;
  365. }
  366. void CXTPSyntaxEditCtrl::OnDragMove()
  367. {
  368. CopyOrMoveText(FALSE);
  369. m_bRightButtonDrag = FALSE;
  370. }
  371. void CXTPSyntaxEditCtrl::RestoreCursor()
  372. {
  373. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  374. // Invalidate(FALSE);
  375. }
  376. void CXTPSyntaxEditCtrl::SetRowBkColor(int nRow, COLORREF crBack)
  377. {
  378. if (nRow == -1)
  379. {
  380. m_mapRowBkColor.RemoveAll();
  381. return;
  382. }
  383. if (crBack == RGB(255,255,255) || crBack == COLORREF_NULL)
  384. m_mapRowBkColor.RemoveKey(nRow);
  385. else
  386. m_mapRowBkColor[nRow] = crBack;
  387. }
  388. void CXTPSyntaxEditCtrl::SetRowColor(int nRow, COLORREF clrFront)
  389. {
  390. if (nRow == -1)
  391. {
  392. m_mapRowColor.RemoveAll();
  393. return;
  394. }
  395. if (clrFront == RGB(0,0,0) || clrFront == COLORREF_NULL)
  396. m_mapRowColor.RemoveKey(nRow);
  397. else
  398. m_mapRowColor[nRow] = clrFront;
  399. }
  400. COLORREF CXTPSyntaxEditCtrl::GetRowColor(int nRow)
  401. {
  402. COLORREF clrFront;
  403. if (!m_mapRowColor.Lookup(nRow, clrFront))
  404. {
  405. clrFront = RGB(0,0,0);
  406. }
  407. return clrFront;
  408. }
  409. COLORREF CXTPSyntaxEditCtrl::GetRowBkColor(int nRow)
  410. {
  411. COLORREF crBack;
  412. if (!m_mapRowBkColor.Lookup(nRow, crBack))
  413. {
  414. crBack = RGB(255,255,255);
  415. }
  416. return crBack;
  417. }
  418. AFX_STATIC void CopyRowColorMap(CXTPSyntaxEditRowColorMap& dest, const CXTPSyntaxEditRowColorMap& src)
  419. {
  420. dest.RemoveAll();
  421. for (POSITION pos = src.GetStartPosition(); pos;)
  422. {
  423. int nRow;
  424. COLORREF crBack;
  425. src.GetNextAssoc(pos, nRow, crBack);
  426. dest[nRow] = crBack;
  427. }
  428. }
  429. const CXTPSyntaxEditCtrl& CXTPSyntaxEditCtrl::operator=(const CXTPSyntaxEditCtrl& src)
  430. {
  431. if (::IsWindow(src.m_hWnd))
  432. {
  433. SetCurrentDocumentRow(src.m_nCurrentDocumentRow);
  434. m_nCurrentCol = src.m_nCurrentCol;
  435. m_nDispCol    = src.m_nDispCol;
  436. CopyRowColorMap(m_mapRowBkColor, src.m_mapRowBkColor);
  437. CopyRowColorMap(m_mapRowColor, src.m_mapRowColor);
  438. SetSelMargin(src.m_bSelMargin);
  439. SetLineNumbers(src.m_bLineNumbers);
  440. SetCollapsibleNodes(src.m_bDrawNodes);
  441. CalculateEditbarLength();
  442. EnableWhiteSpace(src.m_bEnableWhiteSpace);
  443. SetTabWithSpace(src.m_bTabWithSpace);
  444. SetAutoIndent(src.m_bAutoIndent);
  445. SetSyntaxColor(src.m_bSyntaxColor);
  446. EnableVirtualSpace(src.m_bVirtualSpace);
  447. RecalcScrollBars();
  448. SetScrollPos(SB_HORZ, 0);
  449. m_DrawTextProcessor.SetScrollXOffset(0);
  450. }
  451. return *this;
  452. }
  453. CWnd * CXTPSyntaxEditCtrl::GetEffectiveParent() const
  454. {
  455. CWnd *pScrollWnd = GetParent();
  456. if (GetParent())
  457. {
  458. pScrollWnd = GetParent();
  459. if (!IsKindOf(RUNTIME_CLASS(CSplitterWnd)))
  460. {
  461. pScrollWnd = GetParent();
  462. }
  463. }
  464. else
  465. {
  466. pScrollWnd = const_cast<CXTPSyntaxEditCtrl *>(this);
  467. }
  468. return pScrollWnd;
  469. }
  470. DWORD CXTPSyntaxEditCtrl::GetAutoscrollDirection(int* pnCols, int* pnRows)
  471. {
  472. CPoint ptCursor;
  473. GetCursorPos(&ptCursor);
  474. ScreenToClient(&ptCursor);
  475. CRect rcText = m_DrawTextProcessor.GetTextRect();
  476. int nAutoZoneY = min(m_DrawTextProcessor.GetRowHeight(), rcText.Height()/4);
  477. int nAutoZoneX2 = min(m_DrawTextProcessor.GetTextMetrics().tmAveCharWidth, rcText.Width()/4);
  478. DWORD dwDirection = 0;
  479. int nScrollX = 0, nScrollY = 0;
  480. if (ptCursor.y < rcText.top + nAutoZoneY)
  481. {
  482. dwDirection += xtpTop;
  483. nScrollY = -1 - (rcText.top + nAutoZoneY - ptCursor.y) / (nAutoZoneY / 2);
  484. }
  485. if (ptCursor.y > rcText.bottom - nAutoZoneY)
  486. {
  487. dwDirection += xtpBottom;
  488. nScrollY = 1 + (ptCursor.y - rcText.bottom + nAutoZoneY) / (nAutoZoneY / 2);
  489. }
  490. if (ptCursor.x < rcText.left)
  491. {
  492. dwDirection += xtpLeft;
  493. nScrollX = -1 - (rcText.left - ptCursor.x) / (nAutoZoneX2 / 2);
  494. }
  495. if (ptCursor.x > rcText.right - nAutoZoneX2)
  496. {
  497. dwDirection += xtpRight;
  498. nScrollX = 1 + (ptCursor.x - rcText.right + nAutoZoneX2) / (nAutoZoneX2 / 2);
  499. }
  500. if (pnCols)
  501. *pnCols = nScrollX;
  502. if (pnRows)
  503. *pnRows = nScrollY;
  504. return dwDirection;
  505. }
  506. void CXTPSyntaxEditCtrl::OnTimer(UINT_PTR nIDEvent)
  507. {
  508. if (nIDEvent == TIMER_AUTOSCROLL_ID)
  509. {
  510. int nScrollX = 0, nScrollY = 0;
  511. m_dwAutoScrollDirection = GetAutoscrollDirection(&nScrollX, &nScrollY);
  512. if (nScrollX || nScrollY)
  513. {
  514. Scroll(nScrollX, nScrollY);
  515. OnTimer(TIMER_SELECTION_ID);
  516. //UpdateWindow();
  517. }
  518. BOOL bLButton = (::GetKeyState(VK_LBUTTON) & KF_UP) == KF_UP;
  519. if (!m_dwAutoScrollDirection || !bLButton)
  520. {
  521. m_dwAutoScrollDirection = 0;
  522. KillTimer(TIMER_AUTOSCROLL_ID);
  523. }
  524. }
  525. else if (nIDEvent == TIMER_SELECTION_ID)
  526. {
  527. if ((::GetKeyState(VK_LBUTTON) & KF_UP) == 0)
  528. {
  529. m_Selection.bSelectingRunning = FALSE;
  530. KillTimer(TIMER_SELECTION_ID);
  531. return;
  532. }
  533. CPoint ptCursor;
  534. GetCursorPos(&ptCursor);
  535. ScreenToClient(&ptCursor);
  536. BOOL bIsAutoScrollTimer = !!m_dwAutoScrollDirection;
  537. m_dwAutoScrollDirection = GetAutoscrollDirection();
  538. if (m_dwAutoScrollDirection)
  539. {
  540. if (!bIsAutoScrollTimer)
  541. SetTimer(TIMER_AUTOSCROLL_ID, TIMER_AUTOSCROLL_TIME, NULL);
  542. }
  543. else
  544. {
  545. m_dwAutoScrollDirection = 0;
  546. KillTimer(TIMER_AUTOSCROLL_ID);
  547. }
  548. //-------------------------------------------------------------------
  549. CRect rcBookmark, rcLineNum, rcNodes, rcText;
  550. CalcEditRects(&rcBookmark, &rcLineNum, &rcNodes, &rcText);
  551. CRect rcLineNumAndNodes(rcLineNum);
  552. rcLineNumAndNodes.right = rcNodes.right;
  553. int nNewRow = 0, nNewCol = 0, nNewDispRow = 0, nNewDispCol = 0;
  554. BOOL bVirtualSpace = m_Selection.IsbBlockSelectionMode() ? TRUE : -1;
  555. RowColFromPoint(ptCursor, &nNewRow, &nNewCol, &nNewDispRow, &nNewDispCol, bVirtualSpace);
  556. BOOL bSelFromLeftBar = rcLineNumAndNodes.PtInRect(ptCursor);
  557. if (bSelFromLeftBar && m_Selection.nSelStartTextRowFromLeftBar)
  558. {
  559. if (nNewRow >= m_Selection.nSelStartTextRowFromLeftBar)
  560. {
  561. nNewRow++;
  562. m_Selection.SetStart_str(m_Selection.nSelStartTextRowFromLeftBar, 0);
  563. }
  564. else
  565. {
  566. m_Selection.SetStart_str(m_Selection.nSelStartTextRowFromLeftBar+1, 0);
  567. }
  568. }
  569. int nEndRow_prev = m_Selection.GetEnd_disp().nLine;
  570. int nEndDispCol_prev = m_Selection.GetEnd_disp().nCol;
  571. if (nEndRow_prev != nNewRow || nEndDispCol_prev != nNewDispCol)
  572. {
  573. //          TRACE(_T("OnTimer.sel. RowColFromPoint: nNewRow=%d, nNewCol=%d, nNewDispRow=%d, nNewDispCol=%d n"),
  574. //                  nNewRow, nNewCol, nNewDispRow, nNewDispCol);
  575. m_Selection.SetEnd_disp(nNewRow, nNewDispCol);
  576. if (m_Selection.bWordSelectionMode)
  577. {
  578. XTP_EDIT_LINECOL lcWord1, lcWord2;
  579. BOOL bOverSpace = FALSE;
  580. UINT nFindDir = m_Selection.IsSelNormal() ? XTP_EDIT_FINDWORD_NEXT : XTP_EDIT_FINDWORD_PREV;
  581. BOOL bFind = FindWordEx_str(nFindDir, m_Selection.GetEnd_str(), lcWord1, lcWord2, bOverSpace);
  582. //TRACE(_T("2-FindWordEx_str = %d (%d,%d - %d,%d)"), bFind,
  583. //      lcWord1.nLine, lcWord1.nCol,lcWord2.nLine, lcWord2.nCol);
  584. XTP_EDIT_LINECOL& lcWordEnd = m_Selection.IsSelNormal() ? lcWord2 : lcWord1;
  585. lcWordEnd.nCol--;
  586. if (bFind && lcWordEnd.IsValidData() && lcWordEnd != m_Selection.GetEnd_str())
  587. {
  588. m_Selection.SetEnd_str(lcWordEnd.nLine, lcWordEnd.nCol);
  589. }
  590. }
  591. SetCurrentDocumentRow(nNewRow);
  592. m_nDispCol =  m_Selection.GetEnd_disp().nCol;
  593. m_nCurrentCol = m_Selection.GetEnd_str().nCol + 1;
  594. int nCaretLine = m_Selection.GetEnd_disp().nLine;
  595. if (bSelFromLeftBar && m_Selection.GetEnd_disp() >= m_Selection.GetStart_disp())
  596. nCaretLine = max(1, nCaretLine - 1);
  597. SetCurCaretPos(nCaretLine, m_nDispCol, FALSE, FALSE);
  598. //TRACE(_T("Selection: (row=%d, strPos=%d, DispCol=%d)-(row=%d, strPos=%d, DispCol=%d) n"),
  599. //      m_Selection.GetStart_str().nLine, m_Selection.GetStart_str().nCol, m_Selection.GetStart_disp().nCol,
  600. //      m_Selection.GetEnd_str().nLine, m_Selection.GetEnd_str().nCol, m_Selection.GetEnd_disp().nCol);
  601. Invalidate(FALSE);
  602. UpdateWindow();
  603. }
  604. }
  605. else if (nIDEvent == TIMER_REDRAW_WHEN_PARSE)
  606. {
  607. KillTimer(TIMER_REDRAW_WHEN_PARSE);
  608. Invalidate(FALSE);
  609. UpdateWindow();
  610. }
  611. }
  612. BOOL CXTPSyntaxEditCtrl::Select(int nRow1, int nDispCol1, int nRow2, int nDispCol2, BOOL bRedraw)
  613. {
  614. if (nRow1 < 1 || nRow1 > GetRowCount())
  615. return FALSE;
  616. if (nRow2 < 1 || nRow2 > GetRowCount())
  617. return FALSE;
  618. m_Selection.Reset_disp(nRow1, nDispCol1);
  619. m_Selection.SetEnd_disp(nRow2, nDispCol2);
  620. SetCurrentDocumentRow(m_Selection.GetNormalEnd_disp().nLine);
  621. m_nDispCol = m_Selection.GetNormalEnd_disp().nCol;
  622. m_nCurrentCol = m_Selection.GetNormalEnd_str().nCol + 1;
  623. if (bRedraw)
  624. {
  625. Invalidate(FALSE);
  626. }
  627. return TRUE;
  628. }
  629. void CXTPSyntaxEditCtrl::Unselect()
  630. {
  631. if (m_Selection.IsSelExist())
  632. {
  633. m_Selection.Reset_disp(m_Selection.GetEnd_disp().nLine,
  634.    m_Selection.GetEnd_disp().nCol);
  635. Invalidate(FALSE);
  636. }
  637. }
  638. BOOL CXTPSyntaxEditCtrl::ReplaceSel(LPCTSTR szNewText)
  639. {
  640. if (m_Selection.IsSelExist())
  641. {
  642. SetCurrentDocumentRow(m_Selection.GetNormalStart_disp().nLine);
  643. m_nDispCol = m_Selection.GetNormalStart_disp().nCol;
  644. m_nCurrentCol = m_Selection.GetNormalStart_str().nCol + 1; //CalcAbsCol(GetLineText(m_Selection.GetNormalStart_disp().nLine), m_nDispCol);
  645. }
  646. BOOL bOverwriteMode = m_pBuffer->GetOverwriteFlag();
  647. m_pBuffer->SetOverwriteFlag(FALSE);
  648. InsertString(szNewText, GetCurrentDocumentRow(), m_nCurrentCol, TRUE);
  649. m_pBuffer->SetOverwriteFlag(bOverwriteMode);
  650. return TRUE;
  651. }
  652. int CXTPSyntaxEditCtrl::ReplaceAll(LPCTSTR szFindText, LPCTSTR szReplaceText,
  653.   BOOL bMatchWholeWord, BOOL bMatchCase)
  654. {
  655. int nMatchFound = 0;
  656. if (_tcsclen(szFindText) == 0)
  657. {
  658. return 0;
  659. }
  660. int iTopRow = GetTopRow();
  661. int iCurRow = GetCurRow();
  662. int iCurCol = GetCurCol();
  663. SetCurPos(1,1,FALSE,FALSE);
  664. CWaitCursor wait;
  665. BOOL bFirst = TRUE;
  666. while (GetCurRow() <= GetRowCount())
  667. {
  668. BOOL bFound = Find(szFindText,
  669. bMatchWholeWord, bMatchCase,
  670. TRUE, FALSE);
  671. if (bFound)
  672. {
  673. ReplaceSel(szReplaceText);
  674. if (!bFirst)
  675. GetEditBuffer()->GetUndoRedoManager()->ChainLastCommand();
  676. }
  677. else
  678. break;
  679. bFirst = FALSE;
  680. nMatchFound++;
  681. }
  682. if (!bFirst)
  683. GetEditBuffer()->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_REPLACE_ALL);
  684. SetTopRow(iTopRow);
  685. SetCurPos(iCurRow, iCurCol);
  686. return nMatchFound;
  687. }
  688. BOOL CXTPSyntaxEditCtrl::Find(LPCTSTR szText, BOOL bMatchWholeWord, BOOL bMatchCase, BOOL bSearchDown,
  689.   BOOL bRedraw, int nStartRow, int nStartCol)
  690. {
  691. // First check the current row
  692. CString strLineText;
  693. int iStartCol = m_nCurrentCol;
  694. int iStartRow = GetCurrentDocumentRow();
  695. int nTopRow = m_nTopRow;
  696. int nBottomRow = GetDocumentRow(GetRowPerPage());
  697. if (m_Selection.IsSelExist())
  698. {
  699. if (bSearchDown)
  700. {
  701. iStartCol = m_Selection.GetNormalEnd_str().nCol + 1;  // m_ptEndSel.x;
  702. iStartRow = m_Selection.GetNormalEnd_str().nLine; // m_ptEndSel.y;
  703. }
  704. else
  705. {
  706. iStartCol = m_Selection.GetNormalStart_str().nCol + 1;  // m_ptStartSel.x;
  707. iStartRow = m_Selection.GetNormalStart_str().nLine; // m_ptStartSel.y;
  708. }
  709. }
  710. iStartCol = nStartCol < 0 ? iStartCol : nStartCol;
  711. iStartRow = nStartRow < 0 ? iStartRow : nStartRow;
  712. m_pBuffer->GetLineText(iStartRow, strLineText);
  713. BOOL bResult = MatchText(iStartRow, strLineText, szText, iStartCol, bMatchWholeWord, bMatchCase, bSearchDown, bRedraw);
  714. if (bResult)
  715. return TRUE;
  716. if (bSearchDown)
  717. {
  718. for (int i = (iStartRow + 1); i <= GetRowCount(); i++)
  719. {
  720. m_pBuffer->GetLineText(i, strLineText);
  721. bResult = MatchText(i, strLineText, szText, 0, bMatchWholeWord, bMatchCase, TRUE, bRedraw);
  722. if (bResult)
  723. break;
  724. }
  725. }
  726. else
  727. {
  728. for (int i = (iStartRow - 1); i >= 1; i--)
  729. {
  730. m_pBuffer->GetLineText(i, strLineText);
  731. bResult = MatchText(i, strLineText, szText, (int)_tcsclen(strLineText) + 1, bMatchWholeWord, bMatchCase, FALSE, bRedraw);
  732. if (bResult)
  733. break;
  734. }
  735. }
  736. if (bResult)
  737. {
  738. if (m_Selection.GetStart_str().nLine > nBottomRow)
  739. {
  740. ShiftCurrentVisibleRowDown(2 * GetRowPerPage() / 3);
  741. }
  742. else if (m_Selection.GetStart_str().nLine < nTopRow)
  743. {
  744. ShiftCurrentVisibleRowUp(GetRowPerPage() / 3);
  745. }
  746. SetCurCaretPos(m_Selection.GetNormalEnd_disp().nLine, m_Selection.GetNormalEnd_disp().nCol);
  747. }
  748. return bResult;
  749. }
  750. BOOL CXTPSyntaxEditCtrl::MatchText(int nRow, LPCTSTR szLineText, LPCTSTR szMatchText, int nStartPos, BOOL bMatchWholeWord, BOOL bMatchCase, BOOL bSearchForward, BOOL bRedraw)
  751. {
  752. BOOL bMatched = FALSE;
  753. const int nLen = (int)_tcsclen(szLineText);
  754. const int nSearchLen = (int)_tcsclen(szMatchText);
  755. static LPCTSTR szSeps = _T("() t<>{}:;,.=%"'!@#$^&*-\|[]/?");
  756. if (bSearchForward)
  757. {
  758. for (int i = nStartPos; i <= (nLen - nSearchLen + 1); i++)
  759. {
  760. int nFound = -1;
  761. if (bMatchCase)
  762. nFound = _tcsncmp((szLineText + i - 1), szMatchText, nSearchLen);
  763. else
  764. nFound = _tcsnicmp((szLineText + i - 1), szMatchText, nSearchLen);
  765. bMatched = (nFound == 0);
  766. if (bMatchWholeWord && nFound == 0)
  767. {
  768. BOOL bSepAfter = FALSE;
  769. BOOL bSepBefore = FALSE;
  770. if ((i - 1 + nSearchLen) >= nLen)
  771. {
  772. bSepAfter = TRUE;
  773. }
  774. else
  775. {
  776. LPCTSTR pcszTextAfter = (szLineText + i - 1) + nSearchLen;
  777. bSepAfter = (_tcschr(szSeps, *pcszTextAfter) != 0);
  778. }
  779. if (i == 1)
  780. bSepBefore = TRUE;
  781. else
  782. bSepBefore = (_tcschr(szSeps, *(szLineText + i - 2)) != 0);
  783. if (!bSepAfter || !bSepBefore)
  784. bMatched = FALSE;
  785. }
  786. if (bMatched)
  787. {
  788. int nDispCol1 = CalcDispCol(szLineText, i);
  789. int nDispCol2 = CalcDispCol(szLineText, (i + nSearchLen));
  790. Select(nRow, nDispCol1, nRow, nDispCol2, bRedraw);
  791. m_nDispCol = nDispCol2;
  792. m_nCurrentCol = (i + nSearchLen);
  793. SetCurCaretPos(m_Selection.GetNormalEnd_disp().nLine,
  794.    m_Selection.GetNormalEnd_disp().nCol, FALSE/*, FALSE*/);
  795. break;
  796. }
  797. }
  798. }
  799. else
  800. {
  801. for (int i = (nStartPos-nSearchLen); i >= 1 ; i--)
  802. {
  803. int nFound = -1;
  804. if (bMatchCase)
  805. nFound = _tcsncmp((szLineText + i - 1), szMatchText, nSearchLen);
  806. else
  807. nFound = _tcsnicmp((szLineText + i - 1), szMatchText, nSearchLen);
  808. bMatched = (nFound == 0);
  809. if (bMatchWholeWord && nFound == 0)
  810. {
  811. BOOL bSepAfter = FALSE;
  812. BOOL bSepBefore = FALSE;
  813. if ((i - 1 + nSearchLen) >= nLen)
  814. {
  815. bSepAfter = TRUE;
  816. }
  817. else
  818. {
  819. LPCTSTR pcszTextAfter = (szLineText + i - 1) + nSearchLen;
  820. bSepAfter = (_tcschr(szSeps, *pcszTextAfter) != 0);
  821. }
  822. if (i == 1)
  823. bSepBefore = TRUE;
  824. else
  825. bSepBefore = (_tcschr(szSeps, *(szLineText + i - 2)) != 0);
  826. if (!bSepAfter || !bSepBefore)
  827. bMatched = FALSE;
  828. }
  829. if (bMatched)
  830. {
  831. int nDispCol1 = CalcDispCol(szLineText, i);
  832. int nDispCol2 = CalcDispCol(szLineText, (i + nSearchLen));
  833. Select(nRow, nDispCol2, nRow, nDispCol1, bRedraw);
  834. m_nDispCol = nDispCol1;
  835. m_nCurrentCol = i;
  836. SetCurCaretPos(m_Selection.GetNormalEnd_disp().nLine,
  837.    m_Selection.GetNormalEnd_disp().nCol, FALSE/*, FALSE*/);
  838. break;
  839. }
  840. }
  841. }
  842. return bMatched;
  843. }
  844. void CXTPSyntaxEditCtrl::ShowHScrollBar(BOOL bShow)
  845. {
  846. ShowScrollBar(SB_HORZ, bShow);
  847. }
  848. void CXTPSyntaxEditCtrl::SelectWord(CPoint point)
  849. {
  850. int nRow = 0, nCol = 0;
  851. if (!RowColFromPoint(point, &nRow, &nCol))
  852. return;
  853. XTP_EDIT_LINECOL lcWord1, lcWord2;
  854. BOOL bOverSpace = FALSE;
  855. UINT nFindDir = XTP_EDIT_FINDWORD_NEXT;
  856. BOOL bFind = FindWordEx(nFindDir, nRow, nCol, lcWord1, lcWord2, bOverSpace);
  857. if (bFind && lcWord1.IsValidData() && lcWord2.IsValidData() && !bOverSpace)
  858. {
  859. m_Selection.SetStart_str(lcWord1.nLine, lcWord1.nCol-1);
  860. m_Selection.SetEnd_str(lcWord2.nLine, lcWord2.nCol-1);
  861. SetCurCaretPos(m_Selection.GetEnd_disp().nLine,
  862.    m_Selection.GetEnd_disp().nCol);
  863. Invalidate(FALSE);
  864. UpdateWindow();
  865. }
  866. }
  867. void CXTPSyntaxEditCtrl::FindWord(UINT nFindWhat)
  868. {
  869. XTP_EDIT_LINECOL lcWord1, lcWord2;
  870. BOOL bOverSpace;
  871. BOOL bFind = FindWordEx(nFindWhat, GetCurrentDocumentRow(), m_nCurrentCol,
  872. lcWord1, lcWord2, bOverSpace);
  873. XTP_EDIT_LINECOL& lcWordEnd = (nFindWhat == XTP_EDIT_FINDWORD_NEXT) ? lcWord2 : lcWord1;
  874. if (!bFind || !lcWordEnd.IsValidData())
  875. return;
  876. int nDispCol = CalcDispCol(lcWordEnd.nLine, lcWordEnd.nCol);
  877. SetCurCaretPos(lcWordEnd.nLine, nDispCol);
  878. }
  879. BOOL CXTPSyntaxEditCtrl::FindWordEx_str(UINT nFindWhat, XTP_EDIT_LINECOL lcPos_str,
  880. XTP_EDIT_LINECOL& rlcWordStart, XTP_EDIT_LINECOL& rlcWordEnd,
  881. BOOL& rbOverSpace)
  882. {
  883. return FindWordEx(nFindWhat, lcPos_str.nLine, lcPos_str.nCol + 1, rlcWordStart,
  884.   rlcWordEnd, rbOverSpace);
  885. }
  886. BOOL CXTPSyntaxEditCtrl::FindWordEx(UINT nFindWhat, int nTextRow, int nCol,
  887. XTP_EDIT_LINECOL& rlcWordStart, XTP_EDIT_LINECOL& rlcWordEnd, BOOL& rbOverSpace)
  888. {
  889. // set word invalid borders
  890. rlcWordStart = rlcWordEnd = XTP_EDIT_LINECOL::MINPOS;
  891. rbOverSpace = FALSE;
  892. CString strLineText;
  893. const TCHAR* pCh = NULL;
  894. const TCHAR szSeps[]    = _T("[]{}()<>.,;:!?=*&%@^#-"'/");
  895. //  const TCHAR szSepsAll[] = _T("[]{}()<>.,;:!?=*&%@^#-"'/nrt ");
  896. const TCHAR szSpaceSeps[] = _T(" t");
  897. const TCHAR szNLSeps[] = _T("rn");
  898. int nDir = nFindWhat == XTP_EDIT_FINDWORD_PREV ? -1 : 1;
  899. nCol = min(nCol-1, GetEditBuffer()->GetLineTextLengthC(nTextRow));
  900. XTP_EDIT_LINECOL lcPosStart = XTP_EDIT_LINECOL::MakeLineCol(nTextRow, max(0, nCol));
  901. CXTPSyntaxEditTextIterator txtIter(GetEditBuffer());
  902. BOOL bSeek = txtIter.SeekPos(lcPosStart);
  903. if (!bSeek)
  904. return FALSE;
  905. pCh = (nDir < 0) ? txtIter.SeekPrev() : txtIter.GetText(2);
  906. if (!pCh)
  907. return FALSE;
  908. // Is over space/tab
  909. BOOL bOverSpace = !!_tcschr(szSpaceSeps, *pCh);
  910. BOOL bOverNL = !!_tcschr(szNLSeps, *pCh);
  911. rbOverSpace = (bOverSpace || bOverNL);
  912. if (!pCh)
  913. return FALSE;
  914. BOOL bStartCharSep = !!_tcschr(szSeps, *pCh);
  915. if (rbOverSpace && nDir < 0)
  916. {
  917. while (pCh && (_tcschr(szSpaceSeps, *pCh) ||
  918.    !bOverSpace && _tcschr(szNLSeps, *pCh)) )
  919. {
  920. rlcWordEnd = txtIter.GetPosLC();
  921. pCh = txtIter.SeekPrev();
  922. }
  923. if (!pCh || rlcWordEnd.nCol == 0 || bOverNL)
  924. {
  925. rlcWordEnd.nCol++;
  926. rlcWordStart = rlcWordEnd;
  927. return TRUE;
  928. }
  929. bStartCharSep = !!_tcschr(szSeps, *pCh);
  930. }
  931. if (rbOverSpace && nDir > 0) // one word border find - find the other one
  932. {
  933. XTP_EDIT_LINECOL& rlc1 = (nDir < 0) ? rlcWordEnd   : rlcWordStart;
  934. XTP_EDIT_LINECOL& rlc2 = (nDir < 0) ? rlcWordStart : rlcWordEnd;
  935. rlc1 = txtIter.GetPosLC();
  936. rlc2 = txtIter.GetPosLC();
  937. while (pCh && (_tcschr(szSpaceSeps, *pCh) ||
  938.    _tcschr(szNLSeps, *pCh)) )
  939. {
  940. pCh = (nDir < 0) ? txtIter.SeekPrev() : txtIter.SeekNext(1, 2);
  941. rlc2 = txtIter.GetPosLC();
  942. }
  943. }
  944. else //find 2 word borders
  945. {
  946. rlcWordStart = txtIter.GetPosLC();
  947. while (pCh && !((bStartCharSep == (!_tcschr(szSeps, *pCh))) ||
  948. _tcschr(szSpaceSeps, *pCh) || _tcschr(szNLSeps, *pCh)))
  949. {
  950. rlcWordStart = txtIter.GetPosLC();
  951. pCh = txtIter.SeekPrev();
  952. }
  953. pCh = txtIter.SeekNext(1, 2);
  954. rlcWordEnd = txtIter.GetPosLC();
  955. while (pCh && !(bStartCharSep == !_tcschr(szSeps, *pCh) ||
  956. _tcschr(szSpaceSeps, *pCh) || _tcschr(szNLSeps, *pCh))
  957. )
  958. {
  959. pCh = txtIter.SeekNext(1, 2);
  960. rlcWordEnd = txtIter.GetPosLC();
  961. }
  962. if (rlcWordEnd > txtIter.GetPosLC_last(FALSE))
  963. rlcWordEnd = txtIter.GetPosLC_last(FALSE);
  964. }
  965. rlcWordStart.nCol++;
  966. rlcWordEnd.nCol++;
  967. return TRUE;
  968. }
  969. void CXTPSyntaxEditCtrl::DeleteSelectedLines(int iForceDeleteRow)
  970. {
  971. int iLineFrom, iLineTo;
  972. int iColTo = 1;
  973. if (m_Selection.IsSelExist())
  974. {
  975. iLineFrom = m_Selection.GetNormalStart_str().nLine;
  976. iLineTo = m_Selection.GetNormalEnd_str().nLine;;
  977. }
  978. else
  979. {
  980. iLineFrom = iForceDeleteRow;
  981. iLineTo = iLineFrom + 1;
  982. }
  983. if (iLineTo > GetRowCount())
  984. {
  985. iLineTo = GetRowCount();
  986. const int iLen = (int)_tcsclen(GetLineText(iLineTo));
  987. if (iLen == 0 && (iLineTo < 1 || iLineTo < iLineFrom))
  988. iColTo = 1;
  989. else
  990. iColTo = iLen + 1;
  991. if (iLineFrom == iLineTo && iColTo == 1 && iLen == 0)   // Nothing to delete
  992. return;
  993. }
  994. m_Selection.Reset_disp(iLineFrom, 1);
  995. int nFlags = xtpEditLMRefresh_Delete | xtpEditLMRefresh_Delete_only1;
  996. if (iColTo > 1)
  997. nFlags |= xtpEditLMRefresh_Delete_only2;
  998. GetLineMarksManager()->RefreshLineMarks(iLineFrom, iLineTo, nFlags);
  999. DeleteBuffer(iLineFrom, 1, iLineTo, iColTo);
  1000. }
  1001. BOOL CXTPSyntaxEditCtrl::IncreaseIndent()
  1002. {
  1003. if (!m_Selection.IsSelExist() ||
  1004. (m_Selection.GetStart_disp().nLine == m_Selection.GetEnd_disp().nLine &&
  1005.  m_Selection.GetStart_disp().nLine < GetRowCount()))
  1006. {
  1007. if (!m_pBuffer->GetOverwriteFlag())
  1008. return FALSE;
  1009. CString strLineText = GetLineText(GetCurrentDocumentRow());
  1010. const int iLineLen = (int)_tcsclen(strLineText);
  1011. int iMaxDispCol = CalcDispCol(strLineText, (iLineLen + 1));
  1012. if (m_nDispCol >= iMaxDispCol)
  1013. return FALSE;
  1014. if (m_nCurrentCol > (iLineLen + 1))
  1015. m_nCurrentCol = (iLineLen + 1);
  1016. m_nDispCol += (GetTabSize() - (m_nDispCol % GetTabSize()));     // Align by tab size
  1017. if (m_nDispCol < 1)
  1018. m_nDispCol = 1;
  1019. m_nDispCol = CalcValidDispCol(strLineText, m_nDispCol);
  1020. m_nCurrentCol = CalcAbsCol(strLineText, m_nDispCol);
  1021. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol);
  1022. return TRUE;
  1023. }
  1024. int iRowFrom = m_Selection.GetNormalStart_str().nLine;
  1025. int iRowTo = m_Selection.GetNormalEnd_str().nLine;
  1026. if (m_Selection.GetEnd_disp().nCol == 1 && iRowTo > iRowFrom)
  1027. iRowTo--;
  1028. if (!CanEditDoc())
  1029. return FALSE;
  1030. BOOL bModified = FALSE;
  1031. // First insert tabs in front
  1032. // If overwrite flag is not set
  1033. if (!m_pBuffer->GetOverwriteFlag())
  1034. {
  1035. const CString strText(GetTabWithSpace() ? CString((TCHAR)(' '), GetTabSize()) : _T("x09"));
  1036. for (int i = iRowFrom; i <= iRowTo; i++)
  1037. {
  1038. //**----------------------
  1039. OnBeforeEditChanged(i, 1);
  1040. //**----------------------
  1041. m_pBuffer->InsertText(strText, i, 1);
  1042. bModified = TRUE;
  1043. if (i > iRowFrom)
  1044. {
  1045. m_pBuffer->GetUndoRedoManager()->ChainLastCommand();
  1046. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_INDENTATION);
  1047. }
  1048. //**----------------------
  1049. OnEditChanged(i, 1, i, 2, xtpEditActInsert);
  1050. //**----------------------
  1051. }
  1052. if (bModified)
  1053. {
  1054. NotifyEditChanged(iRowFrom, iRowTo, XTP_EDIT_EDITACTION_MODIFYROW);
  1055. }
  1056. }
  1057. // Select the lines
  1058. m_Selection.Reset_disp(iRowFrom, 1);
  1059. m_Selection.SetEnd_disp(iRowTo + 1, 1);
  1060. if (bModified)
  1061. SetDocModified();
  1062. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol);
  1063. UpdateWindow();
  1064. UpdateScrollPos();
  1065. return TRUE;
  1066. }
  1067. BOOL CXTPSyntaxEditCtrl::DecreaseIndent()
  1068. {
  1069. if (!m_Selection.IsSelExist())
  1070. {
  1071. if (m_nCurrentCol < 1)
  1072. m_nCurrentCol = 1;
  1073. LPCTSTR szLineText = GetLineText(GetCurrentDocumentRow());
  1074. int iAddFactor = 1;
  1075. if ((m_nDispCol - 1) % GetTabSize())
  1076. iAddFactor = 0;
  1077. m_nDispCol = 1 + ((((m_nDispCol - 1) / GetTabSize()) - iAddFactor) * GetTabSize()); // Align by tab size
  1078. if (m_nDispCol < 1)
  1079. m_nDispCol = 1;
  1080. m_nDispCol = CalcValidDispCol(szLineText, m_nDispCol);
  1081. m_nCurrentCol = CalcAbsCol(szLineText, m_nDispCol);
  1082. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol);
  1083. return TRUE;
  1084. }
  1085. int iRowFrom = m_Selection.GetNormalStart_str().nLine;
  1086. int iRowTo = m_Selection.GetNormalEnd_str().nLine;
  1087. if (m_Selection.GetEnd_disp().nCol == 1 && iRowTo > iRowFrom)
  1088. iRowTo--;
  1089. if (!CanEditDoc())
  1090. return FALSE;
  1091. CString strLineText;
  1092. BOOL bFrontDeleted = FALSE;
  1093. BOOL bModified = FALSE;
  1094. // First tabs or blank spaces from front
  1095. for (int i = iRowFrom; i <= iRowTo; i++)
  1096. {
  1097. GetLineText(i, strLineText);
  1098. const int iLen = (int)_tcsclen(strLineText);
  1099. if (iLen == 0)
  1100. continue;
  1101. //**----------------------
  1102. OnBeforeEditChanged(i, 1);
  1103. //**----------------------
  1104. if (strLineText.GetAt(0) == _T('t'))
  1105. {
  1106. m_pBuffer->DeleteText(i, 1, i, 2);
  1107. bFrontDeleted = TRUE;
  1108. bModified = TRUE;
  1109. }
  1110. else
  1111. {
  1112. // No tab found in the front
  1113. // try to find blank spaces
  1114. int iSpaceCount = 0;
  1115. for (int iCol = 0; iCol < GetTabSize() && iCol < iLen; iCol++)
  1116. {
  1117. if (strLineText.GetAt(iCol) == 0x20)
  1118. iSpaceCount++;
  1119. else
  1120. break;
  1121. }
  1122. if (iSpaceCount)
  1123. {
  1124. m_pBuffer->DeleteText(i, 1, i, (iSpaceCount + 1));
  1125. bFrontDeleted = TRUE;
  1126. bModified = TRUE;
  1127. }
  1128. }
  1129. if (bFrontDeleted)
  1130. {
  1131. m_pBuffer->GetUndoRedoManager()->ChainLastCommand();
  1132. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_INDENT_DECREASE);
  1133. }
  1134. //**----------------------
  1135. OnEditChanged(i, 1, i, 1, xtpEditActDelete);
  1136. //**----------------------
  1137. }
  1138. if (bModified)
  1139. {
  1140. NotifyEditChanged(iRowFrom, iRowTo, XTP_EDIT_EDITACTION_MODIFYROW);
  1141. }
  1142. // Select the lines
  1143. m_Selection.Reset_disp(iRowFrom, 1);
  1144. m_Selection.SetEnd_disp(iRowTo + 1, 1);
  1145. if (bModified)
  1146. SetDocModified();
  1147. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol);
  1148. UpdateWindow();
  1149. UpdateScrollPos();
  1150. return TRUE;
  1151. }
  1152. void CXTPSyntaxEditCtrl::RecalcScrollBars()
  1153. {
  1154. _RecalcScrollBars();
  1155. CalculateEditbarLength();
  1156. }
  1157. void CXTPSyntaxEditCtrl::_RecalcHScrollMaxWidth()
  1158. {
  1159. CRect rcText = m_DrawTextProcessor.GetTextRect();
  1160. int nPageX = rcText.Width();
  1161. int nMaxX = m_DrawTextProcessor.GetRowsMaxWidth();
  1162. nMaxX += max(nPageX/4, m_DrawTextProcessor.GetTextMetrics().tmAveCharWidth * 20);
  1163. nMaxX = max(nMaxX, rcText.Width() * 2);
  1164. m_nHScrollMaxWidth = max(nMaxX, m_nHScrollMaxWidth);
  1165. }
  1166. void CXTPSyntaxEditCtrl::_RecalcScrollBars()
  1167. {
  1168. if (!GetPaintManager()->GetFont()->GetSafeHandle())
  1169. return;
  1170. if (!m_hWnd || !::IsWindow(m_hWnd))
  1171. return;
  1172. // create scroll bar info
  1173. SCROLLINFO info;
  1174. ZeroMemory(&info, sizeof(SCROLLINFO));
  1175. info.cbSize = sizeof(SCROLLINFO);
  1176. info.fMask = SIF_ALL;
  1177. int nVisRows = GetVisibleRowsCount();
  1178. CRect rcText = m_DrawTextProcessor.GetTextRect();
  1179. int nPageX = rcText.Width();
  1180. int nPageY = m_DrawTextProcessor.GetRowsCount(FALSE);
  1181. _RecalcHScrollMaxWidth();
  1182. if (GetRowCount() <= 0)
  1183. {
  1184. GetScrollInfo(SB_VERT, &info);
  1185. info.cbSize = sizeof(info);
  1186. info.fMask = SIF_DISABLENOSCROLL | SIF_PAGE | SIF_RANGE;
  1187. info.nPage = nPageY;
  1188. info.nMax= nPageY;
  1189. if (GetVertScrollBar())
  1190. {
  1191. SetScrollInfo(SB_VERT, &info);
  1192. EnableScrollBarCtrl(SB_VERT);
  1193. }
  1194. info.fMask = SIF_ALL;
  1195. GetScrollInfo(SB_HORZ, &info);
  1196. info.nMax = m_nHScrollMaxWidth;
  1197. info.nPage = nPageX;
  1198. info.fMask = SIF_DISABLENOSCROLL | SIF_PAGE | SIF_RANGE;
  1199. if (GetHorzScrollBar())
  1200. SetScrollInfo(SB_HORZ, &info);
  1201. }
  1202. else
  1203. {
  1204. info.fMask = SIF_ALL;
  1205. GetScrollInfo(SB_HORZ, &info);
  1206. info.nMin = 0;
  1207. info.nMax = max(m_nHScrollMaxWidth, info.nMax);
  1208. info.nPage = nPageX;
  1209. info.fMask = SIF_PAGE | SIF_POS | SIF_RANGE;
  1210. if (GetHorzScrollBar())
  1211. {
  1212. SetScrollInfo(SB_HORZ, &info);
  1213. EnableScrollBarCtrl(SB_HORZ);
  1214. }
  1215. //---------------------------------------
  1216. info.fMask = SIF_ALL;
  1217. GetScrollInfo(SB_VERT, &info);
  1218. int nMaxRowInPage = GetRowPerPage();
  1219. BOOL bVertScrl = (nVisRows >= nMaxRowInPage);
  1220. if (GetVertScrollBar())
  1221. {
  1222. if (bVertScrl)
  1223. {
  1224. int nVisRow = GetVisibleRowsCount(m_nTopRow);
  1225. info.nMin = 1;
  1226. info.nMax = nVisRows;
  1227. info.nPage = nPageY;
  1228. info.nPos = nVisRow;
  1229. info.fMask = SIF_PAGE | SIF_POS | SIF_RANGE;
  1230. SetScrollInfo(SB_VERT, &info);
  1231. }
  1232. else
  1233. {
  1234. m_nTopRow = 1;
  1235. Invalidate(FALSE);
  1236. }
  1237. EnableScrollBarCtrl(SB_VERT, bVertScrl);
  1238. }
  1239. }
  1240. if (GetCurrentDocumentRow() > GetRowCount())
  1241. SetCurCaretPos(GetRowCount(), m_nDispCol);
  1242. }
  1243. void CXTPSyntaxEditCtrl::RecalcVertScrollPos()
  1244. {
  1245. if (!::IsWindow(m_hWnd))
  1246. return;
  1247. if (!GetPaintManager()->GetFont()->GetSafeHandle())
  1248. return;
  1249. if (!GetVertScrollBar())
  1250. return;
  1251. int nPageRows = GetRowPerPage();
  1252. int nTotalVisRows = GetVisibleRowsCount();
  1253. BOOL bVertScrl = (nTotalVisRows >= nPageRows);
  1254. if (bVertScrl)
  1255. {
  1256. int nVisRow = GetVisibleRowsCount(m_nTopRow);
  1257. SCROLLINFO si;
  1258. ZeroMemory(&si, sizeof(SCROLLINFO));
  1259. si.cbSize = sizeof(si);
  1260. si.fMask = SIF_PAGE | SIF_POS | SIF_RANGE;
  1261. si.nMin = 1;
  1262. si.nMax = nTotalVisRows;
  1263. si.nPage = nPageRows;
  1264. si.nPos = nVisRow;
  1265. VERIFY(SetScrollInfo(SB_VERT, &si));
  1266. }
  1267. else
  1268. {
  1269. m_nTopRow = 1;
  1270. Invalidate(FALSE);
  1271. }
  1272. EnableScrollBarCtrl(SB_VERT, bVertScrl);
  1273. }
  1274. void CXTPSyntaxEditCtrl::RecalcHorzScrollPos()
  1275. {
  1276. if (!::IsWindow(m_hWnd))
  1277. return;
  1278. if (!GetPaintManager()->GetFont()->GetSafeHandle())
  1279. return;
  1280. if (!GetHorzScrollBar())
  1281. return;
  1282. _RecalcHScrollMaxWidth();
  1283. int nPos = GetScrollPos(SB_HORZ);
  1284. SCROLLINFO si;
  1285. ZeroMemory(&si, sizeof(SCROLLINFO));
  1286. si.cbSize = sizeof(si);
  1287. si.fMask = SIF_PAGE | SIF_POS | SIF_RANGE;
  1288. si.nMin = 0;
  1289. si.nMax = si.nMax = max(m_nHScrollMaxWidth, si.nMax);
  1290. si.nPage = m_DrawTextProcessor.GetTextRect().Width();
  1291. si.nPos = nPos;
  1292. VERIFY(SetScrollInfo(SB_HORZ, &si, FALSE));
  1293. }
  1294. void CXTPSyntaxEditCtrl::NotifyCurRowCol(int iRow, int iCol)
  1295. {
  1296. // Notify the parent class that position has been changed
  1297. XTP_EDIT_NMHDR_ROWCOLCHANGED src;
  1298. // NMHDR codes
  1299. src.nmhdr.code = XTP_EDIT_NM_ROWCOLCHANGED;
  1300. src.nmhdr.hwndFrom = m_hWnd;
  1301. src.nmhdr.idFrom = GetDlgCtrlID();
  1302. // Row col specific codes
  1303. src.nRow = iRow;
  1304. src.nCol = iCol;
  1305. // Notify the parent window
  1306. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  1307. {
  1308. m_pParentWnd->SendMessage(
  1309. WM_NOTIFY, (WPARAM)src.nmhdr.idFrom, (LPARAM)&src);
  1310. }
  1311. }
  1312. void CXTPSyntaxEditCtrl::ValidateCol(const int nRow, int &nCol, int& nAbsCol)
  1313. {
  1314. CString strText;
  1315. GetLineText(nRow, strText);
  1316. nCol = CalcValidDispCol(strText, nCol);
  1317. nAbsCol = CalcAbsCol(strText, nCol);
  1318. }
  1319. void CXTPSyntaxEditCtrl::EnableOleDrag(BOOL bEnableDrag)
  1320. {
  1321. m_bEnableOleDrag = bEnableDrag;
  1322. }
  1323. HGLOBAL CXTPSyntaxEditCtrl::GetSelectionBuffer(UINT nFormat)
  1324. {
  1325. if (!m_Selection.IsSelExist())
  1326. return NULL;
  1327. if (nFormat != CF_TEXT && nFormat != CF_UNICODETEXT)
  1328. return NULL;
  1329. CMemFile file(CalcAveDataSize(m_Selection.GetNormalStart_str().nLine,
  1330.   m_Selection.GetNormalEnd_str().nLine));
  1331. BOOL bRes = m_pBuffer->GetBuffer(m_Selection.GetNormalStart_disp(),
  1332.  m_Selection.GetNormalEnd_disp(), file,
  1333.  m_Selection.bBlockSelectionMode, TRUE);
  1334. if (!bRes)
  1335. return NULL;
  1336. file.Seek(0, CFile::end);
  1337. file.Write(_T(""), sizeof(TCHAR));
  1338. int nLen = (int)file.GetLength();
  1339. BYTE *pBytes = file.Detach();
  1340. UINT uCodePage = m_pBuffer->GetCodePage();
  1341. #ifdef _UNICODE
  1342. // If Unicode defined then for CF_TEXT conversion is needed
  1343. if (nFormat == CF_TEXT)
  1344. {
  1345. nLen = ::WideCharToMultiByte(uCodePage, 0, (LPWSTR)pBytes, -1, NULL, 0, NULL, NULL);
  1346. }
  1347. #else
  1348. if (nFormat == CF_UNICODETEXT)
  1349. {
  1350. nLen = ::MultiByteToWideChar(uCodePage, 0, (LPCSTR)pBytes, -1, NULL, 0);
  1351. nLen *= sizeof(wchar_t);
  1352. }
  1353. #endif
  1354. HGLOBAL hMem = ::GlobalAlloc(GMEM_MOVEABLE, nLen);
  1355. if (!hMem)
  1356. return 0;
  1357. void *pText = ::GlobalLock(hMem);
  1358. #ifdef _UNICODE
  1359. if (nFormat == CF_UNICODETEXT)
  1360. MEMCPY_S(pText, pBytes, nLen);
  1361. else
  1362. {
  1363. ::WideCharToMultiByte(uCodePage, 0, (LPWSTR)pBytes, -1, (LPSTR)pText, nLen, NULL, NULL);
  1364. }
  1365. #else
  1366. if (nFormat == CF_TEXT)
  1367. MEMCPY_S(pText, pBytes, nLen);
  1368. else
  1369. {
  1370. ::MultiByteToWideChar(uCodePage, 0, (LPSTR)pBytes, -1, (LPWSTR)pText, (nLen / sizeof(wchar_t)));
  1371. }
  1372. #endif
  1373. ::GlobalUnlock(hMem);
  1374. ::free(pBytes);
  1375. return hMem;
  1376. }
  1377. int CXTPSyntaxEditCtrl::GetSelectionText(CString &strText)
  1378. {
  1379. if (!m_Selection.IsSelExist())
  1380. return 0;
  1381. CMemFile file(CalcAveDataSize(m_Selection.GetNormalStart_str().nLine,
  1382.   m_Selection.GetNormalEnd_str().nLine));
  1383. BOOL bRes = m_pBuffer->GetBuffer(m_Selection.GetNormalStart_disp(),
  1384.  m_Selection.GetNormalEnd_disp(), file,
  1385.  m_Selection.bBlockSelectionMode, TRUE);
  1386. if (!bRes)
  1387. return 0;
  1388. file.Write((LPVOID)_T(""), sizeof(TCHAR));
  1389. int iLen = (int)file.GetLength();
  1390. BYTE *pBytes = file.Detach();
  1391. strText = (LPTSTR)pBytes;
  1392. free(pBytes);
  1393. return iLen;
  1394. }
  1395. void CXTPSyntaxEditCtrl::SetDropPos(int iRow, int iCol)
  1396. {
  1397. m_ptDropPos.y = iRow;
  1398. m_ptDropPos.x = iCol;
  1399. }
  1400. void CXTPSyntaxEditCtrl::OnShowWindow(BOOL bShow, UINT nStatus)
  1401. {
  1402. CWnd::OnShowWindow(bShow, nStatus);
  1403. }
  1404. void CXTPSyntaxEditCtrl::OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized)
  1405. {
  1406. CWnd::OnActivate(nState, pWndOther, bMinimized);
  1407. SetActive(nState != WA_INACTIVE);
  1408. }
  1409. void CXTPSyntaxEditCtrl::SetActive(BOOL bIsActive)
  1410. {
  1411. m_bIsActive = bIsActive;
  1412. if (::IsWindow(m_hWnd))
  1413. {
  1414. RedrawWindow();
  1415. }
  1416. }
  1417. void CXTPSyntaxEditCtrl::HandleDrop(BOOL bCopy)
  1418. {
  1419. if (!CanEditDoc())
  1420. return;
  1421. if (!m_bRightButtonDrag)
  1422. CopyOrMoveText(bCopy);
  1423. else
  1424. {
  1425. CMenu menu;
  1426. menu.CreatePopupMenu();
  1427. CString csMenuText;
  1428. XTPResourceManager()->LoadString(&csMenuText, XTP_IDS_EDIT_MOVE_HERE);
  1429. menu.AppendMenu(MF_STRING, XTP_IDC_EDIT_DRAG_MOVE, csMenuText);
  1430. XTPResourceManager()->LoadString(&csMenuText, XTP_IDS_EDIT_COPY_HERE);
  1431. menu.AppendMenu(MF_STRING, XTP_IDC_EDIT_DRAG_COPY, csMenuText);
  1432. menu.AppendMenu(MF_SEPARATOR);
  1433. XTPResourceManager()->LoadString(&csMenuText, XTP_IDS_EDIT_CANCEL);
  1434. menu.AppendMenu(MF_STRING, IDCANCEL, csMenuText);
  1435. CPoint pt;
  1436. GetCursorPos(&pt);
  1437. int nID = menu.TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON|TPM_NONOTIFY|TPM_RETURNCMD, pt.x, pt.y, this);
  1438. if (nID == XTP_IDC_EDIT_DRAG_MOVE || nID == XTP_IDC_EDIT_DRAG_COPY)
  1439. CopyOrMoveText(nID == XTP_IDC_EDIT_DRAG_COPY);
  1440. }
  1441. m_bRightButtonDrag = FALSE;
  1442. m_bDragging = FALSE;
  1443. }
  1444. void CXTPSyntaxEditCtrl::ShowDefaultContextMenu()
  1445. {
  1446. CPoint ptCur;
  1447. GetCursorPos(&ptCur);
  1448. OnContextMenu(this, ptCur);
  1449. }
  1450. void CXTPSyntaxEditCtrl::CancelRightButtonDrag()
  1451. {
  1452. m_bRightButtonDrag = FALSE;
  1453. }
  1454. BOOL CXTPSyntaxEditCtrl::IsRightButtonDrag()
  1455. {
  1456. return m_bRightButtonDrag;
  1457. }
  1458. void CXTPSyntaxEditCtrl::SetRightButtonDrag(BOOL bRightButtonDrag)
  1459. {
  1460. m_bRightButtonDrag = bRightButtonDrag;
  1461. }
  1462. void CXTPSyntaxEditCtrl::EnableWhiteSpace(BOOL bShow)
  1463. {
  1464. m_bEnableWhiteSpace = bShow;
  1465. Invalidate(FALSE);
  1466. }
  1467. BOOL CXTPSyntaxEditCtrl::IsEnabledWhiteSpace()
  1468. {
  1469. return m_bEnableWhiteSpace;
  1470. }
  1471. void CXTPSyntaxEditCtrl::EnableVirtualSpace(BOOL bEnable, BOOL bUpdateReg)
  1472. {
  1473. SetValueBool(XTP_EDIT_REG_VIRTUALSPACE, bEnable, m_bVirtualSpace, bUpdateReg);
  1474. }
  1475. void CXTPSyntaxEditCtrl::SetCollapsibleNodes(BOOL bDrawNodes, BOOL bUpdateReg)
  1476. {
  1477. SetValueBool(XTP_EDIT_REG_COLLAPSIBLENODES, bDrawNodes, m_bDrawNodes, bUpdateReg);
  1478. CalculateEditbarLength();
  1479. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, TRUE);
  1480. if (!m_bDrawNodes)
  1481. ExpandAll();
  1482. else
  1483. Invalidate(FALSE);
  1484. }
  1485. BOOL CXTPSyntaxEditCtrl::SetTabWithSpace(BOOL bTabWithSpace, BOOL bUpdateReg/*=FALSE*/)
  1486. {
  1487. if (!SetValueBool(XTP_EDIT_REG_TABWITHSPACE, bTabWithSpace, m_bTabWithSpace, bUpdateReg))
  1488. return FALSE;
  1489. return TRUE;
  1490. }
  1491. BOOL CXTPSyntaxEditCtrl::GetTabWithSpace() const
  1492. {
  1493. return m_bTabWithSpace;
  1494. }
  1495. void CXTPSyntaxEditCtrl::NotifySelInit()
  1496. {
  1497. NotifyParent(XTP_EDIT_NM_SELINIT);
  1498. }
  1499. BOOL CXTPSyntaxEditCtrl::SetTabSize(int nTabSize, BOOL bUpdateReg)
  1500. {
  1501. if (!m_pBuffer->SetTabSize(nTabSize, bUpdateReg))
  1502. return FALSE;
  1503. m_DrawTextProcessor.SetTabSize(nTabSize);
  1504. CString strText;
  1505. GetLineText(GetCurrentDocumentRow(), strText);
  1506. m_nDispCol = CalcDispCol(strText, m_nCurrentCol);
  1507. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  1508. return TRUE;
  1509. }
  1510. int CXTPSyntaxEditCtrl::GetTabSize() const
  1511. {
  1512. return m_pBuffer ? m_pBuffer->GetTabSize() : 4;
  1513. }
  1514. BOOL CXTPSyntaxEditCtrl::SetSyntaxColor(BOOL bSyntaxColor, BOOL bUpdateReg)
  1515. {
  1516. if (!SetValueBool(XTP_EDIT_REG_SYNTAXCOLOR, bSyntaxColor, m_bSyntaxColor, bUpdateReg))
  1517. return FALSE;
  1518. m_arOnScreenSchCache.RemoveAll();
  1519. m_pBuffer->EnableParser(bSyntaxColor);
  1520. return TRUE;
  1521. }
  1522. LRESULT CXTPSyntaxEditCtrl::NotifyParent(UINT uCode)
  1523. {
  1524. ASSERT_VALID(m_pParentWnd);
  1525. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  1526. {
  1527. NMHDR hdr;
  1528. hdr.hwndFrom = m_hWnd;
  1529. hdr.code = uCode;
  1530. hdr.idFrom = GetDlgCtrlID();
  1531. return m_pParentWnd->SendMessage(
  1532. WM_NOTIFY, (WPARAM)hdr.idFrom, (LPARAM)&hdr);
  1533. }
  1534. return -1;
  1535. }
  1536. LRESULT CXTPSyntaxEditCtrl::NotifyMouseEvent(UINT uCode, UINT nFlags, const CPoint& point)
  1537. {
  1538. UNREFERENCED_PARAMETER(nFlags);
  1539. ASSERT_VALID(GetParent());
  1540. if (::IsWindow(GetParent()->GetSafeHwnd()))
  1541. {
  1542. NMMOUSE hdrMouse;
  1543. ::ZeroMemory(&hdrMouse, sizeof(hdrMouse));
  1544. hdrMouse.hdr.hwndFrom = m_hWnd;
  1545. hdrMouse.hdr.code = uCode;
  1546. hdrMouse.hdr.idFrom = GetDlgCtrlID();
  1547. hdrMouse.pt = point;
  1548. return GetParent()->SendMessage(WM_NOTIFY, (WPARAM)hdrMouse.hdr.idFrom,
  1549.    (LPARAM)&hdrMouse);
  1550. }
  1551. return -1;
  1552. }
  1553. BOOL CXTPSyntaxEditCtrl::SetSelMargin(BOOL bSelMargin, BOOL bUpdateReg)
  1554. {
  1555. if (!SetValueBool(XTP_EDIT_REG_SELMARGIN, bSelMargin, m_bSelMargin, bUpdateReg))
  1556. return FALSE;
  1557. CalculateEditbarLength();
  1558. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  1559. return TRUE;
  1560. }
  1561. BOOL CXTPSyntaxEditCtrl::SetScrollBars(BOOL bHorzSBar, BOOL bVertSBar, BOOL bUpdateReg/*=FALSE*/)
  1562. {
  1563. if (!SetValueBool(XTP_EDIT_REG_HSCROLLBAR, bHorzSBar, m_bHorzScrollBar, bUpdateReg))
  1564. return FALSE;
  1565. if (!SetValueBool(XTP_EDIT_REG_VSCROLLBAR, bVertSBar, m_bVertScrollBar, bUpdateReg))
  1566. return FALSE;
  1567. DWORD dwStyle = ::GetWindowLong(m_hWnd, GWL_STYLE) & ~(WS_HSCROLL|WS_VSCROLL);
  1568. if (bHorzSBar && !IsCreateScrollbarOnParent())
  1569. dwStyle |= WS_HSCROLL;
  1570. if (bVertSBar && !IsCreateScrollbarOnParent())
  1571. dwStyle |= WS_VSCROLL;
  1572. EnableScrollBarCtrl(SB_HORZ, m_bHorzScrollBar);
  1573. EnableScrollBarCtrl(SB_VERT, m_bVertScrollBar);
  1574. ::SetWindowLong(m_hWnd, GWL_STYLE, dwStyle);
  1575. return TRUE;
  1576. }
  1577. BOOL CXTPSyntaxEditCtrl::CanEditDoc()
  1578. {
  1579. if (IsReadOnly())
  1580. return FALSE;
  1581. BOOL bCancel = (BOOL)NotifyParent(XTP_EDIT_NM_EDITCHANGING);
  1582. return !bCancel;
  1583. }
  1584. void CXTPSyntaxEditCtrl::NotifyEditChanged(int iRowFrom, int iRowTo, UINT nActions)
  1585. {
  1586. XTP_EDIT_NMHDR_EDITCHANGED sec;
  1587. sec.nmhdr.code = XTP_EDIT_NM_EDITCHANGED;
  1588. sec.nmhdr.hwndFrom = m_hWnd;
  1589. sec.nmhdr.idFrom = GetDlgCtrlID();
  1590. sec.nRowFrom = iRowFrom;
  1591. sec.nRowTo = iRowTo;
  1592. sec.nAction = nActions;
  1593. RefreshLineMarks(&sec);
  1594. // Notify the parent window
  1595. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  1596. {
  1597. m_pParentWnd->SendMessage(
  1598. WM_NOTIFY, (WPARAM)sec.nmhdr.idFrom, (LPARAM)&sec);
  1599. }
  1600. }
  1601. BOOL CXTPSyntaxEditCtrl::CreateInsertText(LPTSTR szText,
  1602.   CString& strTextToIns,
  1603.   int& iNewRow,
  1604.   int& iNewCol,
  1605.   int& iNewDispCol,
  1606.   int& iEditRowFrom,
  1607.   int& iEditRowTo,
  1608.   int& iChainActionCount)
  1609. {
  1610. // As this function will always be called by OnChar
  1611. // We can assume szText will contain only one character
  1612. // or the CRLF
  1613. int nCurDocRow = GetCurrentDocumentRow();
  1614. CString strLineText;
  1615. GetLineText(nCurDocRow, strLineText);
  1616. int nLineLenC = (int)_tcsclen(strLineText);
  1617. BOOL bTextCRLF = m_pBuffer->IsTextCRLF(szText);
  1618. BOOL bProcess = TRUE;
  1619. if (!bTextCRLF)
  1620. {
  1621. if (szText[0] == 0x09 && !GetTabWithSpace() &&
  1622. m_nCurrentCol > 1 && nLineLenC > 0)
  1623. {
  1624. int nAlign = (m_nDispCol-1) % GetTabSize();
  1625. iNewDispCol = m_nDispCol - nAlign + GetTabSize();
  1626. int nDispCol = CalcDispCol(strLineText, nLineLenC + 1);
  1627. int nCurCol = m_nCurrentCol;
  1628. if (nDispCol < m_nDispCol)
  1629. {
  1630. nCurCol = nLineLenC+1;
  1631. }
  1632. else
  1633. {
  1634. nDispCol = m_nDispCol;
  1635. }
  1636. int nCharsToCheck = nCurCol-1;
  1637. if (nCharsToCheck)
  1638. {
  1639. LPCTSTR pCheck = _tcsninc((LPCTSTR)strLineText, max(0, nCurCol - 1 - 1));
  1640. int nBlanks;
  1641. for (nBlanks = 0; pCheck && nBlanks < nCharsToCheck; nBlanks++)
  1642. {
  1643. if (!pCheck || *pCheck != _T(' '))
  1644. break;
  1645. pCheck = _tcsdec((LPCTSTR)strLineText, pCheck);
  1646. }
  1647. if (nBlanks > 0)
  1648. {
  1649. int nAdditionalTabs = 0;
  1650. int nDC = m_nDispCol - nBlanks;
  1651. nDC += GetTabSize() - nDC % GetTabSize();
  1652. for(; nDC < m_nDispCol; nDC += GetTabSize())
  1653. {
  1654. nAdditionalTabs++;
  1655. }
  1656. m_pBuffer->SetOverwriteFlag(FALSE);
  1657. nCurDocRow = GetCurrentDocumentRow();
  1658. //**----------------------
  1659. OnBeforeEditChanged(nCurDocRow, nCurCol - nBlanks);
  1660. m_pBuffer->DeleteText(nCurDocRow, nCurCol - nBlanks, nCurDocRow, nCurCol);
  1661. OnEditChanged(nCurDocRow, nCurCol - nBlanks, nCurDocRow, nCurCol, xtpEditActDelete);
  1662. //**----------------------
  1663. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_FORMAT);
  1664. m_nDispCol -= nBlanks;
  1665. m_nCurrentCol -= nBlanks;
  1666. iChainActionCount = 2;
  1667. GetLineText(nCurDocRow, strLineText);
  1668. nLineLenC = (int)_tcsclen(strLineText);
  1669. if (nAdditionalTabs)
  1670. {
  1671. strTextToIns += CString(_T('t'), nAdditionalTabs);
  1672. }
  1673. }
  1674. }
  1675. }
  1676. else
  1677. {
  1678. iNewDispCol = m_nDispCol + (szText[0] == 0x09 ? GetTabSize() : 1);
  1679. iNewCol = m_nCurrentCol + (szText[0] == 0x09 ? (GetTabWithSpace() ? GetTabSize() : 1) : 1);
  1680. if (GetAutoIndent() && m_nAutoIndentCol > 0)
  1681. {
  1682. strTextToIns += CString(_T('t'), m_nInsertTabCount) +
  1683. CString(_T(' '), m_nInsertSpaceCount);
  1684. m_nDispCol = 1;
  1685. m_nCurrentCol = 1;
  1686. }
  1687. }
  1688. iNewRow = GetCurrentDocumentRow();
  1689. strTextToIns += szText;
  1690. m_nAutoIndentCol = 0;
  1691. m_nInsertTabCount = 0;
  1692. m_nInsertSpaceCount = 0;
  1693. }
  1694. else if (bTextCRLF)
  1695. {
  1696. // Enter pressed
  1697. nCurDocRow = GetCurrentDocumentRow();
  1698. iNewRow = nCurDocRow + 1;
  1699. iNewCol = iNewDispCol = 1;
  1700. iEditRowFrom = nCurDocRow;
  1701. iEditRowTo = iEditRowFrom + 1;
  1702. strTextToIns = szText;
  1703. if (!GetAutoIndent())
  1704. return TRUE;
  1705. BOOL bInsertMid = (m_nCurrentCol <= nLineLenC);
  1706. DoAutoIndentIfNeed(GetCurrentDocumentRow(), m_nDispCol);
  1707. if (m_nAutoIndentCol > 0 && bInsertMid)
  1708. {
  1709. CString strTabs;
  1710. FillTabs(strTabs, m_nAutoIndentCol);
  1711. strTextToIns += strTabs;
  1712. iNewDispCol = m_nAutoIndentCol;
  1713. iNewCol = (int)_tcsclen(strTabs);
  1714. m_nAutoIndentCol = 0;
  1715. }
  1716. }
  1717. return bProcess;
  1718. }
  1719. void CXTPSyntaxEditCtrl::DoAutoIndentIfNeed(int nBaseDocRow, int nDispCol_prev)
  1720. {
  1721. if (!m_bAutoIndent)
  1722. return;
  1723. m_nAutoIndentCol = 0;
  1724. if (nDispCol_prev == 1)
  1725. return;
  1726. CString strTempLineText;
  1727. BOOL bFound = FALSE;
  1728. GetLineText(nBaseDocRow, strTempLineText);
  1729. LPCTSTR pChar = (LPCTSTR)strTempLineText;
  1730. for (; pChar && *pChar; pChar = _tcsinc(pChar))
  1731. {
  1732. if (*pChar == 0x09)
  1733. {
  1734. m_nAutoIndentCol += (GetTabSize() - (m_nAutoIndentCol % GetTabSize()));
  1735. }
  1736. else if (*pChar == 0x20)
  1737. {
  1738. m_nAutoIndentCol++;
  1739. }
  1740. else
  1741. {
  1742. bFound = TRUE;
  1743. break;
  1744. }
  1745. }
  1746. if (!bFound && nDispCol_prev > 1)
  1747. {
  1748. bFound = TRUE;
  1749. m_nAutoIndentCol = nDispCol_prev - 1;
  1750. }
  1751. if (nDispCol_prev > 1 && m_nAutoIndentCol > nDispCol_prev - 1)
  1752. m_nAutoIndentCol = nDispCol_prev - 1;
  1753. if (bFound)
  1754. {
  1755. m_nInsertTabCount = m_nAutoIndentCol / GetTabSize();
  1756. m_nInsertSpaceCount = m_nAutoIndentCol % GetTabSize();
  1757. m_nAutoIndentCol++;
  1758. }
  1759. else
  1760. {
  1761. m_nAutoIndentCol = 0;
  1762. m_nInsertTabCount = 0;
  1763. m_nInsertSpaceCount = 0;
  1764. }
  1765. }
  1766. void CXTPSyntaxEditCtrl::FillTabs(CString &rstrTextToIns, int nDispColl)
  1767. {
  1768. int nTabsCount = GetTabWithSpace() ? 0 : (nDispColl-1) / GetTabSize();
  1769. int nSpaceCount = nDispColl - nTabsCount * GetTabSize() - 1;
  1770. if (nTabsCount)
  1771. {
  1772. CString strTabs(_T('t'), nTabsCount);
  1773. rstrTextToIns += strTabs;
  1774. }
  1775. if (nSpaceCount)
  1776. {
  1777. CString strSpaces(_T(' '), nSpaceCount);
  1778. rstrTextToIns += strSpaces;
  1779. }
  1780. }
  1781. BOOL CXTPSyntaxEditCtrl::IsRowVisible(int iRow)
  1782. {
  1783. if (iRow > GetRowCount() || iRow < 1)
  1784. return FALSE;
  1785. int iMaxRowInPage = GetRowPerPage();
  1786. return (iRow >= m_nTopRow && iRow <= (m_nTopRow + iMaxRowInPage));
  1787. }
  1788. BOOL CXTPSyntaxEditCtrl::LowercaseSelection()
  1789. {
  1790. return DoChangeSelectionCase(FALSE);
  1791. }
  1792. BOOL CXTPSyntaxEditCtrl::UppercaseSelection()
  1793. {
  1794. return DoChangeSelectionCase(TRUE);
  1795. }
  1796. BOOL CXTPSyntaxEditCtrl::DoChangeSelectionCase(BOOL bUpper)
  1797. {
  1798. if (!m_Selection.IsSelExist())
  1799. return FALSE;
  1800. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(FALSE);
  1801. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(TRUE);
  1802. int nRowStrat = m_Selection.GetNormalStart_disp().nLine;
  1803. int nRowEnd = m_Selection.GetNormalEnd_disp().nLine;
  1804. for (int nRow = nRowStrat; nRow <= nRowEnd; nRow++)
  1805. {
  1806. int nVisFrom = 0;
  1807. int nVisTo = INT_MAX;
  1808. if (m_Selection.bBlockSelectionMode || nRow == nRowStrat)
  1809. nVisFrom = m_Selection.GetNormalStart_disp().nCol;
  1810. if (m_Selection.bBlockSelectionMode || nRow == nRowEnd)
  1811. nVisTo = m_Selection.GetNormalEnd_disp().nCol;
  1812. m_pBuffer->ChangeCase(nRow, nVisFrom, nVisTo, bUpper);
  1813. }
  1814. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(FALSE);
  1815. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(XTP_IDS_EDIT_CASE);
  1816. InvalidateRows(nRowStrat, nRowEnd);
  1817. Invalidate(FALSE);
  1818. return TRUE;
  1819. }
  1820. BOOL CXTPSyntaxEditCtrl::DoChangeSelectionTabify(BOOL bTabify)
  1821. {
  1822. if (!IsSelectionExist())
  1823. return FALSE;
  1824. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(FALSE);
  1825. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(TRUE);
  1826. int nRowStrat = m_Selection.GetNormalStart_disp().nLine;
  1827. int nRowEnd = m_Selection.GetNormalEnd_disp().nLine;
  1828. for (int nRow = nRowStrat; nRow <= nRowEnd; nRow++)
  1829. {
  1830. int nVisFrom = 0;
  1831. int nVisTo = INT_MAX;
  1832. if (m_Selection.bBlockSelectionMode || nRow == nRowStrat)
  1833. nVisFrom = m_Selection.GetNormalStart_disp().nCol;
  1834. if (m_Selection.bBlockSelectionMode || nRow == nRowEnd)
  1835. nVisTo = m_Selection.GetNormalEnd_disp().nCol;
  1836. m_pBuffer->ChangeTabification(nRow, nVisFrom, nVisTo, bTabify);
  1837. }
  1838. m_pBuffer->GetUndoRedoManager()->SetGroupInsertMode(FALSE);
  1839. m_pBuffer->GetUndoRedoManager()->SetLastCommandText(bTabify ? XTP_IDS_EDIT_TABIFY : XTP_IDS_EDIT_UNTABIFY);
  1840. InvalidateRows(nRowStrat, nRowEnd);
  1841. Invalidate(FALSE);
  1842. return TRUE;
  1843. }
  1844. BOOL CXTPSyntaxEditCtrl::TabifySelection()
  1845. {
  1846. return DoChangeSelectionTabify(TRUE);
  1847. }
  1848. BOOL CXTPSyntaxEditCtrl::UnTabifySelection()
  1849. {
  1850. return DoChangeSelectionTabify(FALSE);
  1851. }
  1852. BOOL CXTPSyntaxEditCtrl::SetLineNumbers(BOOL bLineNumbers, BOOL bUpdateReg/*=FALSE*/)
  1853. {
  1854. if (!SetValueBool(XTP_EDIT_REG_LINENUMBERS, bLineNumbers, m_bLineNumbers, bUpdateReg))
  1855. return FALSE;
  1856. CalculateEditbarLength();
  1857. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, TRUE);
  1858. return TRUE;
  1859. }
  1860. BOOL CXTPSyntaxEditCtrl::GetLineNumbers() const
  1861. {
  1862. return m_bLineNumbers;
  1863. }
  1864. void CXTPSyntaxEditCtrl::CalculateEditbarLength(CDC* pDC)
  1865. {
  1866. int nPrvEditbarLength = m_nEditbarLength;
  1867. // gutter
  1868. m_nEditbarLength = GetSelMargin() ? m_nMarginLength : 0;
  1869. // line numbers
  1870. m_nLineNumLength = 0;
  1871. if (GetLineNumbers())
  1872. {
  1873. int nMaxPageRow = GetRowCount();
  1874. if (pDC && pDC->IsPrinting())
  1875. nMaxPageRow = GetTopRow() + GetRowPerPage();
  1876. int nRowNumLen = (int)log10l(nMaxPageRow) + 1;
  1877. CWindowDC dc(NULL);
  1878. CXTPFontDC fontDC(&dc, GetPaintManager()->GetFontLineNumber());
  1879. TEXTMETRIC tm;
  1880. VERIFY(dc.GetTextMetrics(&tm));
  1881. m_nLineNumLength = (nRowNumLen + 1) * (tm.tmAveCharWidth) + 2;
  1882. m_nEditbarLength += m_nLineNumLength;
  1883. }
  1884. // collapsible nodes area
  1885. if (m_bDrawNodes)
  1886. {
  1887. m_nEditbarLength += m_nNodesWidth;
  1888. }
  1889. if (GetSelMargin() || GetLineNumbers() || m_bDrawNodes)
  1890. m_nEditbarLength += TEXT_LEFT_OFFSET;
  1891. // update window if necessary
  1892. if (nPrvEditbarLength != m_nEditbarLength && (!pDC || !pDC->IsPrinting()))
  1893. {
  1894. CRect rcText;
  1895. CalcEditRects(NULL, NULL, NULL, &rcText);
  1896. m_DrawTextProcessor.SetTextRect(rcText);
  1897. _RecalcScrollBars();
  1898. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  1899. Invalidate(FALSE);
  1900. }
  1901. }
  1902. void CXTPSyntaxEditCtrl::AddRemoveBookmark(int nRow)
  1903. {
  1904. if (GetLineMarksManager())
  1905. {
  1906. GetLineMarksManager()->AddRemoveLineMark(nRow, xtpEditLMT_Bookmark);
  1907. }
  1908. RedrawLineMarks();
  1909. }
  1910. void CXTPSyntaxEditCtrl::AddRemoveBreakPoint(int nRow)
  1911. {
  1912. if (GetLineMarksManager())
  1913. {
  1914. GetLineMarksManager()->AddRemoveLineMark(nRow, xtpEditLMT_Breakpoint);
  1915. }
  1916. RedrawLineMarks();
  1917. }
  1918. void CXTPSyntaxEditCtrl::CollapseExpandBlock(int nRow)
  1919. {
  1920. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  1921. if (!pMgr)
  1922. return;
  1923. // nLineEnd: adjust also column, for example when collapsing to the string of lower length
  1924. int nLineEnd = m_nCurrentCol; //m_nDispCol;
  1925. if (pMgr->HasRowMark(nRow, xtpEditLMT_Collapsed))
  1926. {
  1927. // try expand
  1928. pMgr->DeleteLineMark(nRow, xtpEditLMT_Collapsed);
  1929. }
  1930. else
  1931. {
  1932. // try collapse
  1933. XTP_EDIT_LMPARAM LMCoParam;
  1934. CXTPSyntaxEditRowsBlockArray arCoBlocks;
  1935. GetCollapsableBlocksInfo(nRow, arCoBlocks);
  1936. int nCount = (int)arCoBlocks.GetSize();
  1937. for (int i = 0; i < nCount; i++)
  1938. {
  1939. XTP_EDIT_ROWSBLOCK coBlk = arCoBlocks[i];
  1940. if (coBlk.lcStart.nLine == nRow)
  1941. {
  1942. XTP_EDIT_COLLAPSEDBLOCK* pCoDrawBlk = (XTP_EDIT_COLLAPSEDBLOCK*)LMCoParam.GetPtr();
  1943. if (!pCoDrawBlk)
  1944. {
  1945. pCoDrawBlk = new XTP_EDIT_COLLAPSEDBLOCK;
  1946. if (!pCoDrawBlk)
  1947. {
  1948. return;
  1949. }
  1950. LMCoParam.SetPtr(pCoDrawBlk, XTPSECollapsedBlockDeleteFn);
  1951. }
  1952. pCoDrawBlk->collBlock = coBlk;
  1953. pMgr->SetLineMark(nRow, xtpEditLMT_Collapsed, &LMCoParam);
  1954. m_arCollapsedTextRows.SetAtGrow(m_nCollapsedTextRowsCount, nRow);
  1955. nLineEnd = min(nLineEnd, pCoDrawBlk->collBlock.lcEnd.nCol);
  1956. m_nCollapsedTextRowsCount++;
  1957. }
  1958. }
  1959. }
  1960. SetCurrentDocumentRow(nRow);
  1961. m_nDispCol = nLineEnd + 1;
  1962. SetCurCaretPos(nRow, m_nDispCol);
  1963. Invalidate(FALSE);
  1964. RecalcScrollBars();
  1965. }
  1966. void CXTPSyntaxEditCtrl::DeleteBookmark(int nRow)
  1967. {
  1968. if (GetLineMarksManager())
  1969. {
  1970. GetLineMarksManager()->DeleteLineMark(nRow, xtpEditLMT_Bookmark);
  1971. }
  1972. }
  1973. void CXTPSyntaxEditCtrl::PrevBookmark()
  1974. {
  1975. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  1976. if (!pMgr)
  1977. {
  1978. ASSERT(FALSE);
  1979. return;
  1980. }
  1981. int nRow = GetCurRow();
  1982. int nVisRow = GetCurrentVisibleRow();
  1983. int nPrevRow = nRow;
  1984. do {
  1985. pMgr->FindPrevLineMark(--nPrevRow, xtpEditLMT_Bookmark);
  1986. } while (nPrevRow > 0 && GetVisibleRow(nPrevRow) == nVisRow);
  1987. if (nPrevRow < 0)
  1988. {
  1989. POSITION posLast = pMgr->GetLastLineMark(xtpEditLMT_Bookmark);
  1990. XTP_EDIT_LMDATA* pData = pMgr->GetLineMarkAt(posLast, xtpEditLMT_Bookmark);
  1991. nPrevRow = pData ? pData->m_nRow : -1;
  1992. }
  1993. if (GetVisibleRow(nPrevRow) == nVisRow)
  1994. {
  1995. nPrevRow = -1;
  1996. }
  1997. if (nPrevRow >= 0)
  1998. {
  1999. SetCurPos(nPrevRow, 1);
  2000. Invalidate(FALSE);
  2001. }
  2002. }
  2003. void CXTPSyntaxEditCtrl::NextBookmark()
  2004. {
  2005. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2006. if (!pMgr)
  2007. {
  2008. ASSERT(FALSE);
  2009. return;
  2010. }
  2011. int nRow = GetCurRow();
  2012. int nVisRow = GetCurrentVisibleRow();
  2013. // move down to the next bookmark after the current collapsed block area
  2014. int nNextRow = nRow;
  2015. do {
  2016. pMgr->FindNextLineMark(++nNextRow, xtpEditLMT_Bookmark);
  2017. } while (nNextRow > 0 && GetVisibleRow(nNextRow) == nVisRow);
  2018. if (nNextRow < 0)
  2019. {
  2020. // find first line mark
  2021. POSITION posFirst = pMgr->GetFirstLineMark(xtpEditLMT_Bookmark);
  2022. XTP_EDIT_LMDATA* pData = pMgr->GetNextLineMark(posFirst, xtpEditLMT_Bookmark);
  2023. nNextRow = pData ? pData->m_nRow : -1;
  2024. }
  2025. if (GetVisibleRow(nNextRow) == nVisRow)
  2026. {
  2027. nNextRow = -1;
  2028. }
  2029. if (nNextRow >= 0)
  2030. {
  2031. SetCurPos(nNextRow, 1);
  2032. Invalidate(FALSE);
  2033. }
  2034. }
  2035. BOOL CXTPSyntaxEditCtrl::HasRowMark(int nRow, const XTP_EDIT_LINEMARKTYPE& lmType,
  2036. XTP_EDIT_LMPARAM* pParam)
  2037. {
  2038. if (GetLineMarksManager())
  2039. {
  2040. return GetLineMarksManager()->HasRowMark(nRow, lmType, pParam);
  2041. }
  2042. return FALSE;
  2043. }
  2044. void CXTPSyntaxEditCtrl::DeleteBreakpoint(int nRow)
  2045. {
  2046. if (GetLineMarksManager())
  2047. {
  2048. GetLineMarksManager()->DeleteLineMark(nRow, xtpEditLMT_Breakpoint);
  2049. }
  2050. }
  2051. void CXTPSyntaxEditCtrl::RefreshLineMarks(XTP_EDIT_NMHDR_EDITCHANGED* pEditChanged)
  2052. {
  2053. if (!GetLineMarksManager())
  2054. {
  2055. ASSERT(FALSE);
  2056. return ;
  2057. }
  2058. int eRefreshType = xtpEditLMRefresh_Unknown;
  2059. if (pEditChanged->nAction & XTP_EDIT_EDITACTION_DELETEROW)
  2060. {
  2061. eRefreshType = xtpEditLMRefresh_Delete;
  2062. }
  2063. else if (pEditChanged->nAction & XTP_EDIT_EDITACTION_INSERTROW)
  2064. {
  2065. eRefreshType = xtpEditLMRefresh_Insert;
  2066. if (pEditChanged->nAction & XTP_EDIT_EDITACTION_INSERTROW_NEW)
  2067. eRefreshType |= xtpEditLMRefresh_InsertAt0;
  2068. }
  2069. else
  2070. {
  2071. return;
  2072. }
  2073. GetLineMarksManager()->RefreshLineMarks(pEditChanged->nRowFrom,
  2074. pEditChanged->nRowTo,  eRefreshType);
  2075. RedrawLineMarks();
  2076. }
  2077. void CXTPSyntaxEditCtrl::RedrawLineMarks()
  2078. {
  2079. Invalidate(FALSE);
  2080. }
  2081. BOOL CXTPSyntaxEditCtrl::HasBookmarks()
  2082. {
  2083. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2084. if (pMgr)
  2085. {
  2086. int nCount = pMgr->GetCount(xtpEditLMT_Bookmark);
  2087. return nCount > 0;
  2088. }
  2089. return FALSE;
  2090. }
  2091. BOOL CXTPSyntaxEditCtrl::HasBreakpoints()
  2092. {
  2093. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2094. if (pMgr)
  2095. {
  2096. int nCount = pMgr->GetCount(xtpEditLMT_Breakpoint);
  2097. return nCount > 0;
  2098. }
  2099. return FALSE;
  2100. }
  2101. CXTPSyntaxEditLineMarksManager* CXTPSyntaxEditCtrl::GetLineMarksManager()
  2102. {
  2103. ASSERT(m_pBuffer);
  2104. return m_pBuffer ? m_pBuffer->GetLineMarksManager() : NULL;
  2105. }
  2106. CString CXTPSyntaxEditCtrl::GetModulePath()
  2107. {
  2108. TCHAR zsFileName[_MAX_PATH];
  2109. //DWORD dwRes = ::GetModuleFileName(NULL, zsFileName, _MAX_PATH);
  2110. DWORD dwRes = ::GetModuleFileName(AfxGetInstanceHandle(), zsFileName, _MAX_PATH);
  2111. ASSERT(dwRes);
  2112. CString csFilePath = zsFileName;
  2113. int nFLs = csFilePath.ReverseFind(_T('\'));
  2114. if (nFLs > 0)
  2115. {
  2116. csFilePath.ReleaseBuffer(nFLs + 1);
  2117. }
  2118. return csFilePath;
  2119. }
  2120. CString CXTPSyntaxEditCtrl::GetDefaultCfgFilePath()
  2121. {
  2122. return m_strDefaultCfgFilePath;
  2123. }
  2124. void CXTPSyntaxEditCtrl::RefreshColors()
  2125. {
  2126. m_arOnScreenSchCache.RemoveAll();
  2127. if (!m_pBuffer)
  2128. {
  2129. ASSERT(FALSE);
  2130. return;
  2131. }
  2132. if (GetSyntaxColor())
  2133. {
  2134. XTP_EDIT_LINECOL pos1_0 = {1,0};
  2135. BOOL bParseInThread = m_pBuffer->GetLexParser()->GetSchemaOptions(
  2136. m_pBuffer->GetFileExt() )->m_bFirstParseInSeparateThread;
  2137. if (bParseInThread)
  2138. {
  2139. m_pBuffer->GetLexParser()->StartParseInThread(m_pBuffer, &pos1_0, NULL, 0, TRUE);
  2140. }
  2141. else
  2142. {
  2143. CXTPSyntaxEditLexTextSchema* ptrTextSch = m_pBuffer->GetLexParser()->GetTextSchema();
  2144. if (ptrTextSch)
  2145. {
  2146. CXTPSyntaxEditTextIterator txtIter(m_pBuffer);
  2147. ptrTextSch->RunParseUpdate(TRUE, &txtIter, &pos1_0, NULL);
  2148. }
  2149. }
  2150. }
  2151. else
  2152. {
  2153. GetLineMarksManager()->RemoveAll(xtpEditLMT_Collapsed);
  2154. }
  2155. // Load data into AC
  2156. SetAutoCompleteList();
  2157. Invalidate(FALSE);
  2158. }
  2159. CString CXTPSyntaxEditCtrl::GetConfigFile()
  2160. {
  2161. if (!m_pBuffer)
  2162. return _T("");
  2163. return m_pBuffer->GetConfigFile();
  2164. }
  2165. BOOL CXTPSyntaxEditCtrl::SetConfigFile(LPCTSTR szPath)
  2166. {
  2167. m_strDefaultCfgFilePath = szPath;
  2168. if (!m_pBuffer || !FILEEXISTS_S(szPath))
  2169. {
  2170. return FALSE;
  2171. }
  2172. m_pBuffer->SetConfigFile(szPath);
  2173. return TRUE;
  2174. }
  2175. const CString& CXTPSyntaxEditCtrl::GetCurrentTheme()
  2176. {
  2177. if (!m_pBuffer || !m_pBuffer->GetLexConfigurationManager() || !GetSyntaxColor())
  2178. {
  2179. static CString strEmpty;
  2180. return strEmpty;
  2181. }
  2182. return m_pBuffer->GetLexConfigurationManager()->GetCurrentTheme();
  2183. }
  2184. CStringArray& CXTPSyntaxEditCtrl::GetThemes()
  2185. {
  2186. if (!m_pBuffer || !m_pBuffer->GetLexConfigurationManager())
  2187. {
  2188. static CStringArray s_ar;
  2189. return s_ar;
  2190. }
  2191. return m_pBuffer->GetLexConfigurationManager()->GetThemeManager().GetThemes();
  2192. }
  2193. void CXTPSyntaxEditCtrl::ApplyTheme(CString strTheme)
  2194. {
  2195. if (!m_pBuffer || !m_pBuffer->GetLexConfigurationManager() || !GetSyntaxColor())
  2196. {
  2197. return;
  2198. }
  2199. CXTPSyntaxEditLexTextSchema* ptrTextSch = m_pBuffer->GetLexParser()->GetTextSchema();
  2200. m_pBuffer->GetLexConfigurationManager()->SetTheme(strTheme, ptrTextSch);
  2201. m_arOnScreenSchCache.RemoveAll();
  2202. Invalidate(FALSE);
  2203. UpdateWindow();
  2204. }
  2205. void CXTPSyntaxEditCtrl::OnBeforeEditChanged(int nRow, int nCol)
  2206. {
  2207. UNREFERENCED_PARAMETER(nRow); UNREFERENCED_PARAMETER(nCol);
  2208. if (GetSyntaxColor())
  2209. {
  2210. m_pBuffer->GetLexParser()->OnBeforeEditChanged();
  2211. }
  2212. }
  2213. void CXTPSyntaxEditCtrl::OnEditChanged(const XTP_EDIT_LINECOL& LCFrom,
  2214.   const XTP_EDIT_LINECOL& LCTo, int eEditAction)
  2215. {
  2216. OnEditChanged(LCFrom.nLine, LCFrom.nCol, LCTo.nLine, LCTo.nCol, eEditAction);
  2217. }
  2218. void CXTPSyntaxEditCtrl::OnEditChanged(int nRowFrom, int nColFrom,
  2219.   int nRowTo, int nColTo, int eEditAction)
  2220. {
  2221. InvalidateRows(nRowFrom);
  2222. if (GetSyntaxColor())
  2223. {
  2224. XTP_EDIT_LINECOL posFrom = {nRowFrom, max(0, nColFrom-1) };
  2225. XTP_EDIT_LINECOL posTo = {nRowTo, max(0, nColTo-1) };
  2226. m_pBuffer->GetLexParser()->OnEditChanged(posFrom, posTo, eEditAction, m_pBuffer);
  2227. }
  2228. }
  2229. BOOL CXTPSyntaxEditCtrl::NotifyParseEvent(XTP_NOTIFY_CODE code, WPARAM wParam, LPARAM lParam)
  2230. {
  2231. // Notify the parent class that position has been changed
  2232. XTP_EDIT_NMHDR_PARSEEVENT pe;
  2233. // NMHDR codes
  2234. pe.nmhdr.code = XTP_EDIT_NM_PARSEEVENT;
  2235. pe.nmhdr.hwndFrom = m_hWnd;
  2236. pe.nmhdr.idFrom = GetDlgCtrlID();
  2237. // notification event code.
  2238. pe.code   = code;
  2239. pe.wParam = wParam;
  2240. pe.lParam = lParam;
  2241. // Notify the parent window
  2242. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  2243. {
  2244. return (BOOL)m_pParentWnd->SendMessage(
  2245. WM_NOTIFY, (WPARAM)pe.nmhdr.idFrom, (LPARAM)&pe);
  2246. }
  2247. return FALSE;
  2248. }
  2249. void CXTPSyntaxEditCtrl::OnParseEvent(XTP_NOTIFY_CODE code, WPARAM wParam, LPARAM lParam)
  2250. {
  2251. NotifyParseEvent(code, wParam, lParam);
  2252. //-----------------------------------------------------------------------
  2253. int nRSFactor = 3;
  2254. int nAveRedrawTime = m_aveRedrawScreenTime.GetAverageValue(333);
  2255. int nUpdateTimeOut = max(1000, nAveRedrawTime * nRSFactor);
  2256. BOOL bRedraw = FALSE;
  2257. BOOL bRedrawTimeOut = TRUE;
  2258. if (code == xtpEditOnParserStarted)
  2259. {
  2260. DWORD dwTime = ::GetTickCount();
  2261. m_dwLastRedrawTime = dwTime;
  2262. }
  2263. else if (code == xtpEditOnTextBlockParsed)
  2264. {
  2265. CXTPSyntaxEditLexTextBlock* pTBended = (CXTPSyntaxEditLexTextBlock*)wParam;
  2266. DWORD dwTime = ::GetTickCount();
  2267. bRedrawTimeOut = labs(dwTime - m_dwLastRedrawTime) >= nUpdateTimeOut;
  2268. if (bRedrawTimeOut)
  2269. {
  2270. m_dwLastRedrawTime = dwTime;
  2271. }
  2272. if (pTBended)
  2273. {
  2274. int nDispMax = GetRowPerPage() - 1;
  2275. int nBottomRow = GetDocumentRow(nDispMax+5);
  2276. if (!(pTBended->m_PosStartLC.nLine > nBottomRow ||
  2277. pTBended->m_PosEndLC.nLine < m_nTopRow) )
  2278. {
  2279. bRedraw = TRUE;
  2280. }
  2281. ClearOnScreenSchCache(pTBended->m_PosStartLC.nLine);
  2282. }
  2283. }
  2284. else if (code == xtpEditOnParserEnded)
  2285. {
  2286. bRedraw = TRUE;
  2287. }
  2288. //-----------------------------
  2289. if (bRedraw)
  2290. {
  2291. KillTimer(TIMER_REDRAW_WHEN_PARSE);
  2292. if (bRedrawTimeOut)
  2293. {
  2294. Invalidate(FALSE);
  2295. UpdateWindow();
  2296. }
  2297. else
  2298. {
  2299. SetTimer(TIMER_REDRAW_WHEN_PARSE, nUpdateTimeOut*5, NULL);
  2300. }
  2301. }
  2302. }
  2303. //////////////////////////////////////////////////////////////////////////
  2304. // Row / col functions
  2305. int CXTPSyntaxEditCtrl::GetTopRow()
  2306. {
  2307. return m_nTopRow;
  2308. }
  2309. void CXTPSyntaxEditCtrl::SetTopRow(int nRow)
  2310. {
  2311. if (nRow < 1)
  2312. m_nTopRow = 1;
  2313. else if (nRow > GetRowCount())
  2314. m_nTopRow = max (1, GetRowCount());
  2315. else
  2316. m_nTopRow = max (1, nRow);
  2317. }
  2318. int CXTPSyntaxEditCtrl::GetCurRow() const
  2319. {
  2320. return GetCurrentDocumentRow();
  2321. }
  2322. int CXTPSyntaxEditCtrl::GetCurCol()
  2323. {
  2324. return m_nDispCol;
  2325. }
  2326. int CXTPSyntaxEditCtrl::GetCurAbsCol()
  2327. {
  2328. return m_nCurrentCol;
  2329. }
  2330. void CXTPSyntaxEditCtrl::GoToRow(int iRow, BOOL bSelectRow/* = FALSE*/)
  2331. {
  2332. CString csLineText = GetLineText(iRow);
  2333. m_nCurrentCol = 1;
  2334. m_nDispCol = 1;
  2335. if (bSelectRow)
  2336. {
  2337. m_Selection.Reset_disp(iRow, 1);
  2338. m_Selection.SetEnd_disp(iRow+1, 1);
  2339. SetCurrentDocumentRow(iRow);
  2340. Invalidate(FALSE);
  2341. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, TRUE, TRUE);
  2342. //SetTimer(TIMER_SELECTION_ID, TIMER_SELECTION_TIME, NULL);
  2343. }
  2344. else
  2345. {
  2346. SetCurrentDocumentRow(iRow);
  2347. SetCurPos(iRow, 1);
  2348. }
  2349. }
  2350. int CXTPSyntaxEditCtrl::GetCurrentVisibleRow()
  2351. {
  2352. return GetVisibleRow(GetCurrentDocumentRow());
  2353. }
  2354. int CXTPSyntaxEditCtrl::GetCurrentDocumentRow() const
  2355. {
  2356. return m_nCurrentDocumentRow;
  2357. }
  2358. void CXTPSyntaxEditCtrl::SetCurrentDocumentRow(int nRow)
  2359. {
  2360. if (nRow > GetRowCount())
  2361. nRow = GetRowCount();
  2362. if (nRow < 1)
  2363. nRow = 1;
  2364. m_nCurrentDocumentRow = nRow;
  2365. }
  2366. int CXTPSyntaxEditCtrl::GetDocumentRow(int nVisibleRow)
  2367. {
  2368. return CalculateDocumentRow(m_nTopRow, nVisibleRow);
  2369. }
  2370. int CXTPSyntaxEditCtrl::CalculateDocumentRow(int nStartDocumentRow, int nRowDelta)
  2371. {
  2372. int nDocRow = nStartDocumentRow + nRowDelta - 1;
  2373. int nNextCollapsedRow = nStartDocumentRow - 1;
  2374. POSITION pos = GetLineMarksManager()->FindNextLineMark(nNextCollapsedRow, xtpEditLMT_Collapsed);
  2375. while (pos != NULL)
  2376. {
  2377. XTP_EDIT_LMDATA* pData = GetLineMarksManager()->GetNextLineMark(pos, xtpEditLMT_Collapsed);
  2378. if (pData && pData->m_nRow >= nNextCollapsedRow) // mark should be not within previous collapsed block
  2379. {
  2380. if (pData->m_nRow >= nDocRow) // finish if mark is greater then row to calculate for
  2381. break;
  2382. XTP_EDIT_COLLAPSEDBLOCK* pCoDBlk = (XTP_EDIT_COLLAPSEDBLOCK*)pData->m_Param.GetPtr();
  2383. if (!pCoDBlk)
  2384. continue;
  2385. int nHiddenRows = pCoDBlk->collBlock.lcEnd.nLine - pCoDBlk->collBlock.lcStart.nLine;
  2386. nDocRow += nHiddenRows;
  2387. nNextCollapsedRow = pData->m_nRow + nHiddenRows;
  2388. }
  2389. }
  2390. return nDocRow;
  2391. }
  2392. int CXTPSyntaxEditCtrl::GetVisibleRow(int nDocumentRow)
  2393. {
  2394. return CalculateVisibleRow(m_nTopRow, nDocumentRow);
  2395. }
  2396. int CXTPSyntaxEditCtrl::CalculateVisibleRow(int nStartDocumentRow, int nDocumentRow)
  2397. {
  2398. int nVisRow = nDocumentRow - nStartDocumentRow + 1;
  2399. int nNextCollapsedRow = nStartDocumentRow - 1;
  2400. const int nRowCount = GetRowCount();
  2401. POSITION pos = GetLineMarksManager()->FindNextLineMark(nNextCollapsedRow, xtpEditLMT_Collapsed);
  2402. while (pos != NULL)
  2403. {
  2404. XTP_EDIT_LMDATA* pData = GetLineMarksManager()->GetNextLineMark(pos, xtpEditLMT_Collapsed);
  2405. if (pData && pData->m_nRow >= nNextCollapsedRow) // mark should be not within previous collapsed block
  2406. {
  2407. if (pData->m_nRow >= nDocumentRow) // finish if mark is greater then row to calculate for
  2408. break;
  2409. XTP_EDIT_COLLAPSEDBLOCK* pCoDBlk = (XTP_EDIT_COLLAPSEDBLOCK*)pData->m_Param.GetPtr();
  2410. if (!pCoDBlk)
  2411. continue;
  2412. int nHiddenRows = min(nRowCount, pCoDBlk->collBlock.lcEnd.nLine) -
  2413. pCoDBlk->collBlock.lcStart.nLine;
  2414. nVisRow -= nHiddenRows;
  2415. nNextCollapsedRow = pData->m_nRow + nHiddenRows;
  2416. }
  2417. }
  2418. return max(nVisRow, 1);
  2419. }
  2420. int CXTPSyntaxEditCtrl::MoveCurrentVisibleRowUp(int nCount)
  2421. {
  2422. int nOldVisRow = GetCurrentVisibleRow();
  2423. int nOldDocRow = GetCurrentDocumentRow();
  2424. int nNewVisRow = nOldVisRow - nCount;
  2425. // do not shift a window up
  2426. if (nNewVisRow > 0)
  2427. {
  2428. SetCurrentDocumentRow(GetDocumentRow(nNewVisRow));
  2429. return nNewVisRow;
  2430. }
  2431. // find global visible row index
  2432. int nVisRow = CalculateVisibleRow(1, nOldDocRow);
  2433. int nTargetDocRow = CalculateDocumentRow(1, max(nVisRow - nCount, 1));
  2434. int nDocDiff = nOldDocRow - nTargetDocRow;
  2435. nDocDiff = min(nDocDiff, m_nTopRow - 1);
  2436. nOldDocRow -= nDocDiff;
  2437. m_nCurrentDocumentRow -= nDocDiff;
  2438. m_nTopRow -= nDocDiff;
  2439. nNewVisRow = nOldVisRow - nDocDiff + GetRowPerPage();
  2440. return GetCurrentVisibleRow();
  2441. }
  2442. int CXTPSyntaxEditCtrl::MoveCurrentVisibleRowDown(int nCount)
  2443. {
  2444. int nOldRow = GetCurrentVisibleRow();
  2445. int nNewDocRow = min(GetDocumentRow(nOldRow + nCount), GetRowCount());
  2446. SetCurrentDocumentRow(nNewDocRow);
  2447. int nMaxVisRow = GetRowPerPage();
  2448. int nMaxDocRow = GetDocumentRow(nMaxVisRow);
  2449. if (nNewDocRow > nMaxDocRow)
  2450. {
  2451. m_nTopRow = max (1, GetDocumentRow(nCount + 1));
  2452. }
  2453. return GetCurrentVisibleRow();
  2454. }
  2455. int CXTPSyntaxEditCtrl::GetRowPerPage()
  2456. {
  2457. return m_DrawTextProcessor.GetRowsCount(FALSE);
  2458. }
  2459. int CXTPSyntaxEditCtrl::GetVisibleRowsCount(int nMaxDocRow)
  2460. {
  2461. if (nMaxDocRow < 0)
  2462. nMaxDocRow = GetRowCount();
  2463. int nVisRow = nMaxDocRow;
  2464. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2465. if (pMgr)
  2466. {
  2467. int nNextCollapsedRow = -1;
  2468. POSITION pos = pMgr->GetFirstLineMark(xtpEditLMT_Collapsed);
  2469. while (pos != NULL)
  2470. {
  2471. XTP_EDIT_LMDATA* pData = pMgr->GetNextLineMark(pos, xtpEditLMT_Collapsed);
  2472. if (pData && pData->m_nRow >= nNextCollapsedRow) // mark should be not within previous collapsed block
  2473. {
  2474. if (pData->m_nRow >= nMaxDocRow) // finish if mark is greater then row to calculate for
  2475. break;
  2476. XTP_EDIT_COLLAPSEDBLOCK* pCoDBlk = (XTP_EDIT_COLLAPSEDBLOCK*)pData->m_Param.GetPtr();
  2477. if (!pCoDBlk)
  2478. continue;
  2479. int nHiddenRows = pCoDBlk->collBlock.lcEnd.nLine - pCoDBlk->collBlock.lcStart.nLine;
  2480. nVisRow -= nHiddenRows;
  2481. nNextCollapsedRow = pData->m_nRow + nHiddenRows;
  2482. }
  2483. }
  2484. }
  2485. return nVisRow;
  2486. }
  2487. BOOL CXTPSyntaxEditCtrl::ShiftCurrentVisibleRowUp(int nCount, BOOL bChangeCaret)
  2488. {
  2489. BOOL bChanged = FALSE;
  2490. if (m_nTopRow > 1)
  2491. {
  2492. int nOldTopRow = m_nTopRow;
  2493. int nDocRow = GetCurrentDocumentRow();      // store old document row
  2494. SetCurrentDocumentRow(GetDocumentRow(1));   // move to first visible row
  2495. MoveCurrentVisibleRowUp(nCount);            // move nCount row up
  2496. // move to old document row
  2497. int nBottomDocRow = GetDocumentRow(GetRowPerPage());
  2498. if (bChangeCaret)
  2499. nDocRow = min(nDocRow, nBottomDocRow);
  2500. SetCurrentDocumentRow(nDocRow);
  2501. if (nDocRow > nBottomDocRow)
  2502. CWnd::HideCaret();
  2503. if (nOldTopRow != m_nTopRow)
  2504. bChanged = TRUE;
  2505. }
  2506. return bChanged;
  2507. }
  2508. BOOL CXTPSyntaxEditCtrl::ShiftCurrentVisibleRowDown(int nCount, BOOL bChangeCaret)
  2509. {
  2510. BOOL bChanged = FALSE;
  2511. if (m_nTopRow < GetRowCount())
  2512. {
  2513. int nOldTopRow = m_nTopRow;
  2514. int nDocRow = GetCurrentDocumentRow();
  2515. int nLastVisRow = GetRowPerPage();
  2516. SetCurrentDocumentRow(GetDocumentRow(nLastVisRow));
  2517. MoveCurrentVisibleRowDown(nCount);
  2518. int nTopDocRow = GetDocumentRow(1);
  2519. if (bChangeCaret)
  2520. nDocRow = max(nDocRow, nTopDocRow);
  2521. SetCurrentDocumentRow(nDocRow);
  2522. if (nDocRow < nTopDocRow)
  2523. CWnd::HideCaret();
  2524. if (nOldTopRow != m_nTopRow)
  2525. bChanged = TRUE;
  2526. }
  2527. return bChanged;
  2528. }
  2529. BOOL CXTPSyntaxEditCtrl::GetCollapsedBlockLen(int nStartRow, int& rnLen)
  2530. {
  2531. rnLen = 0;
  2532. XTP_EDIT_LMPARAM LMCoParam;
  2533. BOOL bIsCollapsed = HasRowMark(nStartRow, xtpEditLMT_Collapsed, &LMCoParam);
  2534. if (!bIsCollapsed)
  2535. {
  2536. return FALSE;
  2537. }
  2538. XTP_EDIT_COLLAPSEDBLOCK* pCoDBlk = (XTP_EDIT_COLLAPSEDBLOCK*)LMCoParam.GetPtr();
  2539. if (!pCoDBlk)
  2540. {
  2541. ASSERT(FALSE);
  2542. return FALSE;
  2543. }
  2544. rnLen = pCoDBlk->collBlock.lcEnd.nLine - pCoDBlk->collBlock.lcStart.nLine;
  2545. return TRUE;
  2546. }
  2547. //////////////////////////////////////////////////////////////////////////
  2548. // Line text internal functions
  2549. const CString& CXTPSyntaxEditCtrl::GetLineText(int nRow, BOOL bAddCRLF, int iCRLFStyle)
  2550. {
  2551. GetLineText(nRow, m_strTmpLineTextBuff, bAddCRLF, iCRLFStyle);
  2552. return m_strTmpLineTextBuff;
  2553. }
  2554. void CXTPSyntaxEditCtrl::GetLineText(int nRow, CString& strBuffer, BOOL bAddCRLF, int iCRLFStyle)
  2555. {
  2556. m_pBuffer->GetLineText(nRow, strBuffer, bAddCRLF, iCRLFStyle);
  2557. XTP_EDIT_LMPARAM LMCoParam;
  2558. BOOL bIsCollapsed = HasRowMark(nRow, xtpEditLMT_Collapsed, &LMCoParam);
  2559. if (bIsCollapsed)
  2560. {
  2561. XTP_EDIT_COLLAPSEDBLOCK* pCoDBlk = (XTP_EDIT_COLLAPSEDBLOCK*)LMCoParam.GetPtr();
  2562. if (!pCoDBlk)
  2563. {
  2564. ASSERT(FALSE);
  2565. return;
  2566. }
  2567. if (strBuffer.GetLength() > pCoDBlk->collBlock.lcStart.nCol)
  2568. {
  2569. strBuffer.SetAt(pCoDBlk->collBlock.lcStart.nCol, 0);
  2570. }
  2571. }
  2572. }
  2573. void CXTPSyntaxEditCtrl::SetAutoCompleteList()
  2574. {
  2575. CXTPSyntaxEditLexTokensDefArray arTokens;
  2576. m_pBuffer->GetLexParser()->GetTokensForAutoCompleate(arTokens, FALSE);
  2577. CXTPSyntaxEditACDataArray arrData;
  2578. CXTPSyntaxEditLexTokensDef TokenDef;
  2579. int nTokensDef = (int)arTokens.GetSize();
  2580. if (!nTokensDef)
  2581. return;
  2582. for (int nTokenDef = 0; nTokenDef < nTokensDef; nTokenDef++)
  2583. {
  2584. TokenDef = (arTokens.GetAt(nTokenDef));
  2585. int nTags = (int)TokenDef.m_arTokens.GetSize();
  2586. for (int nTag = 0; nTag < nTags; nTag++)
  2587. {
  2588. arrData.Add(new XTP_EDIT_ACDATA(0, TokenDef.m_arTokens.GetAt(nTag)));
  2589. }
  2590. int nDelims = (int)TokenDef.m_arStartSeps.GetSize();
  2591. int nDelim;
  2592. for (nDelim = 0; nDelim < nDelims; nDelim++)
  2593. {
  2594. if (m_pAutoComplete->m_strDelims.Find(TokenDef.m_arStartSeps.GetAt(nDelim)) < 0)
  2595. m_pAutoComplete->m_strDelims += TokenDef.m_arStartSeps.GetAt(nDelim);
  2596. }
  2597. nDelims = (int)TokenDef.m_arEndSeps.GetSize();
  2598. for (nDelim = 0; nDelim < nDelims; nDelim++)
  2599. {
  2600. if (m_pAutoComplete->m_strDelims.Find(TokenDef.m_arEndSeps.GetAt(nDelim)) < 0)
  2601. m_pAutoComplete->m_strDelims += TokenDef.m_arEndSeps.GetAt(nDelim);
  2602. }
  2603. }
  2604. m_pAutoComplete->SetList(arrData);
  2605. }
  2606. BOOL CXTPSyntaxEditCtrl::PreTranslateMessage(MSG* pMsg)
  2607. {
  2608. if (m_pAutoComplete->IsActive())
  2609. {
  2610. switch (pMsg->message)
  2611. {
  2612. case WM_KEYDOWN:
  2613. case WM_KEYUP:
  2614. case WM_CHAR:
  2615. case WM_DEADCHAR:
  2616. {
  2617. m_pAutoComplete->SendMessage(
  2618. pMsg->message, pMsg->wParam, pMsg->lParam);
  2619. }
  2620. break;
  2621. }
  2622. if (pMsg->message == WM_KEYDOWN)
  2623. {
  2624. switch (pMsg->wParam)
  2625. {
  2626. case VK_SPACE:
  2627. case VK_PRIOR:
  2628. case VK_NEXT:
  2629. case VK_END:
  2630. case VK_HOME:
  2631. case VK_LEFT:
  2632. case VK_UP:
  2633. case VK_RIGHT:
  2634. case VK_DOWN:
  2635. case VK_SELECT:
  2636. case VK_PRINT:
  2637. case VK_EXECUTE:
  2638. case VK_SNAPSHOT:
  2639. case VK_INSERT:
  2640. case VK_DELETE:
  2641. case VK_HELP:
  2642. case VK_RETURN:
  2643. {
  2644. return TRUE;
  2645. }
  2646. }
  2647. }
  2648. }
  2649. else if (m_bEnableEditAccelerators &&
  2650.  (pMsg->message == WM_KEYDOWN || pMsg->message == WM_SYSKEYDOWN))
  2651. {
  2652. BOOL bAltKey  = (::GetKeyState(VK_MENU) & KF_UP) != 0;
  2653. BOOL bCtrlKey  = (::GetKeyState(VK_CONTROL) & KF_UP) != 0;
  2654. BOOL bShiftKey  = (::GetKeyState(VK_SHIFT) & KF_UP) != 0;
  2655. BOOL bAlt   = bAltKey && !bCtrlKey && !bShiftKey;
  2656. BOOL bCtrl  = bCtrlKey && !bAltKey && !bShiftKey;
  2657. BOOL bShift = bShiftKey && !bAltKey && !bCtrlKey;
  2658. BOOL bCtrlShift = bCtrlKey && bShiftKey && !bAltKey;
  2659. BOOL bProcessed = TRUE;
  2660. if (bCtrlKey && pMsg->wParam == (BYTE)'X')
  2661. {
  2662. if (IsSelectionExist() && CanEditDoc())
  2663. Cut();
  2664. }
  2665. else if (bCtrl && pMsg->wParam == (BYTE)'C' ||
  2666.  bCtrl && pMsg->wParam == VK_INSERT)
  2667. {
  2668. if (IsSelectionExist())
  2669. Copy();
  2670. }
  2671. else if (bCtrl && pMsg->wParam == (BYTE)'V' ||
  2672.  bShift && pMsg->wParam == VK_INSERT)
  2673. {
  2674. if (CanEditDoc())
  2675. Paste();
  2676. }
  2677. else if (bAlt && pMsg->wParam == VK_BACK ||
  2678.  bCtrl && pMsg->wParam == (BYTE)'Z')
  2679. {
  2680. if (CanUndo())
  2681. Undo();
  2682. }
  2683. else if (bCtrl && pMsg->wParam == (BYTE)'Y' ||
  2684.  bCtrlShift && pMsg->wParam == (BYTE)'Z')
  2685. {
  2686. if (CanRedo())
  2687. Redo();
  2688. }
  2689. else if (bCtrl && pMsg->wParam == (BYTE)'A')
  2690. {
  2691. SelectAll();
  2692. }
  2693. else
  2694. {
  2695. bProcessed = FALSE;
  2696. }
  2697. if (bProcessed)
  2698. return TRUE;
  2699. }
  2700. return CWnd::PreTranslateMessage(pMsg);
  2701. }
  2702. BOOL CXTPSyntaxEditCtrl::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
  2703. {
  2704. if (m_pAutoComplete->IsActive())
  2705. {
  2706. return (BOOL)m_pAutoComplete->SendMessage(WM_MOUSEWHEEL,
  2707. MAKEWPARAM(nFlags, zDelta), MAKELPARAM(pt.x, pt.y));
  2708. //CXTPWindowRect rcWndAC(&m_wndAutoComplete);
  2709. //if (rcWndAC.PtInRect(pt))
  2710. //  return (BOOL)m_pAutoComplete->SendMessage(WM_MOUSEWHEEL, MAKEWPARAM(nFlags, zDelta), MAKELPARAM(pt.x, pt.y));
  2711. //else
  2712. //  m_pAutoComplete->Hide();
  2713. }
  2714. m_pToolTip->Hide();
  2715. if (zDelta < 0)
  2716. Scroll(0, m_nWheelScroll);
  2717. else
  2718. Scroll(0, -m_nWheelScroll);
  2719. return CWnd::OnMouseWheel(nFlags, zDelta, pt);
  2720. }
  2721. void CXTPSyntaxEditCtrl::CollapseAll()
  2722. {
  2723. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2724. if (!pMgr)
  2725. return;
  2726. CWaitCursor _WC;
  2727. CXTPSyntaxEditRowsBlockArray arCoBlocks;
  2728. GetCollapsableBlocksInfo(-1, arCoBlocks);
  2729. for (int nRow = 0; nRow < GetRowCount(); nRow++)
  2730. {
  2731. XTP_EDIT_LMPARAM LMCoParam;
  2732. int toRemove = -1;
  2733. int nCount = (int)arCoBlocks.GetSize();
  2734. for (int i = 0; i < nCount; i++)
  2735. {
  2736. XTP_EDIT_ROWSBLOCK coBlk = arCoBlocks[i];
  2737. if (coBlk.lcStart.nLine == nRow)
  2738. {
  2739. XTP_EDIT_COLLAPSEDBLOCK* pCoDrawBlk = (XTP_EDIT_COLLAPSEDBLOCK*)LMCoParam.GetPtr();
  2740. if (!pCoDrawBlk)
  2741. {
  2742. pCoDrawBlk = new XTP_EDIT_COLLAPSEDBLOCK;
  2743. if (!pCoDrawBlk)
  2744. {
  2745. return;
  2746. }
  2747. LMCoParam.SetPtr(pCoDrawBlk, XTPSECollapsedBlockDeleteFn);
  2748. }
  2749. pCoDrawBlk->collBlock = coBlk;
  2750. pMgr->SetLineMark(nRow, xtpEditLMT_Collapsed, &LMCoParam);
  2751. m_arCollapsedTextRows.SetAtGrow(m_nCollapsedTextRowsCount, nRow);
  2752. m_nCollapsedTextRowsCount++;
  2753. if (toRemove < 0)
  2754. toRemove = i;
  2755. }
  2756. }
  2757. if (toRemove >= 0)
  2758. arCoBlocks.RemoveAt(0, toRemove+1);
  2759. }
  2760. // refresh picture
  2761. SetCurCaretPos(GetCurrentDocumentRow(), 1);
  2762. UpdateWindow();
  2763. RecalcScrollBars();
  2764. UpdateScrollPos();
  2765. }
  2766. void CXTPSyntaxEditCtrl::ExpandAll()
  2767. {
  2768. CXTPSyntaxEditLineMarksManager* pMgr = GetLineMarksManager();
  2769. if (!pMgr)
  2770. return;
  2771. CWaitCursor _WC;
  2772. pMgr->RemoveAll(xtpEditLMT_Collapsed);
  2773. // refresh picture
  2774. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol);
  2775. UpdateWindow();
  2776. RecalcScrollBars();
  2777. UpdateScrollPos();
  2778. }
  2779. void CXTPSyntaxEditCtrl::GetRowColors(int nTextRow, int nColFrom, int nColTo,
  2780.   const XTP_EDIT_COLORVALUES& clrDefault,
  2781.   CXTPSyntaxEditTextBlockList* rBlocks)
  2782. {
  2783. if (!GetSyntaxColor())
  2784. return;
  2785. CXTPSyntaxEditLexTextSchema* ptrTxtSch = m_pBuffer->GetLexParser()->GetTextSchema();
  2786. if (ptrTxtSch)
  2787. {
  2788. CXTPSyntaxEditLexTextBlock* pScreenSchFirstTB = GetOnScreenSch(nTextRow);
  2789. //CXTPSyntaxEditLexTextBlock* pScreenSchFirstTB = ptrTxtSch->GetBlocks();
  2790. if (pScreenSchFirstTB)
  2791. {
  2792. CXTPSyntaxEditTextIterator txtIter(GetEditBuffer());
  2793. ptrTxtSch->GetRowColors(&txtIter, nTextRow, nColFrom, nColTo,
  2794. clrDefault, rBlocks, NULL, pScreenSchFirstTB);
  2795. }
  2796. }
  2797. }
  2798. void CXTPSyntaxEditCtrl::GetCollapsableBlocksInfo(int nTextRow,
  2799.   CXTPSyntaxEditRowsBlockArray& rarCoBlocks)
  2800. {
  2801. rarCoBlocks.RemoveAll();
  2802. if (GetSyntaxColor())
  2803. {
  2804. CXTPSyntaxEditLexTextSchema* ptrTxtSch = m_pBuffer->GetLexParser()->GetTextSchema();
  2805. if (ptrTxtSch)
  2806. ptrTxtSch->GetCollapsableBlocksInfo(nTextRow, rarCoBlocks, &m_fcCollapsable.ptrTBStart);
  2807. }
  2808. }
  2809. CXTPSyntaxEditLexTextBlock* CXTPSyntaxEditCtrl::GetOnScreenSch(int nForRow)
  2810. {
  2811. int nLifeTime = m_pBuffer->GetLexParser()->GetSchemaOptions(
  2812. m_pBuffer->GetFileExt())->m_dwOnScreenSchCacheLifeTime_sec;
  2813. m_arOnScreenSchCache.RemoveOld(nLifeTime);
  2814. //========================================================================
  2815. int nCount = (int)m_arOnScreenSchCache.GetSize();
  2816. int i;
  2817. for (i = 0; i < nCount; i++)
  2818. {
  2819. CScreenSearchBlock& rSchBlk = m_arOnScreenSchCache[i];
  2820. if (nForRow >= rSchBlk.nRowStart && nForRow <= rSchBlk.nRowEnd)
  2821. {
  2822. rSchBlk.dwLastAccessTime = ::GetTickCount();
  2823. return rSchBlk.ptrTBFirst;
  2824. }
  2825. }
  2826. //---------------------------------------------------------------------------
  2827. CXTPSyntaxEditLexTextSchema* ptrTxtSch = m_pBuffer->GetLexParser()->GetTextSchema();
  2828. if (ptrTxtSch)
  2829. {
  2830. int nRowPerPage = GetRowPerPage()+1;
  2831. CScreenSearchBlock schBlkNew;
  2832. schBlkNew.nRowStart = nForRow;
  2833. schBlkNew.nRowEnd = nForRow + nRowPerPage;
  2834. //--------------------------------------------------------------------
  2835. int nRowStartMin = 1;
  2836. // 1. search next on screen region
  2837. nCount = (int)m_arOnScreenSchCache.GetSize();
  2838. for (i = 0; i < nCount; i++)
  2839. {
  2840. const CScreenSearchBlock& rSchBlk = m_arOnScreenSchCache[i];
  2841. if (rSchBlk.nRowStart > schBlkNew.nRowStart && rSchBlk.nRowStart <= schBlkNew.nRowEnd)
  2842. {
  2843. schBlkNew.nRowEnd = rSchBlk.nRowStart-1;
  2844. }
  2845. if (rSchBlk.nRowEnd+1 > nRowStartMin && rSchBlk.nRowEnd+1 < nForRow)
  2846. {
  2847. nRowStartMin = rSchBlk.nRowEnd+1;
  2848. }
  2849. }
  2850. //--------------------------------------------------------------------
  2851. // 2. search next collapsed block start
  2852. for (i = 0; i < m_nCollapsedTextRowsCount; i++)
  2853. {
  2854. int nCoStartRow = m_arCollapsedTextRows[i];
  2855. if (nCoStartRow >= schBlkNew.nRowStart && nCoStartRow < schBlkNew.nRowEnd)
  2856. {
  2857. schBlkNew.nRowEnd = nCoStartRow;
  2858. }
  2859. }
  2860. //===========================
  2861. if (schBlkNew.nRowEnd - schBlkNew.nRowStart < nRowPerPage/2)
  2862. {
  2863. schBlkNew.nRowStart = max(schBlkNew.nRowStart-nRowPerPage, nRowStartMin);
  2864. }
  2865. //--------------------------------------------------------------------
  2866. CXTPSyntaxEditTextIterator txtIter(m_pBuffer);
  2867. BOOL bParseRes = ptrTxtSch->RunParseOnScreen(&txtIter, schBlkNew.nRowStart,
  2868. schBlkNew.nRowEnd, schBlkNew.ptrTBFirst);
  2869. if (bParseRes)
  2870. {
  2871. schBlkNew.dwLastAccessTime = ::GetTickCount();
  2872. m_arOnScreenSchCache.Add(schBlkNew);
  2873. return schBlkNew.ptrTBFirst;
  2874. }
  2875. }
  2876. return NULL;
  2877. }
  2878. void CXTPSyntaxEditCtrl::InvalidateRows(int nRowFrom, int nRowTo)
  2879. {
  2880. int nCountVDR = (int)m_arValidDispRows.GetSize();
  2881. //-* Invalidate ALL ------------------------------------------------------
  2882. if (nRowFrom < 0 && nRowTo < 0)
  2883. {
  2884. m_arOnScreenSchCache.RemoveAll();
  2885. for (int i = 0; i < nCountVDR; i++) {
  2886. m_arValidDispRows.SetAtGrow(i, 0);
  2887. }
  2888. return ;
  2889. }
  2890. //-* Invalidate Part -----------------------------------------------------
  2891. int nRow1 = nRowFrom;
  2892. int nRow2 = nRowTo;
  2893. if (nRow1 < 0)
  2894. {
  2895. nRow1 = m_nTopRow;
  2896. }
  2897. if (nRow2 < nRow1)
  2898. {
  2899. nRow2 = GetDocumentRow(GetRowPerPage());
  2900. }
  2901. int nDispRow1= CalculateVisibleRow(m_nTopRow, nRow1);
  2902. int nDispRow2 = CalculateVisibleRow(m_nTopRow, nRow2);
  2903. int nI1 = max(min(nDispRow1,1000), 0);
  2904. int nI2 = max(min(nDispRow2,1000), nCountVDR-1);
  2905. for (int i = nI1; i <= nI2; i++)
  2906. {
  2907. m_arValidDispRows.SetAtGrow(i, 0);
  2908. }
  2909. //===========================================================================
  2910. ClearOnScreenSchCache(nRowFrom);
  2911. }
  2912. void CXTPSyntaxEditCtrl::ClearOnScreenSchCache(int nRowFrom)
  2913. {
  2914. //-* clear ALL ------------------------------------------------------
  2915. if (nRowFrom < 0)
  2916. {
  2917. m_arOnScreenSchCache.RemoveAll();
  2918. return ;
  2919. }
  2920. //===========================================================================
  2921. int nCountSSC = (int)m_arOnScreenSchCache.GetSize();
  2922. for (int i = nCountSSC-1; i >= 0; i--)
  2923. {
  2924. CScreenSearchBlock& rSchBlk = m_arOnScreenSchCache[i];
  2925. BOOL bRemove = FALSE;
  2926. if (nRowFrom >= rSchBlk.nRowStart && nRowFrom <= rSchBlk.nRowEnd)
  2927. {
  2928. rSchBlk.nRowEnd = nRowFrom-1;
  2929. bRemove = !(rSchBlk.nRowStart <= rSchBlk.nRowEnd);
  2930. }
  2931. else if (nRowFrom < rSchBlk.nRowStart)
  2932. {
  2933. bRemove = TRUE;
  2934. }
  2935. //-------------------------------------
  2936. if (bRemove)
  2937. {
  2938. m_arOnScreenSchCache.RemoveAt(i);
  2939. }
  2940. else
  2941. {
  2942. ASSERT(m_arOnScreenSchCache[i].nRowStart == rSchBlk.nRowStart);
  2943. ASSERT(m_arOnScreenSchCache[i].nRowEnd == rSchBlk.nRowEnd);
  2944. }
  2945. }
  2946. }
  2947. void CXTPSyntaxEditCtrl::SetRowValid(int nDispRow)
  2948. {
  2949. ASSERT(nDispRow < 1000);
  2950. m_arValidDispRows.SetAtGrow(nDispRow, 1);
  2951. }
  2952. BOOL CXTPSyntaxEditCtrl::IsRowValid(int nDispRow)
  2953. {
  2954. int nCount = (int)m_arValidDispRows.GetSize();
  2955. if (nDispRow > 0 && nDispRow < nCount)
  2956. {
  2957. int nValid = m_arValidDispRows[nDispRow];
  2958. return (nValid != 0);
  2959. }
  2960. return FALSE;
  2961. }
  2962. const CStringList& CXTPSyntaxEditCtrl::GetUndoTextList()
  2963. {
  2964. return m_pBuffer->GetUndoRedoManager()->GetUndoTextList();
  2965. }
  2966. const CStringList& CXTPSyntaxEditCtrl::GetRedoTextList()
  2967. {
  2968. return m_pBuffer->GetUndoRedoManager()->GetRedoTextList();
  2969. }
  2970. void CXTPSyntaxEditCtrl::OnLexCfgWasChanged(XTP_NOTIFY_CODE code, WPARAM wParam, LPARAM lParam)
  2971. {
  2972. UNREFERENCED_PARAMETER(wParam); UNREFERENCED_PARAMETER(lParam);
  2973. if (code == xtpEditAllConfigWasChanged ||
  2974. code == xtpEditClassSchWasChanged ||
  2975. code == xtpEditThemeWasChanged)
  2976. {
  2977. m_arOnScreenSchCache.RemoveAll();
  2978. Invalidate(FALSE);
  2979. }
  2980. else
  2981. {
  2982. ASSERT(FALSE);
  2983. }
  2984. }
  2985. BOOL CXTPSyntaxEditCtrl::GetRowNodes(int nRow, DWORD& dwType)
  2986. {
  2987. dwType = XTP_EDIT_ROWNODE_NOTHING;
  2988. // retrieve row nodes
  2989. CXTPSyntaxEditRowsBlockArray arCoBlocks;
  2990. GetCollapsableBlocksInfo(nRow, arCoBlocks);
  2991. int nCount = (int)arCoBlocks.GetSize();
  2992. for (int i = 0; i < nCount; i++)
  2993. {
  2994. XTP_EDIT_ROWSBLOCK coBlk = arCoBlocks[i];
  2995. if (coBlk.lcStart.nLine == nRow)
  2996. {
  2997. if (HasRowMark(coBlk.lcStart.nLine, xtpEditLMT_Collapsed))
  2998. {
  2999. dwType |= XTP_EDIT_ROWNODE_COLLAPSED;
  3000. }
  3001. else
  3002. {
  3003. dwType |= XTP_EDIT_ROWNODE_EXPANDED;
  3004. }
  3005. }
  3006. if (coBlk.lcEnd.nLine == nRow)
  3007. {
  3008. dwType |= XTP_EDIT_ROWNODE_ENDMARK;
  3009. }
  3010. if (coBlk.lcStart.nLine < nRow)
  3011. {
  3012. dwType |= XTP_EDIT_ROWNODE_NODEUP;
  3013. }
  3014. if (coBlk.lcEnd.nLine > nRow && (dwType & XTP_EDIT_ROWNODE_COLLAPSED))
  3015. {
  3016. dwType |= XTP_EDIT_ROWNODE_NODEDOWN;
  3017. }
  3018. }
  3019. return (dwType != XTP_EDIT_ROWNODE_NOTHING);
  3020. }
  3021. BOOL CXTPSyntaxEditCtrl::NotifyMarginLBtnClick(int nRow, int nDispRow)
  3022. {
  3023. // Notify the parent class that position has been changed
  3024. XTP_EDIT_NMHDR_MARGINCLICKED somc;
  3025. // NMHDR codes
  3026. somc.nmhdr.code = XTP_EDIT_NM_MARGINCLICKED;
  3027. somc.nmhdr.hwndFrom = m_hWnd;
  3028. somc.nmhdr.idFrom = GetDlgCtrlID();
  3029. // Row col specific codes
  3030. somc.nRow = nRow;
  3031. somc.nDispRow = nDispRow;
  3032. // Notify the parent window
  3033. if (::IsWindow(m_pParentWnd->GetSafeHwnd()))
  3034. {
  3035. return (BOOL)m_pParentWnd->SendMessage(
  3036. WM_NOTIFY, (WPARAM)somc.nmhdr.idFrom, (LPARAM)&somc);
  3037. }
  3038. return FALSE;
  3039. }
  3040. UINT CXTPSyntaxEditCtrl::CalcAveDataSize(int nRowStart, int nRowEnd)
  3041. {
  3042. UINT uSize = m_nAverageLineLen * abs(nRowStart - nRowEnd);
  3043. uSize = (uSize / 1024 + 1) * 1024;
  3044. return uSize;
  3045. }
  3046. BOOL CXTPSyntaxEditCtrl::GetWideCaret() const
  3047. {
  3048. return m_bWideCaret && m_pBuffer && m_pBuffer->GetOverwriteFlag();
  3049. }
  3050. BOOL CXTPSyntaxEditCtrl::SetWideCaret(BOOL bWideCaret, BOOL bUpdateReg/*=FALSE*/)
  3051. {
  3052. if (m_bWideCaret != bWideCaret)
  3053. {
  3054. if (!SetValueBool(XTP_EDIT_REG_WIDECARET, bWideCaret, m_bWideCaret, bUpdateReg))
  3055. return FALSE;
  3056. SetCurCaretPos(GetCurrentDocumentRow(), m_nDispCol, FALSE, FALSE);
  3057. }
  3058. return TRUE;
  3059. }
  3060. void CXTPSyntaxEditCtrl::SetPaintManager(CXTPSyntaxEditPaintManager* pPaintManager)
  3061. {
  3062. ASSERT(pPaintManager);
  3063. if (pPaintManager)
  3064. {
  3065. CMDTARGET_RELEASE(m_pPaintManeger);
  3066. m_pPaintManeger = pPaintManager;
  3067. }
  3068. }
  3069. //===========================================================================
  3070. // CXTPSyntaxEditCtrl::CTextSearchCache
  3071. //===========================================================================
  3072. CXTPSyntaxEditCtrl::CTextSearchCache::CTextSearchCache()
  3073. {
  3074. nForTopRow = 0;
  3075. }
  3076. void CXTPSyntaxEditCtrl::CTextSearchCache::Update(int nCurrTopRow)
  3077. {
  3078. if (nCurrTopRow != nForTopRow)
  3079. {
  3080. ptrTBStart = NULL;
  3081. nForTopRow = nCurrTopRow;
  3082. }
  3083. }
  3084. //===========================================================================
  3085. // CXTPSyntaxEditCtrl::CScreenSearchBlock
  3086. //===========================================================================
  3087. CXTPSyntaxEditCtrl::CScreenSearchBlock::CScreenSearchBlock()
  3088. {
  3089. nRowStart = nRowEnd = 0;
  3090. dwLastAccessTime = 0;
  3091. }
  3092. CXTPSyntaxEditCtrl::CScreenSearchBlock::CScreenSearchBlock(const CScreenSearchBlock& rSrc)
  3093. {
  3094. nRowStart = rSrc.nRowStart;
  3095. nRowEnd = rSrc.nRowEnd;
  3096. ptrTBFirst = rSrc.ptrTBFirst;
  3097. dwLastAccessTime = rSrc.dwLastAccessTime;
  3098. }
  3099. CXTPSyntaxEditCtrl::CScreenSearchBlock::~CScreenSearchBlock()
  3100. {
  3101. }
  3102. //===========================================================================
  3103. // CXTPSyntaxEditCtrl::CScreenSearchCache
  3104. //===========================================================================
  3105. CXTPSyntaxEditCtrl::CScreenSearchCache::CScreenSearchCache()
  3106. {
  3107. m_dwLastRemoveOldTime = 0;
  3108. }
  3109. CXTPSyntaxEditCtrl::CScreenSearchCache::~CScreenSearchCache()
  3110. {
  3111. RemoveAll();
  3112. }
  3113. void CXTPSyntaxEditCtrl::CScreenSearchCache::RemoveAll()
  3114. {
  3115. for (int i = 0; i < (int)GetSize(); i++)
  3116. {
  3117. CScreenSearchBlock& rSchBlk = ElementAt(i);
  3118. CXTPSyntaxEditLexTextSchema::Close(rSchBlk.ptrTBFirst);
  3119. }
  3120. Base::RemoveAll();
  3121. }
  3122. void CXTPSyntaxEditCtrl::CScreenSearchCache::RemoveAt(int nIndex)
  3123. {
  3124. CScreenSearchBlock& rSchBlk = ElementAt(nIndex);
  3125. CXTPSyntaxEditLexTextSchema::Close(rSchBlk.ptrTBFirst);
  3126. Base::RemoveAt(nIndex);
  3127. }
  3128. void CXTPSyntaxEditCtrl::CScreenSearchCache::RemoveOld(int nTimeOut_sec)
  3129. {
  3130. int nTimeOut_ms = nTimeOut_sec * 1000;
  3131. DWORD dwTime = ::GetTickCount();
  3132. int cnCheckTimeOut_ms = min(nTimeOut_sec, 10*1000);
  3133. if (nTimeOut_sec == 0 || nTimeOut_sec == -1 ||
  3134. labs(dwTime - m_dwLastRemoveOldTime) < cnCheckTimeOut_ms)
  3135. {
  3136. return;
  3137. }
  3138. //================================================================
  3139. m_dwLastRemoveOldTime = dwTime;
  3140. for (int i = (int)GetSize() - 1; i >= 0; i--)
  3141. {
  3142. CScreenSearchBlock& rSchBlk = ElementAt(i);
  3143. if (labs(dwTime - rSchBlk.dwLastAccessTime) >= nTimeOut_ms)
  3144. {
  3145. CXTPSyntaxEditLexTextSchema::Close(rSchBlk.ptrTBFirst);
  3146. Base::RemoveAt(i);
  3147. }
  3148. }
  3149. }
  3150. //===========================================================================
  3151. // CXTPSyntaxEditCtrl::CAverageVal
  3152. //===========================================================================
  3153. CXTPSyntaxEditCtrl::CAverageVal::CAverageVal(int nDataSize/* = 100*/)
  3154. {
  3155. m_nDataSize = max(1, nDataSize);
  3156. m_nNextIndex = 0;
  3157. }
  3158. void CXTPSyntaxEditCtrl::CAverageVal::AddValue(UINT uVal)
  3159. {
  3160. m_arData.SetAtGrow(m_nNextIndex, uVal);
  3161. m_nNextIndex = (m_nNextIndex+1) % m_nDataSize;
  3162. }
  3163. UINT CXTPSyntaxEditCtrl::CAverageVal::GetAverageValue(UINT uDefaultIfNoData/* = 0*/)
  3164. {
  3165. int nCount = (int)m_arData.GetSize();
  3166. if (nCount <= 0)
  3167. {
  3168. return uDefaultIfNoData;
  3169. }
  3170. UINT uSumm = 0;
  3171. for (int i = 0; i < nCount; i++) {
  3172. uSumm += m_arData[i];
  3173. }
  3174. UINT uAveVal = uSumm / nCount;
  3175. return uAveVal;
  3176. }
  3177. int CXTPSyntaxEditCtrl::GetRowCount() const
  3178. {
  3179. return m_pBuffer ? m_pBuffer->GetRowCount() : 0;
  3180. }
  3181. CXTPSyntaxEditConfigurationManager* CXTPSyntaxEditCtrl::GetLexConfigurationManager() const
  3182. {
  3183. return m_pBuffer? m_pBuffer->GetLexConfigurationManager(): NULL;
  3184. }