#include "documentstyle.hpp" #include "kristall.hpp" #include #include #include #include #include #include #include #include #include static QString encodeCssFont (const QFont& refFont) { //----------------------------------------------------------------------- // This function assembles a CSS Font specification string from // a QFont. This supports most of the QFont attributes settable in // the Qt 4.8 and Qt 5.3 QFontDialog. // // (1) Font Family // (2) Font Weight (just bold or not) // (3) Font Style (possibly Italic or Oblique) // (4) Font Size (in either pixels or points) // (5) Decorations (possibly Underline or Strikeout) // // Not supported: Writing System (e.g. Latin). // // See the corresponding decode function, below. // QFont decodeCssFontString (const QString cssFontStr) //----------------------------------------------------------------------- QStringList fields; // CSS font attribute fields // *************************************************** // *** (1) Font Family: Primary plus Substitutes *** // *************************************************** const QString family = refFont.family(); // NOTE [9-2014, Qt 4.8.6]: This isn't what I thought it was. It // does not return a list of "fallback" font faces (e.g. Georgia, // Serif for "Times New Roman"). In my testing, this is always // returning an empty list. // QStringList famSubs = QFont::substitutes (family); if (!famSubs.contains (family)) famSubs.prepend (family); static const QChar DBL_QUOT ('"'); const int famCnt = famSubs.count(); QStringList famList; for (int inx = 0; inx < famCnt; ++inx) { // Place double quotes around family names having space characters, // but only if double quotes are not already there. // const QString fam = famSubs [inx]; if (fam.contains (' ') && !fam.startsWith (DBL_QUOT)) famList << (DBL_QUOT + fam + DBL_QUOT); else famList << fam; } const QString famStr = QString ("font-family: ") + famList.join (", "); fields << famStr; // ************************************** // *** (2) Font Weight: Bold or Not *** // ************************************** const bool bold = refFont.bold(); if (bold) fields << "font-weight: bold"; // **************************************************** // *** (3) Font Style: possibly Italic or Oblique *** // **************************************************** const QFont::Style style = refFont.style(); switch (style) { case QFont::StyleNormal: break; case QFont::StyleItalic: fields << "font-style: italic"; break; case QFont::StyleOblique: fields << "font-style: oblique"; break; } // ************************************************ // *** (4) Font Size: either Pixels or Points *** // ************************************************ const double sizeInPoints = refFont.pointSizeF(); // <= 0 if not defined. const int sizeInPixels = refFont.pixelSize(); // <= 0 if not defined. if (sizeInPoints > 0.0) fields << QString ("font-size: %1pt") .arg (sizeInPoints); else if (sizeInPixels > 0) fields << QString ("font-size: %1px") .arg (sizeInPixels); // *********************************************** // *** (5) Decorations: Underline, Strikeout *** // *********************************************** const bool underline = refFont.underline(); const bool strikeOut = refFont.strikeOut(); if (underline && strikeOut) fields << "text-decoration: underline line-through"; else if (underline) fields << "text-decoration: underline"; else if (strikeOut) fields << "text-decoration: line-through"; const QString cssFontStr = fields.join ("; "); return cssFontStr; } DocumentStyle::DocumentStyle(bool do_init) : theme(Fixed), standard_font(), h1_font(), h2_font(), h3_font(), preformatted_font(), background_color(0xed, 0xef, 0xff), standard_color(0x00, 0x00, 0x00), preformatted_color(0x00, 0x00, 0x00), h1_color(0x02, 0x2f, 0x90), h2_color(0x02, 0x2f, 0x90), h3_color(0x02, 0x2f, 0x90), blockquote_color(0xFF, 0xFF, 0xFF), internal_link_color(0x0e, 0x8f, 0xff), external_link_color(0x0e, 0x8f, 0xff), cross_scheme_link_color(0x09, 0x60, 0xa7), internal_link_prefix("→ "), external_link_prefix("⇒ "), margin(55.0) { if (do_init) this->initialiseDefaultFonts(); } void DocumentStyle::initialiseDefaultFonts() { // Setup default fonts #ifdef Q_OS_WIN32 // Windows static const QString FONT_NORMAL = "Segoe UI"; static const QString FONT_MONO = "Consolas"; //#elif defined Q_OS_DARWIN // Mac (No idea what they use) // static const QString FONT_NORMAL = "???"; // static const QString FONT_MONO = "???"; #else // Ganoo slash linooks static const QString FONT_NORMAL = kristall::default_font_family; static const QString FONT_MONO = kristall::default_font_family_fixed; #endif preformatted_font.setFamily(FONT_MONO); preformatted_font.setPointSizeF(10.0); standard_font.setFamily(FONT_NORMAL); standard_font.setPointSizeF(10.0); h1_font.setFamily(FONT_NORMAL); h1_font.setBold(true); h1_font.setPointSizeF(20.0); h2_font.setFamily(FONT_NORMAL); h2_font.setBold(true); h2_font.setPointSizeF(15.0); h3_font.setFamily(FONT_NORMAL); h3_font.setBold(true); h3_font.setPointSizeF(12.0); this->cookie = []() { QByteArray arr(8, ' '); for(auto & b : arr) b = rand(); return arr.toBase64(); }(); } QString DocumentStyle::createFileNameFromName(const QString &src, int index) { QString result; result.reserve(src.size() + 5); for(int i = 0; i < src.size(); i++) { QChar c = src.at(i); if(c.isLetterOrNumber()) { result.append(c.toLower()); } else if(c.isSpace()) { result.append('-'); } else { result.append(QString::number(c.unicode())); } } if(index > 0) { result.append(QString("-%1").arg(index)); } result.append(".kthm"); return result; } bool DocumentStyle::save(QSettings &settings) const { settings.setValue("version", 1); settings.setValue("theme", int(theme)); settings.setValue("background_color", background_color.name()); settings.setValue("blockquote_color", blockquote_color.name()); settings.setValue("margins", margin); { settings.beginGroup("Standard"); settings.setValue("font", standard_font.toString()); settings.setValue("color", standard_color.name()); settings.endGroup(); } { settings.beginGroup("Preformatted"); settings.setValue("font", preformatted_font.toString()); settings.setValue("color", preformatted_color.name()); settings.endGroup(); } { settings.beginGroup("H1"); settings.setValue("font", h1_font.toString()); settings.setValue("color", h1_color.name()); settings.endGroup(); } { settings.beginGroup("H2"); settings.setValue("font", h2_font.toString()); settings.setValue("color", h2_color.name()); settings.endGroup(); } { settings.beginGroup("H3"); settings.setValue("font", h3_font.toString()); settings.setValue("color", h3_color.name()); settings.endGroup(); } { settings.beginGroup("Link"); settings.setValue("color_internal", internal_link_color.name()); settings.setValue("color_external", external_link_color.name()); settings.setValue("color_cross_scheme", cross_scheme_link_color.name()); settings.setValue("internal_prefix", internal_link_prefix); settings.setValue("external_prefix", external_link_prefix); settings.endGroup(); } return true; } bool DocumentStyle::load(QSettings &settings) { switch(settings.value("version", 0).toInt()) { case 0: { if(settings.contains("standard_color")) { standard_font.fromString(settings.value("standard_font").toString()); h1_font.fromString(settings.value("h1_font").toString()); h2_font.fromString(settings.value("h2_font").toString()); h3_font.fromString(settings.value("h3_font").toString()); preformatted_font.fromString(settings.value("preformatted_font").toString()); background_color = QColor(settings.value("background_color").toString()); standard_color = QColor(settings.value("standard_color").toString()); preformatted_color = QColor(settings.value("preformatted_color").toString()); blockquote_color = QColor(settings.value("blockquote_color").toString()); h1_color = QColor(settings.value("h1_color").toString()); h2_color = QColor(settings.value("h2_color").toString()); h3_color = QColor(settings.value("h3_color").toString()); internal_link_color = QColor(settings.value("internal_link_color").toString()); external_link_color = QColor(settings.value("external_link_color").toString()); cross_scheme_link_color = QColor(settings.value("cross_scheme_link_color").toString()); internal_link_prefix = settings.value("internal_link_prefix").toString(); external_link_prefix = settings.value("external_link_prefix").toString(); margin = settings.value("margins").toDouble(); theme = Theme(settings.value("theme").toInt()); } break; } case 1: { theme = Theme(settings.value("theme", int(theme)).toInt()); background_color = QColor { settings.value("background_color", background_color.name()).toString() }; blockquote_color = QColor { settings.value("blockquote_color", blockquote_color.name()).toString() }; margin = settings.value("margins", 55).toInt(); { settings.beginGroup("Standard"); standard_font.fromString(settings.value("font", standard_font.toString()).toString()); standard_color = QString { settings.value("color", standard_color.name()).toString() }; settings.endGroup(); } { settings.beginGroup("Preformatted"); preformatted_font.fromString(settings.value("font", preformatted_font.toString()).toString()); preformatted_color = QString { settings.value("color", preformatted_color.name()).toString() }; settings.endGroup(); } { settings.beginGroup("H1"); h1_font.fromString(settings.value("font", h1_font.toString()).toString()); h1_color = QString { settings.value("color", h1_color.name()).toString() }; settings.endGroup(); } { settings.beginGroup("H2"); h2_font.fromString(settings.value("font", h2_font.toString()).toString()); h2_color = QString { settings.value("color", h2_color.name()).toString() }; settings.endGroup(); } { settings.beginGroup("H3"); h3_font.fromString(settings.value("font", h3_font.toString()).toString()); h3_color = QString { settings.value("color", h3_color.name()).toString() }; settings.endGroup(); } { settings.beginGroup("Link"); internal_link_color = QString { settings.value("color_internal", internal_link_color.name()).toString() }; external_link_color = QString {settings.value("color_external", external_link_color.name()).toString() }; cross_scheme_link_color = QString {settings.value("color_cross_scheme", cross_scheme_link_color.name()).toString() }; internal_link_prefix = settings.value("internal_prefix", internal_link_prefix).toString(); external_link_prefix = settings.value("external_prefix", external_link_prefix).toString(); settings.endGroup(); } } break; default: return false; } return true; } DocumentStyle DocumentStyle::derive(const QUrl &url) const { DocumentStyle themed = *this; // Patch font lists to allow improved emoji display: // Now this is a bit tricky to get right: // 1. We need a list of fonts that provide emojis. This is `emojiFonts` // 2. We need our own unique font name (This is "Kristall XX" + cookie) // 3. We need to substitutions for that unique random name so Qt will look up missing symbols // 4. We MUST NOT use a system font name as we would replace the user interface font (which we don't want to touch) static QStringList emojiFonts = { "", "Apple Color Emoji", "Segoe UI Emoji", "Twitter Color Emoji", "Noto Color Emoji", "JoyPixels", }; auto const patchup_font = [this](QFont & font, QString custom_family) { emojiFonts.front() = font.family(); QFont::insertSubstitutions(custom_family + cookie, emojiFonts); font.setFamily(custom_family + cookie); // from docs: // > After substituting a font, you must trigger the updating of the font by destroying and re-creating all QFont objects. font.fromString(font.toString()); }; patchup_font(themed.h1_font, "Kristall H1"); patchup_font(themed.h2_font, "Kristall H2"); patchup_font(themed.h3_font, "Kristall H3"); patchup_font(themed.standard_font, "Kristall Standard"); patchup_font(themed.preformatted_font, "Kristall Monospace"); if (this->theme == Fixed) return themed; QByteArray hash = QCryptographicHash::hash(url.host().toUtf8(), QCryptographicHash::Md5); std::array items; assert(items.size() == hash.size()); memcpy(items.data(), hash.data(), items.size()); float hue = (items[0] + items[1]) / 510.0; float saturation = items[2] / 255.0; double tmp; switch (this->theme) { case AutoDarkTheme: { themed.background_color = QColor::fromHslF(hue, saturation, 0.25f); themed.standard_color = QColor{0xFF, 0xFF, 0xFF}; themed.h1_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.75); themed.h2_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.75); themed.h3_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.75); themed.external_link_color = QColor::fromHslF(std::modf(hue + 0.25, &tmp), 1.0, 0.75); themed.internal_link_color = themed.external_link_color.lighter(110); themed.cross_scheme_link_color = themed.external_link_color.darker(110); themed.blockquote_color = themed.background_color.lighter(130); break; } case AutoLightTheme: { themed.background_color = QColor::fromHslF(hue, items[2] / 255.0, 0.85); themed.standard_color = QColor{0x00, 0x00, 0x00}; themed.h1_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.25); themed.h2_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.25); themed.h3_color = QColor::fromHslF(std::modf(hue + 0.5, &tmp), 1.0 - saturation, 0.25); themed.external_link_color = QColor::fromHslF(std::modf(hue + 0.25, &tmp), 1.0, 0.25); themed.internal_link_color = themed.external_link_color.darker(110); themed.cross_scheme_link_color = themed.external_link_color.lighter(110); themed.blockquote_color = themed.background_color.darker(120); break; } case Fixed: assert(false); } // Same for all themes themed.preformatted_color = themed.standard_color; return themed; } QString DocumentStyle::toStyleSheet() const { QString css; css += QString("p { color: %2; %1 }\n").arg(encodeCssFont (standard_font), standard_color.name()); css += QString("a { color: %2; %1 }\n").arg(encodeCssFont (standard_font), external_link_color.name()); css += QString("pre { color: %2; %1 }\n").arg(encodeCssFont (preformatted_font), preformatted_color.name()); css += QString("h1 { color: %2; %1 }\n").arg(encodeCssFont (h1_font), h1_color.name()); css += QString("h2 { color: %2; %1 }\n").arg(encodeCssFont (h2_font), h2_color.name()); css += QString("h3 { color: %2; %1 }\n").arg(encodeCssFont (h3_font), h3_color.name()); css += QString("blockquote { background: %1 }\n").arg(blockquote_color.name()); // qDebug() << "CSS → " << css; return css; }