fb2view.cpp 11 KB

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