fb2main.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594
  1. #include <QtGui>
  2. #include <QtDebug>
  3. #include "fb2main.h"
  4. #include "fb2doc.h"
  5. #include <Qsci/qsciscintilla.h>
  6. #include <Qsci/qscilexerxml.h>
  7. MainWindow::MainWindow()
  8. {
  9. init();
  10. createText();
  11. setCurrentFile("");
  12. }
  13. MainWindow::MainWindow(const QString &filename)
  14. {
  15. init();
  16. createQsci();
  17. loadXML(filename);
  18. setCurrentFile(filename);
  19. }
  20. MainWindow::MainWindow(const QString &filename, Fb2MainDocument * document)
  21. {
  22. init();
  23. createText();
  24. if (!document) document = loadFB2(filename);
  25. setCurrentFile(filename, document);
  26. }
  27. bool MainWindow::loadXML(const QString &filename)
  28. {
  29. if (!filename.isEmpty()) {
  30. QFile file(filename);
  31. if (file.open(QFile::ReadOnly | QFile::Text)) {
  32. qsciEdit->clear();
  33. return qsciEdit->read(&file);
  34. }
  35. }
  36. return false;
  37. }
  38. Fb2MainDocument * MainWindow::loadFB2(const QString &filename)
  39. {
  40. if (filename.isEmpty()) return NULL;
  41. QFile file(filename);
  42. if (!file.open(QFile::ReadOnly | QFile::Text)) {
  43. qCritical() << tr("Cannot read file %1:\n%2.").arg(filename).arg(file.errorString());
  44. return NULL;
  45. }
  46. return Fb2MainDocument::load(file);
  47. }
  48. void MainWindow::closeEvent(QCloseEvent *event)
  49. {
  50. if (maybeSave()) {
  51. writeSettings();
  52. event->accept();
  53. } else {
  54. event->ignore();
  55. }
  56. }
  57. void MainWindow::fileNew()
  58. {
  59. MainWindow *other = new MainWindow;
  60. other->move(x() + 40, y() + 40);
  61. other->show();
  62. }
  63. void MainWindow::fileOpen()
  64. {
  65. QString filename = QFileDialog::getOpenFileName(this);
  66. if (filename.isEmpty()) return;
  67. MainWindow * existing = findMainWindow(filename);
  68. if (existing) {
  69. existing->show();
  70. existing->raise();
  71. existing->activateWindow();
  72. return;
  73. }
  74. if (textEdit) {
  75. Fb2MainDocument * document = loadFB2(filename);
  76. if (!document) return;
  77. if (isUntitled && textEdit->document()->isEmpty() && !isWindowModified()) {
  78. setCurrentFile(filename, document);
  79. } else {
  80. MainWindow * other = new MainWindow(filename, document);
  81. other->move(x() + 40, y() + 40);
  82. other->show();
  83. }
  84. } else if (qsciEdit) {
  85. if (isUntitled && !isWindowModified()) {
  86. loadXML(filename);
  87. setCurrentFile(filename);
  88. } else {
  89. MainWindow * other = new MainWindow(filename);
  90. other->move(x() + 40, y() + 40);
  91. other->show();
  92. }
  93. }
  94. }
  95. bool MainWindow::fileSave()
  96. {
  97. if (isUntitled) {
  98. return fileSaveAs();
  99. } else {
  100. return saveFile(curFile);
  101. }
  102. }
  103. bool MainWindow::fileSaveAs()
  104. {
  105. QString fileName = QFileDialog::getSaveFileName(this, tr("Save As"), curFile);
  106. if (fileName.isEmpty()) return false;
  107. return saveFile(fileName);
  108. }
  109. void MainWindow::about()
  110. {
  111. QMessageBox::about(this, tr("About SDI"),
  112. tr("The <b>SDI</b> example demonstrates how to write single "
  113. "document interface applications using Qt."));
  114. }
  115. void MainWindow::documentWasModified()
  116. {
  117. QFileInfo info = windowFilePath();
  118. QString title = info.fileName();
  119. title += QString("[*]") += QString(" - ") += qApp->applicationName();
  120. setWindowTitle(title);
  121. setWindowModified(true);
  122. }
  123. void MainWindow::init()
  124. {
  125. setAttribute(Qt::WA_DeleteOnClose);
  126. isUntitled = true;
  127. createActions();
  128. createStatusBar();
  129. textEdit = NULL;
  130. noteEdit = NULL;
  131. qsciEdit = NULL;
  132. readSettings();
  133. setUnifiedTitleAndToolBarOnMac(true);
  134. }
  135. QIcon MainWindow::icon(const QString &name)
  136. {
  137. QString file = QString(":/images/%1.png").arg(name);
  138. return QIcon::fromTheme(name, QIcon(file));
  139. }
  140. void MainWindow::createActions()
  141. {
  142. QAction * act;
  143. QMenu * menu;
  144. QToolBar * tool;
  145. menu = menuBar()->addMenu(tr("&File"));
  146. tool = addToolBar(tr("File"));
  147. act = new QAction(icon("document-new"), tr("&New"), this);
  148. act->setPriority(QAction::LowPriority);
  149. act->setShortcuts(QKeySequence::New);
  150. act->setStatusTip(tr("Create a new file"));
  151. connect(act, SIGNAL(triggered()), this, SLOT(fileNew()));
  152. menu->addAction(act);
  153. tool->addAction(act);
  154. act = new QAction(icon("document-open"), tr("&Open..."), this);
  155. act->setShortcuts(QKeySequence::Open);
  156. act->setStatusTip(tr("Open an existing file"));
  157. connect(act, SIGNAL(triggered()), this, SLOT(fileOpen()));
  158. menu->addAction(act);
  159. tool->addAction(act);
  160. act = new QAction(icon("document-save"), tr("&Save"), this);
  161. act->setShortcuts(QKeySequence::Save);
  162. act->setStatusTip(tr("Save the document to disk"));
  163. connect(act, SIGNAL(triggered()), this, SLOT(fileSave()));
  164. menu->addAction(act);
  165. tool->addAction(act);
  166. act = new QAction(icon("document-save-as"), tr("Save &As..."), this);
  167. act->setShortcuts(QKeySequence::SaveAs);
  168. act->setStatusTip(tr("Save the document under a new name"));
  169. connect(act, SIGNAL(triggered()), this, SLOT(fileSaveAs()));
  170. menu->addAction(act);
  171. menu->addSeparator();
  172. act = new QAction(icon("window-close"), tr("&Close"), this);
  173. act->setShortcuts(QKeySequence::Close);
  174. act->setStatusTip(tr("Close this window"));
  175. connect(act, SIGNAL(triggered()), this, SLOT(close()));
  176. menu->addAction(act);
  177. act = new QAction(icon("application-exit"), tr("E&xit"), this);
  178. act->setShortcuts(QKeySequence::Quit);
  179. act->setStatusTip(tr("Exit the application"));
  180. connect(act, SIGNAL(triggered()), qApp, SLOT(closeAllWindows()));
  181. menu->addAction(act);
  182. menu = menuBar()->addMenu(tr("&Edit"));
  183. tool = addToolBar(tr("Edit"));
  184. connect(QApplication::clipboard(), SIGNAL(dataChanged()), this, SLOT(clipboardDataChanged()));
  185. actionUndo = act = new QAction(icon("edit-undo"), tr("&Undo"), this);
  186. act->setPriority(QAction::LowPriority);
  187. act->setShortcut(QKeySequence::Undo);
  188. menu->addAction(act);
  189. tool->addAction(act);
  190. actionRedo = act = new QAction(icon("edit-redo"), tr("&Redo"), this);
  191. act->setPriority(QAction::LowPriority);
  192. act->setShortcut(QKeySequence::Redo);
  193. menu->addAction(act);
  194. tool->addAction(act);
  195. menu->addSeparator();
  196. tool->addSeparator();
  197. actionCut = act = new QAction(icon("edit-cut"), tr("Cu&t"), this);
  198. act->setPriority(QAction::LowPriority);
  199. act->setShortcuts(QKeySequence::Cut);
  200. act->setStatusTip(tr("Cut the current selection's contents to the clipboard"));
  201. act->setEnabled(false);
  202. menu->addAction(act);
  203. tool->addAction(act);
  204. actionCopy = act = new QAction(icon("edit-copy"), tr("&Copy"), this);
  205. act->setPriority(QAction::LowPriority);
  206. act->setShortcuts(QKeySequence::Copy);
  207. act->setStatusTip(tr("Copy the current selection's contents to the clipboard"));
  208. act->setEnabled(false);
  209. menu->addAction(act);
  210. tool->addAction(act);
  211. actionPaste = act = new QAction(icon("edit-paste"), tr("&Paste"), this);
  212. act->setPriority(QAction::LowPriority);
  213. act->setShortcuts(QKeySequence::Paste);
  214. act->setStatusTip(tr("Paste the clipboard's contents into the current selection"));
  215. menu->addAction(act);
  216. tool->addAction(act);
  217. clipboardDataChanged();
  218. menu = menuBar()->addMenu(tr("Format"));
  219. tool = addToolBar(tr("Format"));
  220. actionTextBold = act = new QAction(icon("format-text-bold"), tr("Bold"), this);
  221. act->setShortcuts(QKeySequence::Bold);
  222. act->setCheckable(true);
  223. connect(act, SIGNAL(triggered()), this, SLOT(textBold()));
  224. menu->addAction(act);
  225. tool->addAction(act);
  226. actionTextItalic = act = new QAction(icon("format-text-italic"), tr("Italic"), this);
  227. act->setShortcuts(QKeySequence::Italic);
  228. act->setCheckable(true);
  229. connect(act, SIGNAL(triggered()), this, SLOT(textItalic()));
  230. menu->addAction(act);
  231. tool->addAction(act);
  232. actionTextUnder = act = new QAction(icon("format-text-underline"), tr("Underline"), this);
  233. act->setShortcuts(QKeySequence::Underline);
  234. act->setCheckable(true);
  235. connect(act, SIGNAL(triggered()), this, SLOT(textUnder()));
  236. menu->addAction(act);
  237. tool->addAction(act);
  238. actionTextStrike = act = new QAction(icon("format-text-strikethrough"), tr("Strikethrough"), this);
  239. act->setCheckable(true);
  240. connect(act, SIGNAL(triggered()), this, SLOT(textStrike()));
  241. menu->addAction(act);
  242. tool->addAction(act);
  243. // format-text-subscript
  244. // format-text-superscript
  245. menu = menuBar()->addMenu(tr("&View"));
  246. QAction * actText = act = new QAction(tr("&Text"), this);
  247. act->setCheckable(true);
  248. connect(act, SIGNAL(triggered()), this, SLOT(viewText()));
  249. menu->addAction(act);
  250. QAction * actQsci = act = new QAction(tr("&XML"), this);
  251. act->setCheckable(true);
  252. connect(act, SIGNAL(triggered()), this, SLOT(viewQsci()));
  253. menu->addAction(act);
  254. QActionGroup * viewGroup = new QActionGroup(this);
  255. viewGroup->addAction(actText);
  256. viewGroup->addAction(actQsci);
  257. actText->setChecked(true);
  258. menuBar()->addSeparator();
  259. menu = menuBar()->addMenu(tr("&Help"));
  260. act = new QAction(icon("help-about"), tr("&About"), this);
  261. act->setStatusTip(tr("Show the application's About box"));
  262. connect(act, SIGNAL(triggered()), this, SLOT(about()));
  263. menu->addAction(act);
  264. act = new QAction(tr("About &Qt"), this);
  265. act->setStatusTip(tr("Show the Qt library's About box"));
  266. connect(act, SIGNAL(triggered()), qApp, SLOT(aboutQt()));
  267. menu->addAction(act);
  268. }
  269. void MainWindow::connectTextDocument(QTextDocument * document)
  270. {
  271. connect(document, SIGNAL(contentsChanged()), this, SLOT(documentWasModified()));
  272. connect(document, SIGNAL(undoAvailable(bool)), actionUndo, SLOT(setEnabled(bool)));
  273. connect(document, SIGNAL(redoAvailable(bool)), actionRedo, SLOT(setEnabled(bool)));
  274. }
  275. void MainWindow::createText()
  276. {
  277. textEdit = new QTextEdit;
  278. textEdit->setAcceptRichText(true);
  279. setCentralWidget(textEdit);
  280. connect(actionCut, SIGNAL(triggered()), textEdit, SLOT(cut()));
  281. connect(actionCopy, SIGNAL(triggered()), textEdit, SLOT(copy()));
  282. connect(actionPaste, SIGNAL(triggered()), textEdit, SLOT(paste()));
  283. connect(textEdit, SIGNAL(copyAvailable(bool)), actionCut, SLOT(setEnabled(bool)));
  284. connect(textEdit, SIGNAL(copyAvailable(bool)), actionCopy, SLOT(setEnabled(bool)));
  285. connect(textEdit, SIGNAL(currentCharFormatChanged(const QTextCharFormat &)), this, SLOT(currentCharFormatChanged(const QTextCharFormat &)));
  286. connect(actionUndo, SIGNAL(triggered()), textEdit, SLOT(undo()));
  287. connect(actionRedo, SIGNAL(triggered()), textEdit, SLOT(redo()));
  288. actionUndo->setEnabled(false);
  289. actionRedo->setEnabled(false);
  290. connectTextDocument(textEdit->document());
  291. }
  292. void MainWindow::createQsci()
  293. {
  294. // http://qtcoder.blogspot.com/2010/10/qscintills.html
  295. // http://www.riverbankcomputing.co.uk/static/Docs/QScintilla2/classQsciScintilla.html
  296. qsciEdit = new QsciScintilla;
  297. qsciEdit->setUtf8(true);
  298. qsciEdit->setCaretLineVisible(true);
  299. qsciEdit->setCaretLineBackgroundColor(QColor("gainsboro"));
  300. qsciEdit->setWrapMode(QsciScintilla::WrapWord);
  301. qsciEdit->setEolMode(QsciScintilla::EolWindows);
  302. qsciEdit->setAutoIndent(true);
  303. qsciEdit->setIndentationGuides(true);
  304. qsciEdit->setAutoCompletionSource(QsciScintilla::AcsAll);
  305. qsciEdit->setAutoCompletionCaseSensitivity(true);
  306. qsciEdit->setAutoCompletionReplaceWord(true);
  307. qsciEdit->setAutoCompletionShowSingle(true);
  308. qsciEdit->setAutoCompletionThreshold(2);
  309. qsciEdit->setMarginsBackgroundColor(QColor("gainsboro"));
  310. qsciEdit->setMarginWidth(0, 0);
  311. qsciEdit->setMarginLineNumbers(1, true);
  312. qsciEdit->setMarginWidth(1, QString("1000"));
  313. qsciEdit->setFolding(QsciScintilla::BoxedFoldStyle, 2);
  314. qsciEdit->setBraceMatching(QsciScintilla::SloppyBraceMatch);
  315. qsciEdit->setMatchedBraceBackgroundColor(Qt::yellow);
  316. qsciEdit->setUnmatchedBraceForegroundColor(Qt::blue);
  317. QFont font("Courier", 10);
  318. font.setStyleHint(QFont::TypeWriter);
  319. QsciLexerXML * lexer = new QsciLexerXML;
  320. lexer->setFont(font, -1);
  321. qsciEdit->setBraceMatching(QsciScintilla::SloppyBraceMatch);
  322. qsciEdit->setLexer(lexer);
  323. setCentralWidget(qsciEdit);
  324. qsciEdit->setFocus();
  325. // connect(qsciEdit, SIGNAL(textChanged()), this, SLOT(documentWasModified()));
  326. // connect(qsciEdit, SIGNAL(cursorPositionChanged(int, int)), this, SLOT(cursorMoved(int, int)));
  327. actionUndo->setEnabled(false);
  328. actionRedo->setEnabled(false);
  329. }
  330. void MainWindow::createStatusBar()
  331. {
  332. statusBar()->showMessage(tr("Ready"));
  333. }
  334. void MainWindow::readSettings()
  335. {
  336. QSettings settings;
  337. QPoint pos = settings.value("pos", QPoint(200, 200)).toPoint();
  338. QSize size = settings.value("size", QSize(400, 400)).toSize();
  339. move(pos);
  340. resize(size);
  341. }
  342. void MainWindow::writeSettings()
  343. {
  344. QSettings settings;
  345. settings.setValue("pos", pos());
  346. settings.setValue("size", size());
  347. }
  348. bool MainWindow::maybeSave()
  349. {
  350. if (textEdit && textEdit->document()->isModified()) {
  351. QMessageBox::StandardButton ret;
  352. ret = QMessageBox::warning(this, tr("SDI"),
  353. tr("The document has been modified.\n"
  354. "Do you want to save your changes?"),
  355. QMessageBox::Save | QMessageBox::Discard
  356. | QMessageBox::Cancel);
  357. if (ret == QMessageBox::Save)
  358. return fileSave();
  359. else if (ret == QMessageBox::Cancel)
  360. return false;
  361. }
  362. return true;
  363. }
  364. bool MainWindow::saveFile(const QString &fileName)
  365. {
  366. return false;
  367. QFile file(fileName);
  368. if (!file.open(QFile::WriteOnly | QFile::Text)) {
  369. QMessageBox::warning(this, tr("SDI"),
  370. tr("Cannot write file %1:\n%2.")
  371. .arg(fileName)
  372. .arg(file.errorString()));
  373. return false;
  374. }
  375. QTextStream out(&file);
  376. QApplication::setOverrideCursor(Qt::WaitCursor);
  377. out << textEdit->toPlainText();
  378. QApplication::restoreOverrideCursor();
  379. setCurrentFile(fileName);
  380. statusBar()->showMessage(tr("File saved"), 2000);
  381. return true;
  382. }
  383. void MainWindow::setCurrentFile(const QString &filename, Fb2MainDocument * document)
  384. {
  385. static int sequenceNumber = 1;
  386. QString title;
  387. isUntitled = filename.isEmpty();
  388. if (isUntitled) {
  389. curFile = tr("book%1.fb2").arg(sequenceNumber++);
  390. title = curFile;
  391. } else {
  392. QFileInfo info = filename;
  393. curFile = info.canonicalFilePath();
  394. title = info.fileName();
  395. }
  396. title += QString(" - ") += qApp->applicationName();
  397. if (textEdit && document) {
  398. textEdit->setDocument(document);
  399. connectTextDocument(textEdit->document());
  400. if (!document->child().isEmpty()) {
  401. if (!noteEdit) {
  402. noteEdit = new QTextEdit(this);
  403. noteEdit->setDocument(&document->child());
  404. QDockWidget * dock = new QDockWidget("Footnotes", this);
  405. dock->setAttribute(Qt::WA_DeleteOnClose);
  406. dock->setFeatures(QDockWidget::AllDockWidgetFeatures);
  407. dock->setWidget(noteEdit);
  408. addDockWidget(Qt::BottomDockWidgetArea, dock);
  409. }
  410. }
  411. }
  412. setWindowModified(false);
  413. setWindowFilePath(curFile);
  414. setWindowTitle(title);
  415. }
  416. MainWindow *MainWindow::findMainWindow(const QString &fileName)
  417. {
  418. QString canonicalFilePath = QFileInfo(fileName).canonicalFilePath();
  419. foreach (QWidget *widget, qApp->topLevelWidgets()) {
  420. MainWindow *mainWin = qobject_cast<MainWindow *>(widget);
  421. if (mainWin && mainWin->curFile == canonicalFilePath)
  422. return mainWin;
  423. }
  424. return 0;
  425. }
  426. void MainWindow::viewQsci()
  427. {
  428. if (centralWidget() == qsciEdit) return;
  429. if (textEdit) { delete textEdit; textEdit = NULL; }
  430. createQsci();
  431. }
  432. void MainWindow::viewText()
  433. {
  434. if (centralWidget() == textEdit) return;
  435. if (qsciEdit) { delete qsciEdit; qsciEdit = NULL; }
  436. createText();
  437. }
  438. void MainWindow::currentCharFormatChanged(const QTextCharFormat &format)
  439. {
  440. QFont font = format.font();
  441. actionTextBold -> setChecked(font.bold());
  442. actionTextItalic -> setChecked(font.italic());
  443. actionTextUnder -> setChecked(font.underline());
  444. actionTextStrike -> setChecked(font.strikeOut());
  445. }
  446. void MainWindow::cursorPositionChanged()
  447. {
  448. // alignmentChanged(textEdit->alignment());
  449. }
  450. void MainWindow::clipboardDataChanged()
  451. {
  452. if (const QMimeData *md = QApplication::clipboard()->mimeData()) {
  453. actionPaste->setEnabled(md->hasText());
  454. }
  455. }
  456. void MainWindow::textBold()
  457. {
  458. QTextCharFormat fmt;
  459. fmt.setFontWeight(actionTextBold->isChecked() ? QFont::Bold : QFont::Normal);
  460. mergeFormatOnWordOrSelection(fmt);
  461. }
  462. void MainWindow::textUnder()
  463. {
  464. QTextCharFormat fmt;
  465. fmt.setFontUnderline(actionTextUnder->isChecked());
  466. mergeFormatOnWordOrSelection(fmt);
  467. }
  468. void MainWindow::textItalic()
  469. {
  470. QTextCharFormat fmt;
  471. fmt.setFontItalic(actionTextItalic->isChecked());
  472. mergeFormatOnWordOrSelection(fmt);
  473. }
  474. void MainWindow::textStrike()
  475. {
  476. QTextCharFormat fmt;
  477. fmt.setFontStrikeOut(actionTextStrike->isChecked());
  478. mergeFormatOnWordOrSelection(fmt);
  479. }
  480. void MainWindow::mergeFormatOnWordOrSelection(const QTextCharFormat &format)
  481. {
  482. if (!textEdit) return;
  483. QTextCursor cursor = textEdit->textCursor();
  484. cursor.beginEditBlock();
  485. if (!cursor.hasSelection()) cursor.select(QTextCursor::WordUnderCursor);
  486. cursor.mergeCharFormat(format);
  487. textEdit->mergeCurrentCharFormat(format);
  488. cursor.endEditBlock();
  489. }