fb2view.cpp 11 KB

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