/*************************************************************************** * Copyright (C) 2017 by Julian Wolff * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * ***************************************************************************/ #include "converter.h" #include "generator_md.h" #include #include #include #include #include #include #include #include "debug_md.h" extern "C" { #include } // older versions of discount might not have these flags. // defining them to 0 allows us to convert without them #ifndef MKD_FENCEDCODE #define MKD_FENCEDCODE 0 #endif #ifndef MKD_GITHUBTAGS #define MKD_GITHUBTAGS 0 #endif #ifndef MKD_AUTOLINK #define MKD_AUTOLINK 0 #endif using namespace Markdown; Converter::Converter() : m_markdownFile( nullptr ) { } Converter::~Converter() { if ( m_markdownFile ) { fclose( m_markdownFile ); } } QTextDocument* Converter::convert( const QString &fileName ) { m_markdownFile = fopen( fileName.toLocal8Bit(), "rb" ); if ( !m_markdownFile ) { emit error( i18n( "Failed to open the document" ), -1 ); return nullptr; } m_fileDir = QDir( fileName.left( fileName.lastIndexOf( '/' ) ) ); QTextDocument *doc = convertOpenFile(); QHash internalLinks; QHash documentAnchors; extractLinks( doc->rootFrame(), internalLinks, documentAnchors ); for (auto linkIt = internalLinks.constBegin(); linkIt != internalLinks.constEnd(); ++linkIt) { auto anchorIt = documentAnchors.constFind(linkIt.key()); if (anchorIt != documentAnchors.constEnd()) { const Okular::DocumentViewport viewport = calculateViewport(doc, anchorIt.value()); Okular::GotoAction *action = new Okular::GotoAction(QString(), viewport); emit addAction(action, linkIt.value().position(), linkIt.value().position()+linkIt.value().length()); } else { qDebug() << "Could not find destination for" << linkIt.key(); } } return doc; } void Converter::convertAgain() { setDocument( convertOpenFile() ); } QTextDocument *Converter::convertOpenFile() { rewind( m_markdownFile ); MMIOT *markdownHandle = mkd_in( m_markdownFile, 0 ); int flags = MKD_FENCEDCODE | MKD_GITHUBTAGS | MKD_AUTOLINK | MKD_TOC | MKD_IDANCHOR; if (!MarkdownGenerator::isFancyPantsEnabled()) flags |= MKD_NOPANTS; if ( !mkd_compile( markdownHandle, flags ) ) { emit error( i18n( "Failed to compile the Markdown document." ), -1 ); return nullptr; } char *htmlDocument; const int size = mkd_document( markdownHandle, &htmlDocument ); const QString html = QString::fromUtf8( htmlDocument, size ); QTextDocument *textDocument = new QTextDocument; textDocument->setPageSize( QSizeF( 980, 1307 ) ); textDocument->setHtml( html ); textDocument->setDefaultFont( generator()->generalSettings()->font() ); mkd_cleanup( markdownHandle ); QTextFrameFormat frameFormat; frameFormat.setMargin( 45 ); QTextFrame *rootFrame = textDocument->rootFrame(); rootFrame->setFrameFormat( frameFormat ); convertImages( rootFrame, m_fileDir, textDocument ); return textDocument; } void Converter::extractLinks(QTextFrame * parent, QHash &internalLinks, QHash &documentAnchors) { for ( QTextFrame::iterator it = parent->begin(); !it.atEnd(); ++it ) { QTextFrame *textFrame = it.currentFrame(); const QTextBlock textBlock = it.currentBlock(); if ( textFrame ) { extractLinks(textFrame, internalLinks, documentAnchors); } else if ( textBlock.isValid() ) { extractLinks(textBlock, internalLinks, documentAnchors); } } } void Converter::extractLinks(const QTextBlock & parent, QHash &internalLinks, QHash &documentAnchors) { for ( QTextBlock::iterator it = parent.begin(); !it.atEnd(); ++it ) { const QTextFragment textFragment = it.fragment(); if ( textFragment.isValid() ) { const QTextCharFormat textCharFormat = textFragment.charFormat(); if ( textCharFormat.isAnchor() ) { const QString href = textCharFormat.anchorHref(); if ( href.startsWith( '#' ) ) { // It's an internal link, store it and we'll resolve it at the end internalLinks.insert(href.mid(1), textFragment); } else { Okular::BrowseAction *action = new Okular::BrowseAction( QUrl( textCharFormat.anchorHref() ) ); emit addAction( action, textFragment.position(), textFragment.position()+textFragment.length() ); } const QStringList anchorNames = textCharFormat.anchorNames(); for (const QString &anchorName : anchorNames) { documentAnchors.insert(anchorName, parent); } } } } } void Converter::convertImages(QTextFrame * parent, const QDir &dir, QTextDocument *textDocument) { for ( QTextFrame::iterator it = parent->begin(); !it.atEnd(); ++it ) { QTextFrame *textFrame = it.currentFrame(); const QTextBlock textBlock = it.currentBlock(); if ( textFrame ) { convertImages(textFrame, dir, textDocument); } else if ( textBlock.isValid() ) { convertImages(textBlock, dir, textDocument); } } } void Converter::convertImages(const QTextBlock & parent, const QDir &dir, QTextDocument *textDocument) { for ( QTextBlock::iterator it = parent.begin(); !it.atEnd(); ++it ) { const QTextFragment textFragment = it.fragment(); if ( textFragment.isValid() ) { const QTextCharFormat textCharFormat = textFragment.charFormat(); if( textCharFormat.isImageFormat() ) { //TODO: Show images from http URIs QTextImageFormat format; format.setName( QDir::cleanPath( dir.absoluteFilePath( textCharFormat.toImageFormat().name() ) ) ); const QImage img = QImage( format.name() ); if ( img.width() > 890 ) { format.setWidth( 890 ); format.setHeight( img.height() * 890. / img.width() ); } else { format.setWidth( img.width() ); format.setHeight( img.height() ); } QTextCursor cursor( textDocument ); cursor.setPosition( textFragment.position(), QTextCursor::MoveAnchor ); cursor.setPosition( textFragment.position() + textFragment.length(), QTextCursor::KeepAnchor ); cursor.removeSelectedText(); cursor.insertImage( format ); } } } }