fb2view.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  1. #include "fb2view.hpp"
  2. #include "fb2dlgs.hpp"
  3. #include "fb2read.hpp"
  4. #include "fb2save.hpp"
  5. #include "fb2utils.h"
  6. #include "fb2html.h"
  7. #include "fb2xml2.h"
  8. #include <QAction>
  9. #include <QtDebug>
  10. #include <QFileDialog>
  11. #include <QNetworkRequest>
  12. #include <QToolTip>
  13. #include <QUndoCommand>
  14. #include <QUndoStack>
  15. #include <QWebElement>
  16. #include <QWebInspector>
  17. #include <QWebFrame>
  18. #include <QWebPage>
  19. //---------------------------------------------------------------------------
  20. // Fb2NoteView
  21. //---------------------------------------------------------------------------
  22. class Fb2NoteView : public QWebView
  23. {
  24. public:
  25. explicit Fb2NoteView(QWidget *parent, const QUrl &url);
  26. void hint(const QWebElement element, const QRect &rect);
  27. protected:
  28. void paintEvent(QPaintEvent *event);
  29. const QUrl m_url;
  30. };
  31. Fb2NoteView::Fb2NoteView(QWidget *parent, const QUrl &url)
  32. : QWebView(parent)
  33. , m_url(url)
  34. {
  35. }
  36. void Fb2NoteView::paintEvent(QPaintEvent *event)
  37. {
  38. QWebView::paintEvent(event);
  39. QPainter painter(this);
  40. painter.setPen(Qt::black);
  41. QSize size = geometry().size() - QSize(1, 1);
  42. painter.drawRect( QRect(QPoint(0, 0), size) );
  43. }
  44. void Fb2NoteView::hint(const QWebElement element, const QRect &rect)
  45. {
  46. QString html = element.toOuterXml();
  47. html.prepend(
  48. "<body bgcolor=lightyellow style='overflow:hidden;padding:0;margin:0;margin-top:2;'>"
  49. "<div class=body fb2_name=notes style='padding:0;margin:0;'>"
  50. );
  51. html.append("</div></body>");
  52. setGeometry(rect);
  53. setHtml(html, m_url);
  54. show();
  55. }
  56. //---------------------------------------------------------------------------
  57. // Fb2WebPage
  58. //---------------------------------------------------------------------------
  59. Fb2WebPage::Fb2WebPage(QObject *parent)
  60. : QWebPage(parent)
  61. {
  62. QWebSettings *s = settings();
  63. s->setAttribute(QWebSettings::AutoLoadImages, true);
  64. s->setAttribute(QWebSettings::DeveloperExtrasEnabled, true);
  65. s->setAttribute(QWebSettings::JavaEnabled, false);
  66. s->setAttribute(QWebSettings::JavascriptEnabled, true);
  67. s->setAttribute(QWebSettings::PrivateBrowsingEnabled, true);
  68. s->setAttribute(QWebSettings::PluginsEnabled, false);
  69. s->setAttribute(QWebSettings::ZoomTextOnly, true);
  70. s->setUserStyleSheetUrl(QUrl::fromLocalFile(":style.css"));
  71. }
  72. bool Fb2WebPage::acceptNavigationRequest(QWebFrame *frame, const QNetworkRequest &request, NavigationType type)
  73. {
  74. Q_UNUSED(frame);
  75. if (type == NavigationTypeLinkClicked) {
  76. qCritical() << request.url().fragment();
  77. return false;
  78. // QToolTip::showText(request.url().fragment());
  79. }
  80. return QWebPage::acceptNavigationRequest(frame, request, type);
  81. }
  82. QWebElement Fb2WebPage::body()
  83. {
  84. return doc().findFirst("body");
  85. }
  86. QWebElement Fb2WebPage::doc()
  87. {
  88. return mainFrame()->documentElement();
  89. }
  90. class Fb2InsertBodyCommand : public QUndoCommand
  91. {
  92. public:
  93. explicit Fb2InsertBodyCommand(Fb2WebPage &page, QUndoCommand *parent = 0) : QUndoCommand(parent), m_page(page) {}
  94. virtual void undo();
  95. virtual void redo();
  96. private:
  97. Fb2WebPage & m_page;
  98. };
  99. void Fb2InsertBodyCommand::undo()
  100. {
  101. m_page.body().lastChild().removeFromDocument();
  102. Fb2WebElement(m_page.body().lastChild()).select();
  103. }
  104. void Fb2InsertBodyCommand::redo()
  105. {
  106. m_page.body().appendInside("<div class=body><div class=section><p>text</p></div></div>");
  107. Fb2WebElement(m_page.body().lastChild()).select();
  108. }
  109. void Fb2WebPage::insertBody()
  110. {
  111. undoStack()->beginMacro("Insert title");
  112. undoStack()->push(new Fb2InsertBodyCommand(*this));
  113. undoStack()->endMacro();
  114. emit contentsChanged();
  115. }
  116. //---------------------------------------------------------------------------
  117. // Fb2WebView
  118. //---------------------------------------------------------------------------
  119. Fb2WebView::Fb2WebView(QWidget *parent)
  120. : Fb2BaseWebView(parent)
  121. , m_inspector(0)
  122. , m_noteView(0)
  123. , m_thread(0)
  124. {
  125. setPage(new Fb2WebPage(this));
  126. page()->setNetworkAccessManager(new Fb2NetworkAccessManager(*this));
  127. page()->setContentEditable(true);
  128. connect(page(), SIGNAL(contentsChanged()), this, SLOT(fixContents()));
  129. connect(page(), SIGNAL(linkHovered(QString,QString,QString)), this, SLOT(linkHovered(QString,QString,QString)));
  130. connect(this, SIGNAL(loadFinished(bool)), SLOT(loadFinished()));
  131. }
  132. Fb2WebView::~Fb2WebView()
  133. {
  134. FB2DELETE(m_inspector);
  135. FB2DELETE(m_noteView);
  136. }
  137. Fb2NoteView & Fb2WebView::noteView()
  138. {
  139. if (m_noteView) return *m_noteView;
  140. m_noteView = new Fb2NoteView(qobject_cast<QWidget*>(parent()), url());
  141. m_noteView->setPage(new Fb2WebPage(this));
  142. m_noteView->page()->setNetworkAccessManager(page()->networkAccessManager());
  143. m_noteView->page()->setContentEditable(false);
  144. m_noteView->setGeometry(QRect(100, 100, 400, 200));
  145. return *m_noteView;
  146. }
  147. QWebElement Fb2WebView::body()
  148. {
  149. return doc().findFirst("body");
  150. }
  151. QWebElement Fb2WebView::doc()
  152. {
  153. return page()->mainFrame()->documentElement();
  154. }
  155. void Fb2WebView::fixContents()
  156. {
  157. foreach (QWebElement span, doc().findAll("span.apple-style-span[style]")) {
  158. span.removeAttribute("style");
  159. }
  160. }
  161. void Fb2WebView::mouseMoveEvent(QMouseEvent *event)
  162. {
  163. m_point = event->pos();
  164. QWebView::mouseMoveEvent(event);
  165. }
  166. void Fb2WebView::linkHovered(const QString &link, const QString &title, const QString &textContent)
  167. {
  168. Q_UNUSED(title);
  169. Q_UNUSED(textContent);
  170. const QString href = QUrl(link).fragment();
  171. if (href.isEmpty()) {
  172. if (m_noteView) m_noteView->hide();
  173. return;
  174. }
  175. const QString query = QString("DIV#%1").arg(href);
  176. const QWebElement element = doc().findFirst(query);
  177. if (element.isNull()) {
  178. if (m_noteView) m_noteView->hide();
  179. return;
  180. }
  181. QRect rect = geometry();
  182. QSize size = element.geometry().size() + QSize(2, 4);
  183. int center = rect.size().height() / 2;
  184. int h = size.height();
  185. if (h > center) size.setHeight(center - 10);
  186. int x = (rect.size().width() - size.width()) / 2;
  187. int y = m_point.y();
  188. if ( y > h ) y = y - h - 10; else y = y + 10;
  189. QPoint point = QPoint(x, y) + rect.topLeft();
  190. noteView().hint(element, QRect(point, size));
  191. }
  192. void Fb2WebView::load(const QString &filename, const QString &xml)
  193. {
  194. if (m_thread) return;
  195. m_thread = new Fb2ReadThread(this, filename, xml);
  196. m_thread->start();
  197. }
  198. bool Fb2WebView::save(QIODevice *device, const QString &codec)
  199. {
  200. Fb2SaveWriter writer(*this, device);
  201. if (!codec.isEmpty()) writer.setCodec(codec.toLatin1());
  202. bool ok = Fb2SaveHandler(writer).save();
  203. if (ok) page()->undoStack()->setClean();
  204. return ok;
  205. }
  206. bool Fb2WebView::save(QByteArray *array)
  207. {
  208. Fb2SaveWriter writer(*this, array);
  209. return Fb2SaveHandler(writer).save();
  210. }
  211. bool Fb2WebView::save(QString *string)
  212. {
  213. // Use class QByteArray instead QString
  214. // to store information about encoding.
  215. QByteArray data;
  216. bool ok = save(&data);
  217. if (ok) *string = QString::fromUtf8(data.data());
  218. return ok;
  219. }
  220. void Fb2WebView::data(QString name, QByteArray data)
  221. {
  222. m_files.set(name, data);
  223. }
  224. void Fb2WebView::html(QString name, QString html)
  225. {
  226. static int number = 0;
  227. setHtml(html, QUrl(QString("fb2:/%1/").arg(number++)));
  228. if (m_thread) m_thread->deleteLater();
  229. m_thread = 0;
  230. }
  231. void Fb2WebView::zoomIn()
  232. {
  233. qreal zoom = zoomFactor();
  234. setZoomFactor(zoom * 1.1);
  235. }
  236. void Fb2WebView::zoomOut()
  237. {
  238. qreal zoom = zoomFactor();
  239. setZoomFactor(zoom * 0.9);
  240. }
  241. void Fb2WebView::zoomReset()
  242. {
  243. setZoomFactor(1);
  244. }
  245. bool Fb2WebView::UndoEnabled()
  246. {
  247. return pageAction(QWebPage::Undo)->isEnabled();
  248. }
  249. bool Fb2WebView::RedoEnabled()
  250. {
  251. return pageAction(QWebPage::Redo)->isEnabled();
  252. }
  253. bool Fb2WebView::CutEnabled()
  254. {
  255. return pageAction(QWebPage::Cut)->isEnabled();
  256. }
  257. bool Fb2WebView::CopyEnabled()
  258. {
  259. return pageAction(QWebPage::Copy)->isEnabled();
  260. }
  261. bool Fb2WebView::BoldChecked()
  262. {
  263. return pageAction(QWebPage::ToggleBold)->isChecked();
  264. }
  265. bool Fb2WebView::ItalicChecked()
  266. {
  267. return pageAction(QWebPage::ToggleItalic)->isChecked();
  268. }
  269. bool Fb2WebView::StrikeChecked()
  270. {
  271. return pageAction(QWebPage::ToggleStrikethrough)->isChecked();
  272. }
  273. bool Fb2WebView::SubChecked()
  274. {
  275. return pageAction(QWebPage::ToggleSubscript)->isChecked();
  276. }
  277. bool Fb2WebView::SupChecked()
  278. {
  279. return pageAction(QWebPage::ToggleSuperscript)->isChecked();
  280. }
  281. void Fb2WebView::find()
  282. {
  283. Fb2TextFindDlg dlg(*this);
  284. dlg.exec();
  285. }
  286. void Fb2WebView::insertImage()
  287. {
  288. QString filters;
  289. filters += tr("Common Graphics (*.png *.jpg *.jpeg *.gif);;");
  290. filters += tr("Portable Network Graphics (PNG) (*.png);;");
  291. filters += tr("JPEG (*.jpg *.jpeg);;");
  292. filters += tr("Graphics Interchange Format (*.gif);;");
  293. filters += tr("All Files (*)");
  294. QString path = QFileDialog::getOpenFileName(this, tr("Insert image..."), QString(), filters);
  295. if (path.isEmpty()) return;
  296. QFile file(path);
  297. if (!file.open(QIODevice::ReadOnly)) return;
  298. QByteArray data = file.readAll();
  299. QString name = m_files.add(path, data);
  300. execCommand("insertImage", name.prepend("#"));
  301. }
  302. void Fb2WebView::insertNote()
  303. {
  304. Fb2NoteDlg dlg(*this);
  305. dlg.exec();
  306. }
  307. void Fb2WebView::insertLink()
  308. {
  309. }
  310. void Fb2WebView::execCommand(const QString &cmd, const QString &arg)
  311. {
  312. QString javascript = QString("document.execCommand(\"%1\",false,\"%2\")").arg(cmd).arg(arg);
  313. page()->mainFrame()->evaluateJavaScript(javascript);
  314. }
  315. QString Fb2WebView::status()
  316. {
  317. static const QString javascript = FB2::read(":/js/get_status.js");
  318. return page()->mainFrame()->evaluateJavaScript(javascript).toString();
  319. return QString();
  320. }
  321. void Fb2WebView::showInspector()
  322. {
  323. if (!m_inspector) {
  324. m_inspector = new QWebInspector();
  325. m_inspector->setAttribute(Qt::WA_DeleteOnClose, false);
  326. m_inspector->setPage(page());
  327. }
  328. m_inspector->show();
  329. }
  330. void Fb2WebView::loadFinished()
  331. {
  332. Fb2WebElement element = body().findFirst("div.body");
  333. if (element.isNull()) element = body();
  334. element.select();
  335. }
  336. void Fb2WebView::insertTitle()
  337. {
  338. page()->undoStack()->beginMacro("Insert title");
  339. static const QString javascript = FB2::read(":/js/insert_title.js");
  340. page()->mainFrame()->evaluateJavaScript(javascript);
  341. page()->undoStack()->endMacro();
  342. }