vmdeditor.cpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872
  1. #include "vmdeditor.h"
  2. #include <QtWidgets>
  3. #include <QMenu>
  4. #include <QDebug>
  5. #include "vdocument.h"
  6. #include "utils/veditutils.h"
  7. #include "vedittab.h"
  8. #include "hgmarkdownhighlighter.h"
  9. #include "vcodeblockhighlighthelper.h"
  10. #include "vmdeditoperations.h"
  11. #include "vtableofcontent.h"
  12. #include "utils/veditutils.h"
  13. #include "dialog/vselectdialog.h"
  14. #include "dialog/vconfirmdeletiondialog.h"
  15. #include "vtextblockdata.h"
  16. #include "vorphanfile.h"
  17. #include "vnotefile.h"
  18. #include "vpreviewmanager.h"
  19. extern VConfigManager *g_config;
  20. VMdEditor::VMdEditor(VFile *p_file,
  21. VDocument *p_doc,
  22. MarkdownConverterType p_type,
  23. QWidget *p_parent)
  24. : VTextEdit(p_parent),
  25. VEditor(p_file, this),
  26. m_mdHighlighter(NULL),
  27. m_freshEdit(true)
  28. {
  29. Q_ASSERT(p_file->getDocType() == DocType::Markdown);
  30. VEditor::init();
  31. // Hook functions from VEditor.
  32. connect(this, &VTextEdit::cursorPositionChanged,
  33. this, [this]() {
  34. highlightOnCursorPositionChanged();
  35. });
  36. connect(this, &VTextEdit::selectionChanged,
  37. this, [this]() {
  38. highlightSelectedWord();
  39. });
  40. // End.
  41. m_mdHighlighter = new HGMarkdownHighlighter(g_config->getMdHighlightingStyles(),
  42. g_config->getCodeBlockStyles(),
  43. g_config->getMarkdownHighlightInterval(),
  44. document());
  45. connect(m_mdHighlighter, &HGMarkdownHighlighter::headersUpdated,
  46. this, &VMdEditor::updateHeaders);
  47. // After highlight, the cursor may trun into non-visible. We should make it visible
  48. // in this case.
  49. connect(m_mdHighlighter, &HGMarkdownHighlighter::highlightCompleted,
  50. this, [this]() {
  51. makeBlockVisible(textCursor().block());
  52. });
  53. m_cbHighlighter = new VCodeBlockHighlightHelper(m_mdHighlighter,
  54. p_doc,
  55. p_type);
  56. m_previewMgr = new VPreviewManager(this);
  57. connect(m_mdHighlighter, &HGMarkdownHighlighter::imageLinksUpdated,
  58. m_previewMgr, &VPreviewManager::imageLinksUpdated);
  59. connect(m_previewMgr, &VPreviewManager::requestUpdateImageLinks,
  60. m_mdHighlighter, &HGMarkdownHighlighter::updateHighlight);
  61. m_editOps = new VMdEditOperations(this, m_file);
  62. connect(m_editOps, &VEditOperations::statusMessage,
  63. m_object, &VEditorObject::statusMessage);
  64. connect(m_editOps, &VEditOperations::vimStatusUpdated,
  65. m_object, &VEditorObject::vimStatusUpdated);
  66. connect(this, &VTextEdit::cursorPositionChanged,
  67. this, &VMdEditor::updateCurrentHeader);
  68. updateFontAndPalette();
  69. updateConfig();
  70. }
  71. void VMdEditor::updateFontAndPalette()
  72. {
  73. setFont(g_config->getMdEditFont());
  74. setPalette(g_config->getMdEditPalette());
  75. }
  76. void VMdEditor::beginEdit()
  77. {
  78. updateFontAndPalette();
  79. updateConfig();
  80. initInitImages();
  81. setModified(false);
  82. setReadOnlyAndHighlightCurrentLine(false);
  83. emit statusChanged();
  84. updateHeaders(m_mdHighlighter->getHeaderRegions());
  85. if (m_freshEdit) {
  86. m_freshEdit = false;
  87. emit m_object->ready();
  88. }
  89. }
  90. void VMdEditor::endEdit()
  91. {
  92. setReadOnlyAndHighlightCurrentLine(true);
  93. clearUnusedImages();
  94. }
  95. void VMdEditor::saveFile()
  96. {
  97. Q_ASSERT(m_file->isModifiable());
  98. if (!document()->isModified()) {
  99. return;
  100. }
  101. m_file->setContent(toPlainText());
  102. setModified(false);
  103. }
  104. void VMdEditor::reloadFile()
  105. {
  106. const QString &content = m_file->getContent();
  107. setPlainText(content);
  108. setModified(false);
  109. }
  110. bool VMdEditor::scrollToBlock(int p_blockNumber)
  111. {
  112. QTextBlock block = document()->findBlockByNumber(p_blockNumber);
  113. if (block.isValid()) {
  114. VEditUtils::scrollBlockInPage(this, block.blockNumber(), 0);
  115. moveCursor(QTextCursor::EndOfBlock);
  116. return true;
  117. }
  118. return false;
  119. }
  120. // Get the visual offset of a block.
  121. #define GETVISUALOFFSETY (contentOffsetY() + (int)rect.y())
  122. void VMdEditor::makeBlockVisible(const QTextBlock &p_block)
  123. {
  124. if (!p_block.isValid() || !p_block.isVisible()) {
  125. return;
  126. }
  127. QScrollBar *vbar = verticalScrollBar();
  128. if (!vbar || !vbar->isVisible()) {
  129. // No vertical scrollbar. No need to scroll.
  130. return;
  131. }
  132. int height = rect().height();
  133. QScrollBar *hbar = horizontalScrollBar();
  134. if (hbar && hbar->isVisible()) {
  135. height -= hbar->height();
  136. }
  137. bool moved = false;
  138. QAbstractTextDocumentLayout *layout = document()->documentLayout();
  139. QRectF rect = layout->blockBoundingRect(p_block);
  140. int y = GETVISUALOFFSETY;
  141. int rectHeight = (int)rect.height();
  142. // Handle the case rectHeight >= height.
  143. if (rectHeight >= height) {
  144. if (y < 0) {
  145. // Need to scroll up.
  146. while (y + rectHeight < height && vbar->value() > vbar->minimum()) {
  147. moved = true;
  148. vbar->setValue(vbar->value() - vbar->singleStep());
  149. rect = layout->blockBoundingRect(p_block);
  150. rectHeight = (int)rect.height();
  151. y = GETVISUALOFFSETY;
  152. }
  153. } else if (y > 0) {
  154. // Need to scroll down.
  155. while (y > 0 && vbar->value() < vbar->maximum()) {
  156. moved = true;
  157. vbar->setValue(vbar->value() + vbar->singleStep());
  158. rect = layout->blockBoundingRect(p_block);
  159. rectHeight = (int)rect.height();
  160. y = GETVISUALOFFSETY;
  161. }
  162. if (y < 0) {
  163. // One step back.
  164. moved = true;
  165. vbar->setValue(vbar->value() - vbar->singleStep());
  166. }
  167. }
  168. if (moved) {
  169. qDebug() << "scroll to make huge block visible";
  170. }
  171. return;
  172. }
  173. while (y < 0 && vbar->value() > vbar->minimum()) {
  174. qDebug() << y << vbar->value() << vbar->minimum() << rectHeight;
  175. moved = true;
  176. vbar->setValue(vbar->value() - vbar->singleStep());
  177. rect = layout->blockBoundingRect(p_block);
  178. rectHeight = (int)rect.height();
  179. y = GETVISUALOFFSETY;
  180. }
  181. if (moved) {
  182. qDebug() << "scroll page down to make block visible";
  183. return;
  184. }
  185. while (y + rectHeight > height && vbar->value() < vbar->maximum()) {
  186. moved = true;
  187. vbar->setValue(vbar->value() + vbar->singleStep());
  188. rect = layout->blockBoundingRect(p_block);
  189. rectHeight = (int)rect.height();
  190. y = GETVISUALOFFSETY;
  191. }
  192. if (moved) {
  193. qDebug() << "scroll page up to make block visible";
  194. }
  195. }
  196. void VMdEditor::contextMenuEvent(QContextMenuEvent *p_event)
  197. {
  198. QMenu *menu = createStandardContextMenu();
  199. menu->setToolTipsVisible(true);
  200. const QList<QAction *> actions = menu->actions();
  201. if (!textCursor().hasSelection()) {
  202. VEditTab *editTab = dynamic_cast<VEditTab *>(parent());
  203. Q_ASSERT(editTab);
  204. if (editTab->isEditMode()) {
  205. QAction *saveExitAct = new QAction(QIcon(":/resources/icons/save_exit.svg"),
  206. tr("&Save Changes And Read"),
  207. menu);
  208. saveExitAct->setToolTip(tr("Save changes and exit edit mode"));
  209. connect(saveExitAct, &QAction::triggered,
  210. this, [this]() {
  211. emit m_object->saveAndRead();
  212. });
  213. QAction *discardExitAct = new QAction(QIcon(":/resources/icons/discard_exit.svg"),
  214. tr("&Discard Changes And Read"),
  215. menu);
  216. discardExitAct->setToolTip(tr("Discard changes and exit edit mode"));
  217. connect(discardExitAct, &QAction::triggered,
  218. this, [this]() {
  219. emit m_object->discardAndRead();
  220. });
  221. menu->insertAction(actions.isEmpty() ? NULL : actions[0], discardExitAct);
  222. menu->insertAction(discardExitAct, saveExitAct);
  223. if (!actions.isEmpty()) {
  224. menu->insertSeparator(actions[0]);
  225. }
  226. }
  227. }
  228. menu->exec(p_event->globalPos());
  229. delete menu;
  230. }
  231. void VMdEditor::mousePressEvent(QMouseEvent *p_event)
  232. {
  233. if (handleMousePressEvent(p_event)) {
  234. return;
  235. }
  236. VTextEdit::mousePressEvent(p_event);
  237. emit m_object->selectionChangedByMouse(textCursor().hasSelection());
  238. }
  239. void VMdEditor::mouseReleaseEvent(QMouseEvent *p_event)
  240. {
  241. if (handleMouseReleaseEvent(p_event)) {
  242. return;
  243. }
  244. VTextEdit::mousePressEvent(p_event);
  245. }
  246. void VMdEditor::mouseMoveEvent(QMouseEvent *p_event)
  247. {
  248. if (handleMouseMoveEvent(p_event)) {
  249. return;
  250. }
  251. VTextEdit::mouseMoveEvent(p_event);
  252. emit m_object->selectionChangedByMouse(textCursor().hasSelection());
  253. }
  254. QVariant VMdEditor::inputMethodQuery(Qt::InputMethodQuery p_query) const
  255. {
  256. QVariant ret;
  257. if (handleInputMethodQuery(p_query, ret)) {
  258. return ret;
  259. }
  260. return VTextEdit::inputMethodQuery(p_query);
  261. }
  262. bool VMdEditor::isBlockVisible(const QTextBlock &p_block)
  263. {
  264. if (!p_block.isValid() || !p_block.isVisible()) {
  265. return false;
  266. }
  267. QScrollBar *vbar = verticalScrollBar();
  268. if (!vbar || !vbar->isVisible()) {
  269. // No vertical scrollbar.
  270. return true;
  271. }
  272. int height = rect().height();
  273. QScrollBar *hbar = horizontalScrollBar();
  274. if (hbar && hbar->isVisible()) {
  275. height -= hbar->height();
  276. }
  277. QAbstractTextDocumentLayout *layout = document()->documentLayout();
  278. QRectF rect = layout->blockBoundingRect(p_block);
  279. int y = GETVISUALOFFSETY;
  280. int rectHeight = (int)rect.height();
  281. return (y >= 0 && y < height) || (y < 0 && y + rectHeight > 0);
  282. }
  283. static void addHeaderSequence(QVector<int> &p_sequence, int p_level, int p_baseLevel)
  284. {
  285. Q_ASSERT(p_level >= 1 && p_level < p_sequence.size());
  286. if (p_level < p_baseLevel) {
  287. p_sequence.fill(0);
  288. return;
  289. }
  290. ++p_sequence[p_level];
  291. for (int i = p_level + 1; i < p_sequence.size(); ++i) {
  292. p_sequence[i] = 0;
  293. }
  294. }
  295. static QString headerSequenceStr(const QVector<int> &p_sequence)
  296. {
  297. QString res;
  298. for (int i = 1; i < p_sequence.size(); ++i) {
  299. if (p_sequence[i] != 0) {
  300. res = res + QString::number(p_sequence[i]) + '.';
  301. } else if (res.isEmpty()) {
  302. continue;
  303. } else {
  304. break;
  305. }
  306. }
  307. return res;
  308. }
  309. static void insertSequenceToHeader(QTextBlock p_block,
  310. QRegExp &p_reg,
  311. QRegExp &p_preReg,
  312. const QString &p_seq)
  313. {
  314. if (!p_block.isValid()) {
  315. return;
  316. }
  317. QString text = p_block.text();
  318. bool matched = p_reg.exactMatch(text);
  319. Q_ASSERT(matched);
  320. matched = p_preReg.exactMatch(text);
  321. Q_ASSERT(matched);
  322. int start = p_reg.cap(1).length() + 1;
  323. int end = p_preReg.cap(1).length();
  324. Q_ASSERT(start <= end);
  325. QTextCursor cursor(p_block);
  326. cursor.setPosition(p_block.position() + start);
  327. if (start != end) {
  328. cursor.setPosition(p_block.position() + end, QTextCursor::KeepAnchor);
  329. }
  330. if (p_seq.isEmpty()) {
  331. cursor.removeSelectedText();
  332. } else {
  333. cursor.insertText(p_seq + ' ');
  334. }
  335. }
  336. void VMdEditor::updateHeaders(const QVector<VElementRegion> &p_headerRegions)
  337. {
  338. QTextDocument *doc = document();
  339. QVector<VTableOfContentItem> headers;
  340. QVector<int> headerBlockNumbers;
  341. QVector<QString> headerSequences;
  342. if (!p_headerRegions.isEmpty()) {
  343. headers.reserve(p_headerRegions.size());
  344. headerBlockNumbers.reserve(p_headerRegions.size());
  345. headerSequences.reserve(p_headerRegions.size());
  346. }
  347. // Assume that each block contains only one line
  348. // Only support # syntax for now
  349. QRegExp headerReg(VUtils::c_headerRegExp);
  350. int baseLevel = -1;
  351. for (auto const & reg : p_headerRegions) {
  352. QTextBlock block = doc->findBlock(reg.m_startPos);
  353. if (!block.isValid()) {
  354. continue;
  355. }
  356. if (!block.contains(reg.m_endPos - 1)) {
  357. continue;
  358. }
  359. if ((block.userState() == HighlightBlockState::Normal)
  360. && headerReg.exactMatch(block.text())) {
  361. int level = headerReg.cap(1).length();
  362. VTableOfContentItem header(headerReg.cap(2).trimmed(),
  363. level,
  364. block.blockNumber(),
  365. headers.size());
  366. headers.append(header);
  367. headerBlockNumbers.append(block.blockNumber());
  368. headerSequences.append(headerReg.cap(3));
  369. if (baseLevel == -1) {
  370. baseLevel = level;
  371. } else if (baseLevel > level) {
  372. baseLevel = level;
  373. }
  374. }
  375. }
  376. m_headers.clear();
  377. bool autoSequence = m_config.m_enableHeadingSequence
  378. && !isReadOnly()
  379. && m_file->isModifiable();
  380. int headingSequenceBaseLevel = g_config->getHeadingSequenceBaseLevel();
  381. if (headingSequenceBaseLevel < 1 || headingSequenceBaseLevel > 6) {
  382. headingSequenceBaseLevel = 1;
  383. }
  384. QVector<int> seqs(7, 0);
  385. QRegExp preReg(VUtils::c_headerPrefixRegExp);
  386. int curLevel = baseLevel - 1;
  387. for (int i = 0; i < headers.size(); ++i) {
  388. VTableOfContentItem &item = headers[i];
  389. while (item.m_level > curLevel + 1) {
  390. curLevel += 1;
  391. // Insert empty level which is an invalid header.
  392. m_headers.append(VTableOfContentItem(c_emptyHeaderName,
  393. curLevel,
  394. -1,
  395. m_headers.size()));
  396. if (autoSequence) {
  397. addHeaderSequence(seqs, curLevel, headingSequenceBaseLevel);
  398. }
  399. }
  400. item.m_index = m_headers.size();
  401. m_headers.append(item);
  402. curLevel = item.m_level;
  403. if (autoSequence) {
  404. addHeaderSequence(seqs, item.m_level, headingSequenceBaseLevel);
  405. QString seqStr = headerSequenceStr(seqs);
  406. if (headerSequences[i] != seqStr) {
  407. // Insert correct sequence.
  408. insertSequenceToHeader(doc->findBlockByNumber(headerBlockNumbers[i]),
  409. headerReg,
  410. preReg,
  411. seqStr);
  412. }
  413. }
  414. }
  415. emit headersChanged(m_headers);
  416. updateCurrentHeader();
  417. }
  418. void VMdEditor::updateCurrentHeader()
  419. {
  420. emit currentHeaderChanged(textCursor().block().blockNumber());
  421. }
  422. void VMdEditor::initInitImages()
  423. {
  424. m_initImages = VUtils::fetchImagesFromMarkdownFile(m_file,
  425. ImageLink::LocalRelativeInternal);
  426. }
  427. void VMdEditor::clearUnusedImages()
  428. {
  429. QVector<ImageLink> images = VUtils::fetchImagesFromMarkdownFile(m_file,
  430. ImageLink::LocalRelativeInternal);
  431. QVector<QString> unusedImages;
  432. if (!m_insertedImages.isEmpty()) {
  433. for (int i = 0; i < m_insertedImages.size(); ++i) {
  434. const ImageLink &link = m_insertedImages[i];
  435. if (link.m_type != ImageLink::LocalRelativeInternal) {
  436. continue;
  437. }
  438. int j;
  439. for (j = 0; j < images.size(); ++j) {
  440. if (VUtils::equalPath(link.m_path, images[j].m_path)) {
  441. break;
  442. }
  443. }
  444. // This inserted image is no longer in the file.
  445. if (j == images.size()) {
  446. unusedImages.push_back(link.m_path);
  447. }
  448. }
  449. m_insertedImages.clear();
  450. }
  451. for (int i = 0; i < m_initImages.size(); ++i) {
  452. const ImageLink &link = m_initImages[i];
  453. V_ASSERT(link.m_type == ImageLink::LocalRelativeInternal);
  454. int j;
  455. for (j = 0; j < images.size(); ++j) {
  456. if (VUtils::equalPath(link.m_path, images[j].m_path)) {
  457. break;
  458. }
  459. }
  460. // Original local relative image is no longer in the file.
  461. if (j == images.size()) {
  462. unusedImages.push_back(link.m_path);
  463. }
  464. }
  465. if (!unusedImages.isEmpty()) {
  466. if (g_config->getConfirmImagesCleanUp()) {
  467. QVector<ConfirmItemInfo> items;
  468. for (auto const & img : unusedImages) {
  469. items.push_back(ConfirmItemInfo(img,
  470. img,
  471. img,
  472. NULL));
  473. }
  474. QString text = tr("Following images seems not to be used in this note anymore. "
  475. "Please confirm the deletion of these images.");
  476. QString info = tr("Deleted files could be found in the recycle "
  477. "bin of this note.<br>"
  478. "Click \"Cancel\" to leave them untouched.");
  479. VConfirmDeletionDialog dialog(tr("Confirm Cleaning Up Unused Images"),
  480. text,
  481. info,
  482. items,
  483. true,
  484. true,
  485. true,
  486. this);
  487. unusedImages.clear();
  488. if (dialog.exec()) {
  489. items = dialog.getConfirmedItems();
  490. g_config->setConfirmImagesCleanUp(dialog.getAskAgainEnabled());
  491. for (auto const & item : items) {
  492. unusedImages.push_back(item.m_name);
  493. }
  494. }
  495. }
  496. for (int i = 0; i < unusedImages.size(); ++i) {
  497. bool ret = false;
  498. if (m_file->getType() == FileType::Note) {
  499. const VNoteFile *tmpFile = dynamic_cast<const VNoteFile *>((VFile *)m_file);
  500. ret = VUtils::deleteFile(tmpFile->getNotebook(), unusedImages[i], false);
  501. } else if (m_file->getType() == FileType::Orphan) {
  502. const VOrphanFile *tmpFile = dynamic_cast<const VOrphanFile *>((VFile *)m_file);
  503. ret = VUtils::deleteFile(tmpFile, unusedImages[i], false);
  504. } else {
  505. Q_ASSERT(false);
  506. }
  507. if (!ret) {
  508. qWarning() << "fail to delete unused original image" << unusedImages[i];
  509. } else {
  510. qDebug() << "delete unused image" << unusedImages[i];
  511. }
  512. }
  513. }
  514. m_initImages.clear();
  515. }
  516. void VMdEditor::keyPressEvent(QKeyEvent *p_event)
  517. {
  518. if (m_editOps && m_editOps->handleKeyPressEvent(p_event)) {
  519. return;
  520. }
  521. VTextEdit::keyPressEvent(p_event);
  522. }
  523. bool VMdEditor::canInsertFromMimeData(const QMimeData *p_source) const
  524. {
  525. return p_source->hasImage()
  526. || p_source->hasUrls()
  527. || VTextEdit::canInsertFromMimeData(p_source);
  528. }
  529. void VMdEditor::insertFromMimeData(const QMimeData *p_source)
  530. {
  531. VSelectDialog dialog(tr("Insert From Clipboard"), this);
  532. dialog.addSelection(tr("Insert As Image"), 0);
  533. dialog.addSelection(tr("Insert As Text"), 1);
  534. if (p_source->hasImage()) {
  535. // Image data in the clipboard
  536. if (p_source->hasText()) {
  537. if (dialog.exec() == QDialog::Accepted) {
  538. if (dialog.getSelection() == 1) {
  539. // Insert as text.
  540. Q_ASSERT(p_source->hasText() && p_source->hasImage());
  541. VTextEdit::insertFromMimeData(p_source);
  542. return;
  543. }
  544. } else {
  545. return;
  546. }
  547. }
  548. m_editOps->insertImageFromMimeData(p_source);
  549. return;
  550. } else if (p_source->hasUrls()) {
  551. QList<QUrl> urls = p_source->urls();
  552. if (urls.size() == 1 && VUtils::isImageURL(urls[0])) {
  553. if (dialog.exec() == QDialog::Accepted) {
  554. // FIXME: After calling dialog.exec(), p_source->hasUrl() returns false.
  555. if (dialog.getSelection() == 0) {
  556. // Insert as image.
  557. m_editOps->insertImageFromURL(urls[0]);
  558. return;
  559. }
  560. QMimeData newSource;
  561. newSource.setUrls(urls);
  562. VTextEdit::insertFromMimeData(&newSource);
  563. return;
  564. } else {
  565. return;
  566. }
  567. }
  568. } else if (p_source->hasText()) {
  569. QString text = p_source->text();
  570. if (VUtils::isImageURLText(text)) {
  571. // The text is a URL to an image.
  572. if (dialog.exec() == QDialog::Accepted) {
  573. if (dialog.getSelection() == 0) {
  574. // Insert as image.
  575. QUrl url(text);
  576. if (url.isValid()) {
  577. m_editOps->insertImageFromURL(QUrl(text));
  578. }
  579. return;
  580. }
  581. } else {
  582. return;
  583. }
  584. }
  585. Q_ASSERT(p_source->hasText());
  586. }
  587. VTextEdit::insertFromMimeData(p_source);
  588. }
  589. void VMdEditor::imageInserted(const QString &p_path)
  590. {
  591. ImageLink link;
  592. link.m_path = p_path;
  593. if (m_file->useRelativeImageFolder()) {
  594. link.m_type = ImageLink::LocalRelativeInternal;
  595. } else {
  596. link.m_type = ImageLink::LocalAbsolute;
  597. }
  598. m_insertedImages.append(link);
  599. }
  600. bool VMdEditor::scrollToHeader(int p_blockNumber)
  601. {
  602. if (p_blockNumber < 0) {
  603. return false;
  604. }
  605. return scrollToBlock(p_blockNumber);
  606. }
  607. int VMdEditor::indexOfCurrentHeader() const
  608. {
  609. if (m_headers.isEmpty()) {
  610. return -1;
  611. }
  612. int blockNumber = textCursor().block().blockNumber();
  613. for (int i = m_headers.size() - 1; i >= 0; --i) {
  614. if (!m_headers[i].isEmpty()
  615. && m_headers[i].m_blockNumber <= blockNumber) {
  616. return i;
  617. }
  618. }
  619. return -1;
  620. }
  621. bool VMdEditor::jumpTitle(bool p_forward, int p_relativeLevel, int p_repeat)
  622. {
  623. if (m_headers.isEmpty()) {
  624. return false;
  625. }
  626. QTextCursor cursor = textCursor();
  627. int cursorLine = cursor.block().blockNumber();
  628. int targetIdx = -1;
  629. // -1: skip level check.
  630. int targetLevel = 0;
  631. int idx = indexOfCurrentHeader();
  632. if (idx == -1) {
  633. // Cursor locates at the beginning, before any headers.
  634. if (p_relativeLevel < 0 || !p_forward) {
  635. return false;
  636. }
  637. }
  638. int delta = 1;
  639. if (!p_forward) {
  640. delta = -1;
  641. }
  642. bool firstHeader = true;
  643. for (targetIdx = idx == -1 ? 0 : idx;
  644. targetIdx >= 0 && targetIdx < m_headers.size();
  645. targetIdx += delta) {
  646. const VTableOfContentItem &header = m_headers[targetIdx];
  647. if (header.isEmpty()) {
  648. continue;
  649. }
  650. if (targetLevel == 0) {
  651. // The target level has not been init yet.
  652. Q_ASSERT(firstHeader);
  653. targetLevel = header.m_level;
  654. if (p_relativeLevel < 0) {
  655. targetLevel += p_relativeLevel;
  656. if (targetLevel < 1) {
  657. // Invalid level.
  658. return false;
  659. }
  660. } else if (p_relativeLevel > 0) {
  661. targetLevel = -1;
  662. }
  663. }
  664. if (targetLevel == -1 || header.m_level == targetLevel) {
  665. if (firstHeader
  666. && (cursorLine == header.m_blockNumber
  667. || p_forward)
  668. && idx != -1) {
  669. // This header is not counted for the repeat.
  670. firstHeader = false;
  671. continue;
  672. }
  673. if (--p_repeat == 0) {
  674. // Found.
  675. break;
  676. }
  677. } else if (header.m_level < targetLevel) {
  678. // Stop by higher level.
  679. return false;
  680. }
  681. firstHeader = false;
  682. }
  683. if (targetIdx < 0 || targetIdx >= m_headers.size()) {
  684. return false;
  685. }
  686. // Jump to target header.
  687. int line = m_headers[targetIdx].m_blockNumber;
  688. if (line > -1) {
  689. QTextBlock block = document()->findBlockByNumber(line);
  690. if (block.isValid()) {
  691. cursor.setPosition(block.position());
  692. setTextCursor(cursor);
  693. return true;
  694. }
  695. }
  696. return false;
  697. }
  698. void VMdEditor::scrollBlockInPage(int p_blockNum, int p_dest)
  699. {
  700. VEditUtils::scrollBlockInPage(this, p_blockNum, p_dest);
  701. }
  702. void VMdEditor::updateTextEditConfig()
  703. {
  704. m_previewMgr->setPreviewEnabled(g_config->getEnablePreviewImages());
  705. setBlockImageEnabled(g_config->getEnablePreviewImages());
  706. setImageWidthConstrainted(g_config->getEnablePreviewImageConstraint());
  707. setLineLeading(m_config.m_lineDistanceHeight);
  708. setImageLineColor(g_config->getEditorPreviewImageLineFg());
  709. int lineNumber = g_config->getEditorLineNumber();
  710. if (lineNumber < (int)LineNumberType::None || lineNumber >= (int)LineNumberType::Invalid) {
  711. lineNumber = (int)LineNumberType::None;
  712. }
  713. setLineNumberType((LineNumberType)lineNumber);
  714. setLineNumberColor(g_config->getEditorLineNumberFg(),
  715. g_config->getEditorLineNumberBg());
  716. }
  717. void VMdEditor::updateConfig()
  718. {
  719. updateEditConfig();
  720. updateTextEditConfig();
  721. }