21#include "markupdirector.h"
22#include "markupdirector_p.h"
24#include "abstractmarkupbuilder.h"
26#include <QtCore/QFlags>
28#include <QtCore/QStack>
29#include <QtCore/QString>
30#include <QtGui/QBrush>
31#include <QtGui/QColor>
32#include <QtGui/QTextCharFormat>
33#include <QtGui/QTextCursor>
34#include <QtGui/QTextDocument>
35#include <QtGui/QTextDocumentFragment>
36#include <QtGui/QTextFrame>
37#include <QtGui/QTextList>
38#include <QtGui/QTextTable>
43 : d_ptr(new MarkupDirectorPrivate(this)),
m_builder(builder)
52 while (!start.
atEnd() && start != end) {
55 auto table = qobject_cast<QTextTable *>(frame);
63 Q_ASSERT(block.isValid());
101 auto format = table->
format();
105 auto tableWidth = format.width();
109 sWidth = QStringLiteral(
"%1%");
110 sWidth = sWidth.
arg(tableWidth.rawValue());
112 sWidth = QStringLiteral(
"%1");
113 sWidth = sWidth.
arg(tableWidth.rawValue());
116 m_builder->beginTable(format.cellPadding(), format.cellSpacing(), sWidth);
118 auto headerRowCount = format.headerRowCount();
122 for (
auto row = 0; row < table->
rows(); ++row) {
136 for (
auto column = 0; column < table->
columns(); ++column) {
138 auto tableCell = table->
cellAt(row, column);
141 auto rowSpan = tableCell.rowSpan();
142 if ((rowSpan > 1) || (columnSpan > 1)) {
143 if (alreadyProcessedCells.
contains(tableCell)) {
147 alreadyProcessedCells.
append(tableCell);
151 auto cellWidth = colLengths.at(column);
156 sCellWidth = QStringLiteral(
"%1%");
157 sCellWidth = sCellWidth.
arg(cellWidth.rawValue());
159 sCellWidth = QStringLiteral(
"%1");
160 sCellWidth = sCellWidth.
arg(cellWidth.rawValue());
164 if (row < headerRowCount) {
165 m_builder->beginTableHeaderCell(sCellWidth, columnSpan, rowSpan);
167 m_builder->beginTableCell(sCellWidth, columnSpan, rowSpan);
172 if (row < headerRowCount) {
194std::pair<QTextFrame::iterator, QTextBlock>
201 while (block.isValid() && block.textList()) {
208 block = block.next();
209 if (block.isValid()) {
210 auto obj = block.document()->objectForFormat(block.blockFormat());
211 auto group = qobject_cast<QTextBlockGroup *>(obj);
212 if (group && group != list) {
228 auto blockAlignment = blockFormat.
alignment();
234 if (!frameIt.
atEnd())
239 auto it = block.
begin();
246 if (!frameIt.
atEnd())
267 while (!it.atEnd()) {
276 if (!frameIt.
atEnd())
355 auto paraClosed =
false;
366 }
else if (paraClosed) {
391 auto group = qobject_cast<QTextBlockGroup *>(
object);
400std::pair<QTextFrame::iterator, QTextBlock>
406 auto lastBlock = _block;
408 auto obj = block.document()->objectForFormat(block.blockFormat());
412 return {lastIt, lastBlock};
414 auto group = qobject_cast<QTextBlockGroup *>(obj);
416 return {lastIt, lastBlock};
418 while (block.isValid()) {
422 block = block.next();
426 obj = block.document()->objectForFormat(block.blockFormat());
430 nextGroup = qobject_cast<QTextBlockGroup *>(obj);
432 if (group == blockGroup || !nextGroup) {
438 return {lastIt, lastBlock};
441std::pair<QTextFrame::iterator, QTextBlock>
446 auto list = qobject_cast<QTextList *>(blockGroup);
464 if (fragmentFormat.isImageFormat()) {
465 auto imageFormat = fragmentFormat.toImageFormat();
496 if (d->m_openElements.isEmpty())
502 auto remainingSize = elementsToClose.size();
503 while (!elementsToClose.isEmpty()) {
504 auto tag = d->m_openElements.last();
505 if (elementsToClose.contains(tag)) {
544 d->m_openElements.removeLast();
545 elementsToClose.remove(tag);
547 previousSize = remainingSize;
548 remainingSize = elementsToClose.size();
550 if (previousSize == remainingSize) {
558 elementsToClose.insert(d->m_openElements.last());
568 if (!fragment.isValid())
574 Q_FOREACH (
int tag, elementsToOpenList) {
589 m_builder->beginFontPointSize(fragmentFormat.font().pointSize());
590 d->m_openFontPointSize = fragmentFormat.font().pointSize();
593 d->m_openFontFamily = fragmentFormat.fontFamilies().toStringList().first();
594 m_builder->beginFontFamily(d->m_openFontFamily);
597 m_builder->beginBackground(fragmentFormat.background());
598 d->m_openBackground = fragmentFormat.background();
601 m_builder->beginForeground(fragmentFormat.foreground());
602 d->m_openForeground = fragmentFormat.foreground();
606 auto anchorNames = fragmentFormat.anchorNames();
607 if (!anchorNames.isEmpty()) {
608 while (!anchorNames.isEmpty()) {
609 auto n = anchorNames.last();
610 anchorNames.removeLast();
611 if (anchorNames.isEmpty()) {
613 m_builder->beginAnchor(fragmentFormat.anchorHref(), n);
623 m_builder->beginAnchor(fragmentFormat.anchorHref());
625 d->m_openAnchorHref = fragmentFormat.anchorHref();
637 d->m_openElements.append(tag);
638 d->m_elementsToOpen.remove(tag);
649 auto elementsToClose =
QSet<int>(d->m_openElements.begin(), d->m_openElements.end());
650 return elementsToClose.unite(d->m_elementsToOpen);
655 if (!fragment.isValid())
656 return closedElements;
658 auto fragmentFormat = fragment.charFormat();
660 auto fontWeight = fragmentFormat.fontWeight();
661 auto fontItalic = fragmentFormat.fontItalic();
662 auto fontUnderline = fragmentFormat.fontUnderline();
663 auto fontStrikeout = fragmentFormat.fontStrikeOut();
665 auto fontForeground = fragmentFormat.foreground();
666 auto fontBackground = fragmentFormat.background();
668 const QStringList fontFamilies = fragmentFormat.fontFamilies().toStringList();
670 auto fontPointSize = fragmentFormat.font().pointSize();
671 auto anchorHref = fragmentFormat.anchorHref();
673 auto vAlign = fragmentFormat.verticalAlignment();
678 && (d->m_openElements.contains(
StrikeOut)
679 || d->m_elementsToOpen.contains(
StrikeOut))) {
684 && (d->m_openElements.contains(
Underline)
685 || d->m_elementsToOpen.contains(
Underline))
686 && !(d->m_openElements.contains(
Anchor)
687 || d->m_elementsToOpen.contains(
Anchor))) {
692 && (d->m_openElements.contains(
Emph)
693 || d->m_elementsToOpen.contains(
Emph))) {
698 && (d->m_openElements.contains(
Strong)
699 || d->m_elementsToOpen.contains(
Strong))) {
705 && (d->m_openFontPointSize != fontPointSize)) {
711 && (d->m_openFontFamily != fontFamily)) {
716 && (d->m_openBackground != fontBackground))
718 && (d->m_backgroundToOpen != fontBackground))) {
723 && (d->m_openForeground != fontForeground))
725 && (d->m_foregroundToOpen != fontForeground))) {
729 if ((d->m_openElements.contains(
Anchor)
730 && (d->m_openAnchorHref != anchorHref))
731 || (d->m_elementsToOpen.contains(
Anchor)
732 && (d->m_anchorHrefToOpen != anchorHref))) {
737 && (d->m_openElements.contains(
SubScript)
738 || d->m_elementsToOpen.contains(
SubScript))) {
747 return closedElements;
754 if (!fragment.isValid()) {
759 auto fontWeight = fragmentFormat.
fontWeight();
760 auto fontItalic = fragmentFormat.fontItalic();
761 auto fontUnderline = fragmentFormat.fontUnderline();
762 auto fontStrikeout = fragmentFormat.fontStrikeOut();
764 auto fontForeground = fragmentFormat.foreground();
765 auto fontBackground = fragmentFormat.background();
767 const QStringList fontFamilies = fragmentFormat.fontFamilies().toStringList();
769 auto fontPointSize = fragmentFormat.font().pointSize();
770 auto anchorHref = fragmentFormat.anchorHref();
772 auto vAlign = fragmentFormat.verticalAlignment();
776 if (superscript && !(d->m_openElements.contains(
SuperScript))) {
780 if (subscript && !(d->m_openElements.contains(
SubScript))) {
784 if (!anchorHref.isEmpty() && !(d->m_openElements.contains(
Anchor))
785 && (d->m_openAnchorHref != anchorHref)) {
786 d->m_elementsToOpen.insert(
Anchor);
787 d->m_anchorHrefToOpen = anchorHref;
794 && (fontForeground != d->m_openForeground)
795 && !((d->m_openElements.contains(
797 || d->m_elementsToOpen.contains(
Anchor)))) {
799 d->m_foregroundToOpen = fontForeground;
804 && (fontBackground != d->m_openBackground)) {
806 d->m_backgroundToOpen = fontBackground;
809 if (!fontFamily.isEmpty() && !(d->m_openElements.contains(
SpanFontFamily))
810 && (fontFamily != d->m_openFontFamily)) {
812 d->m_fontFamilyToOpen = fontFamily;
818 && (fontPointSize != d->m_openFontPointSize)) {
820 d->m_fontPointSizeToOpen = fontPointSize;
828 d->m_elementsToOpen.insert(
Strong);
831 if (fontItalic && !(d->m_openElements.contains(
Emph))) {
832 d->m_elementsToOpen.insert(
Emph);
835 if (fontUnderline && !(d->m_openElements.contains(
Underline))
836 && !(d->m_openElements.contains(
Anchor)
837 || d->m_elementsToOpen.contains(
843 if (fontStrikeout && !(d->m_openElements.contains(
StrikeOut))) {
847 if (d->m_elementsToOpen.size() <= 1) {
848 return QList<int>(d->m_elementsToOpen.begin(), d->m_elementsToOpen.end());
863 while (openingOrder.
size() != 0) {
876 Q_FOREACH (
int tag, elementsToClose) {
877 if (openingOrder.
remove(tag)) {
878 sortedOpenedElements.
prepend(tag);
885 Q_FOREACH (
int tag, openingOrder) {
886 sortedOpenedElements.
prepend(tag);
891 return sortedOpenedElements;
Interface for creating marked-up text output.
virtual QTextFrame::iterator processBlock(QTextFrame::iterator it, const QTextBlock &block)
virtual void processTableCell(const QTextTableCell &tableCell, QTextTable *table)
virtual void processOpeningElements(QTextBlock::iterator it)
virtual QTextBlock::iterator processCharTextObject(QTextBlock::iterator it, const QTextFragment &fragment, QTextObject *textObject)
virtual QTextFrame::iterator processTable(QTextFrame::iterator it, QTextTable *table)
virtual void processCustomFragment(const QTextFragment &fragment, QTextDocument const *doc)
virtual void processDocument(QTextDocument *doc)
virtual QList< int > getElementsToOpen(QTextBlock::iterator it)
AbstractMarkupBuilder * m_builder
virtual QTextBlock::iterator processImage(QTextBlock::iterator it, const QTextImageFormat &imageFormat, QTextDocument *doc)
@ SpanFontPointSize
A font family altering span tag is open.
@ SpanForeground
An anchor tag is open.
@ SuperScript
No tags are open.
@ Underline
A emphasis tag is open.
@ SubScript
A superscript tag is open.
@ Emph
A strong tag is open.
@ Anchor
A subscript tag is open.
@ SpanFontFamily
A background altering span tag is open.
@ Strong
A font size altering span tag is open.
@ StrikeOut
An underline tag is open.
@ SpanBackground
A foreground altering span tag is open.
MarkupDirector(AbstractMarkupBuilder *builder)
virtual QTextFrame::iterator processFrame(QTextFrame::iterator it, QTextFrame *frame)
virtual std::pair< QTextFrame::iterator, QTextBlock > processBlockGroup(QTextFrame::iterator it, const QTextBlock &block, QTextBlockGroup *textBlockGroup)
virtual std::pair< QTextFrame::iterator, QTextBlock > processList(QTextFrame::iterator it, const QTextBlock &block, QTextList *textList)
virtual QSet< int > getElementsToClose(QTextBlock::iterator it) const
QList< int > sortOpeningOrder(QSet< int > openingTags, QTextBlock::iterator it) const
void processDocumentContents(QTextFrame::iterator begin, QTextFrame::iterator end)
virtual void processClosingElements(QTextBlock::iterator it)
virtual QTextFrame::iterator processObject(QTextFrame::iterator it, const QTextBlock &block, QTextObject *textObject)
virtual QTextFrame::iterator processBlockContents(QTextFrame::iterator it, const QTextBlock &block)
std::pair< QTextFrame::iterator, QTextBlock > skipBlockGroup(QTextFrame::iterator it, const QTextBlock &_block, QTextBlockGroup *blockGroup)
virtual QTextBlock::iterator processFragment(QTextBlock::iterator it, const QTextFragment &fragment, QTextDocument const *doc)
virtual ~MarkupDirector()
The Cutelee namespace holds all public Cutelee API.
QChar::Category category(char32_t ucs4)
void append(QList< T > &&value)
QList< T >::iterator begin()
bool contains(const AT &value) const const
QList< T >::iterator end()
void prepend(QList< T >::parameter_type value)
QSet< T >::iterator insert(QSet< T >::const_iterator it, const T &value)
bool remove(const T &value)
qsizetype size() const const
QString arg(Args &&... args) const const
const QChar at(qsizetype position) const const
QStringList split(QChar sep, Qt::SplitBehavior behavior, Qt::CaseSensitivity cs) const const
QTextFragment fragment() const const
QTextBlock::iterator begin() const const
QTextBlockFormat blockFormat() const const
const QTextDocument * document() const const
bool isValid() const const
QTextList * textList() const const
Qt::Alignment alignment() const const
int fontWeight() const const
QTextObject * objectForFormat(const QTextFormat &f) const const
QTextFrame * rootFrame() const const
BlockTrailingHorizontalRulerWidth
QTextCharFormat charFormat() const const
QString text() const const
QTextBlock currentBlock() const const
QTextFrame * currentFrame() const const
QTextFrame::iterator begin() const const
QTextFrame::iterator end() const const
qreal height() const const
QString name() const const
qreal width() const const
QTextListFormat format() const const
QTextListFormat::Style style() const const
QTextDocument * document() const const
QTextTableCell cellAt(const QTextCursor &cursor) const const
int columns() const const
QTextTableFormat format() const const
QTextFrame::iterator begin() const const
int columnSpan() const const
QTextFrame::iterator end() const const
QList< QTextLength > columnWidthConstraints() const const