-// FIXME remove with migration code
-#include <QSettings>
-
-UiStyle::UiStyle(const QString &settingsKey) : _settingsKey(settingsKey) {
- // register FormatList if that hasn't happened yet
- // FIXME I don't think this actually avoids double registration... then again... does it hurt?
- if(QVariant::nameToType("UiStyle::FormatList") == QVariant::Invalid) {
- qRegisterMetaType<FormatList>("UiStyle::FormatList");
- qRegisterMetaTypeStreamOperators<FormatList>("UiStyle::FormatList");
- Q_ASSERT(QVariant::nameToType("UiStyle::FormatList") != QVariant::Invalid);
- }
-
- // FIXME remove migration at some point
- // We remove old settings if we find them, since they conflict
-#ifdef Q_WS_MAC
- QSettings mys(QCoreApplication::organizationDomain(), Quassel::buildInfo().clientApplicationName);
-#else
- QSettings mys(QCoreApplication::organizationName(), Quassel::buildInfo().clientApplicationName);
-#endif
- mys.beginGroup("QtUi");
- if(mys.childGroups().contains("Colors")) {
- qDebug() << "Removing obsolete UiStyle settings!";
- mys.endGroup();
- mys.remove("Ui");
- mys.remove("QtUiStyle");
- mys.remove("QtUiStyleNew");
- mys.remove("QtUi/Colors");
- mys.sync();
- }
-
- _defaultFont = QFont("Monospace", QApplication::font().pointSize());
-
- // Default format
- _defaultPlainFormat.setForeground(QBrush("#000000"));
- _defaultPlainFormat.setFont(_defaultFont);
- _defaultPlainFormat.font().setFixedPitch(true);
- _defaultPlainFormat.font().setStyleHint(QFont::TypeWriter);
- setFormat(None, _defaultPlainFormat, Settings::Default);
-
- // Load saved custom formats
- UiStyleSettings s(_settingsKey);
- foreach(FormatType type, s.availableFormats()) {
- _customFormats[type] = s.customFormat(type);
- }
-
- // Now initialize the mapping between FormatCodes and FormatTypes...
- _formatCodes["%O"] = None;
- _formatCodes["%B"] = Bold;
- _formatCodes["%S"] = Italic;
- _formatCodes["%U"] = Underline;
- _formatCodes["%R"] = Reverse;
-
- _formatCodes["%D0"] = PlainMsg;
- _formatCodes["%Dn"] = NoticeMsg;
- _formatCodes["%Ds"] = ServerMsg;
- _formatCodes["%De"] = ErrorMsg;
- _formatCodes["%Dj"] = JoinMsg;
- _formatCodes["%Dp"] = PartMsg;
- _formatCodes["%Dq"] = QuitMsg;
- _formatCodes["%Dk"] = KickMsg;
- _formatCodes["%Dr"] = RenameMsg;
- _formatCodes["%Dm"] = ModeMsg;
- _formatCodes["%Da"] = ActionMsg;
-
- _formatCodes["%DT"] = Timestamp;
- _formatCodes["%DS"] = Sender;
- _formatCodes["%DN"] = Nick;
- _formatCodes["%DH"] = Hostmask;
- _formatCodes["%DC"] = ChannelName;
- _formatCodes["%DM"] = ModeFlags;
- _formatCodes["%DU"] = Url;
-
- // Initialize color codes according to mIRC "standard"
- QStringList colors;
- //colors << "white" << "black" << "navy" << "green" << "red" << "maroon" << "purple" << "orange";
- //colors << "yellow" << "lime" << "teal" << "aqua" << "royalblue" << "fuchsia" << "grey" << "silver";
- colors << "#ffffff" << "#000000" << "#000080" << "#008000" << "#ff0000" << "#800000" << "#800080" << "#ffa500";
- colors << "#ffff00" << "#00ff00" << "#008080" << "#00ffff" << "#4169E1" << "#ff00ff" << "#808080" << "#c0c0c0";
-
- // Set color formats
- for(int i = 0; i < 16; i++) {
- QString idx = QString("%1").arg(i, (int)2, (int)10, (QChar)'0');
- _formatCodes[QString("%Dcf%1").arg(idx)] = (FormatType)(FgCol00 | i<<24);
- _formatCodes[QString("%Dcb%1").arg(idx)] = (FormatType)(BgCol00 | i<<28);
- QTextCharFormat fgf, bgf;
- fgf.setForeground(QBrush(QColor(colors[i]))); setFormat((FormatType)(FgCol00 | i<<24), fgf, Settings::Default);
- bgf.setBackground(QBrush(QColor(colors[i]))); setFormat((FormatType)(BgCol00 | i<<28), bgf, Settings::Default);
- }
-
- // Set a few more standard formats
- QTextCharFormat bold; bold.setFontWeight(QFont::Bold);
- setFormat(Bold, bold, Settings::Default);
-
- QTextCharFormat italic; italic.setFontItalic(true);
- setFormat(Italic, italic, Settings::Default);
-
- QTextCharFormat underline; underline.setFontUnderline(true);
- setFormat(Underline, underline, Settings::Default);
-
- // All other formats should be defined in derived classes.
-}
-
-UiStyle::~ UiStyle() {
- qDeleteAll(_cachedFontMetrics);
-}
-
-void UiStyle::setFormat(FormatType ftype, QTextCharFormat fmt, Settings::Mode mode) {
- if(mode == Settings::Default) {
- _defaultFormats[ftype] = fmt;
- } else {
- UiStyleSettings s(_settingsKey);
- if(fmt != _defaultFormats[ftype]) {
- _customFormats[ftype] = fmt;
- s.setCustomFormat(ftype, fmt);
- } else {
- _customFormats.remove(ftype);
- s.removeCustomFormat(ftype);
- }
- }
- // TODO: invalidate only affected cached formats... if that's possible with less overhead than just rebuilding them
- _cachedFormats.clear();
- _cachedFontMetrics.clear();
-}
-
-QTextCharFormat UiStyle::format(FormatType ftype, Settings::Mode mode) const {
- if(mode == Settings::Custom && _customFormats.contains(ftype)) return _customFormats.value(ftype);
- else return _defaultFormats.value(ftype, QTextCharFormat());
-}
-
-// NOTE: This function is intimately tied to the values in FormatType. Don't change this
+QHash<QString, UiStyle::FormatType> UiStyle::_formatCodes;
+QString UiStyle::_timestampFormatString;
+
+UiStyle::UiStyle(QObject *parent)
+ : QObject(parent),
+ _channelJoinedIcon(SmallIcon("irc-channel-active")),
+ _channelPartedIcon(SmallIcon("irc-channel-inactive")),
+ _userOfflineIcon(SmallIcon("im-user-offline")),
+ _userOnlineIcon(SmallIcon("im-user")),
+ _userAwayIcon(SmallIcon("im-user-away")),
+ _categoryOpIcon(SmallIcon("irc-operator")),
+ _categoryVoiceIcon(SmallIcon("irc-voice")),
+ _opIconLimit(UserCategoryItem::categoryFromModes("o")),
+ _voiceIconLimit(UserCategoryItem::categoryFromModes("v"))
+{
+ // register FormatList if that hasn't happened yet
+ // FIXME I don't think this actually avoids double registration... then again... does it hurt?
+ if (QVariant::nameToType("UiStyle::FormatList") == QVariant::Invalid) {
+ qRegisterMetaType<FormatList>("UiStyle::FormatList");
+ qRegisterMetaTypeStreamOperators<FormatList>("UiStyle::FormatList");
+ Q_ASSERT(QVariant::nameToType("UiStyle::FormatList") != QVariant::Invalid);
+ }
+
+ _uiStylePalette = QVector<QBrush>(NumRoles, QBrush());
+
+ // Now initialize the mapping between FormatCodes and FormatTypes...
+ _formatCodes["%O"] = Base;
+ _formatCodes["%B"] = Bold;
+ _formatCodes["%S"] = Italic;
+ _formatCodes["%U"] = Underline;
+ _formatCodes["%R"] = Reverse;
+
+ _formatCodes["%DN"] = Nick;
+ _formatCodes["%DH"] = Hostmask;
+ _formatCodes["%DC"] = ChannelName;
+ _formatCodes["%DM"] = ModeFlags;
+ _formatCodes["%DU"] = Url;
+
+ setTimestampFormatString("[hh:mm:ss]");
+
+ // BufferView / NickView settings
+ UiStyleSettings s;
+ _showBufferViewIcons = _showNickViewIcons = s.value("ShowItemViewIcons", true).toBool();
+ s.notify("ShowItemViewIcons", this, SLOT(showItemViewIconsChanged(QVariant)));
+
+ _allowMircColors = s.value("AllowMircColors", true).toBool();
+ s.notify("AllowMircColors", this, SLOT(allowMircColorsChanged(QVariant)));
+
+ loadStyleSheet();
+}
+
+
+UiStyle::~UiStyle()
+{
+ qDeleteAll(_metricsCache);
+}
+
+
+void UiStyle::reload()
+{
+ loadStyleSheet();
+}
+
+
+void UiStyle::loadStyleSheet()
+{
+ qDeleteAll(_metricsCache);
+ _metricsCache.clear();
+ _formatCache.clear();
+ _formats.clear();
+
+ UiStyleSettings s;
+
+ QString styleSheet;
+ styleSheet += loadStyleSheet("file:///" + Quassel::findDataFilePath("stylesheets/default.qss"));
+ styleSheet += loadStyleSheet("file:///" + Quassel::configDirPath() + "settings.qss");
+ if (s.value("UseCustomStyleSheet", false).toBool())
+ styleSheet += loadStyleSheet("file:///" + s.value("CustomStyleSheetPath").toString(), true);
+ styleSheet += loadStyleSheet("file:///" + Quassel::optionValue("qss"), true);
+
+ if (!styleSheet.isEmpty()) {
+ QssParser parser;
+ parser.processStyleSheet(styleSheet);
+ QApplication::setPalette(parser.palette());
+
+ _uiStylePalette = parser.uiStylePalette();
+ _formats = parser.formats();
+ _listItemFormats = parser.listItemFormats();
+
+ styleSheet = styleSheet.trimmed();
+ if (!styleSheet.isEmpty())
+ qApp->setStyleSheet(styleSheet); // pass the remaining sections to the application
+ }
+
+ emit changed();
+}
+
+
+QString UiStyle::loadStyleSheet(const QString &styleSheet, bool shouldExist)
+{
+ QString ss = styleSheet;
+ if (ss.startsWith("file:///")) {
+ ss.remove(0, 8);
+ if (ss.isEmpty())
+ return QString();
+
+ QFile file(ss);
+ if (file.open(QFile::ReadOnly)) {
+ QTextStream stream(&file);
+ ss = stream.readAll();
+ file.close();
+ }
+ else {
+ if (shouldExist)
+ qWarning() << "Could not open stylesheet file:" << file.fileName();
+ return QString();
+ }
+ }
+ return ss;
+}
+
+
+void UiStyle::setTimestampFormatString(const QString &format)
+{
+ if (_timestampFormatString != format) {
+ _timestampFormatString = format;
+ // FIXME reload
+ }
+}
+
+
+void UiStyle::allowMircColorsChanged(const QVariant &v)
+{
+ _allowMircColors = v.toBool();
+ emit changed();
+}
+
+
+/******** ItemView Styling *******/
+
+void UiStyle::showItemViewIconsChanged(const QVariant &v)
+{
+ _showBufferViewIcons = _showNickViewIcons = v.toBool();
+}
+
+
+QVariant UiStyle::bufferViewItemData(const QModelIndex &index, int role) const
+{
+ BufferInfo::Type type = (BufferInfo::Type)index.data(NetworkModel::BufferTypeRole).toInt();
+ bool isActive = index.data(NetworkModel::ItemActiveRole).toBool();
+
+ if (role == Qt::DecorationRole) {
+ if (!_showBufferViewIcons)
+ return QVariant();
+
+ switch (type) {
+ case BufferInfo::ChannelBuffer:
+ if (isActive)
+ return _channelJoinedIcon;
+ else
+ return _channelPartedIcon;
+ case BufferInfo::QueryBuffer:
+ if (!isActive)
+ return _userOfflineIcon;
+ if (index.data(NetworkModel::UserAwayRole).toBool())
+ return _userAwayIcon;
+ else
+ return _userOnlineIcon;
+ default:
+ return QVariant();
+ }
+ }
+
+ quint32 fmtType = BufferViewItem;
+ switch (type) {
+ case BufferInfo::StatusBuffer:
+ fmtType |= NetworkItem;
+ break;
+ case BufferInfo::ChannelBuffer:
+ fmtType |= ChannelBufferItem;
+ break;
+ case BufferInfo::QueryBuffer:
+ fmtType |= QueryBufferItem;
+ break;
+ default:
+ return QVariant();
+ }
+
+ QTextCharFormat fmt = _listItemFormats.value(BufferViewItem);
+ fmt.merge(_listItemFormats.value(fmtType));
+
+ BufferInfo::ActivityLevel activity = (BufferInfo::ActivityLevel)index.data(NetworkModel::BufferActivityRole).toInt();
+ if (activity & BufferInfo::Highlight) {
+ fmt.merge(_listItemFormats.value(BufferViewItem | HighlightedBuffer));
+ fmt.merge(_listItemFormats.value(fmtType | HighlightedBuffer));
+ }
+ else if (activity & BufferInfo::NewMessage) {
+ fmt.merge(_listItemFormats.value(BufferViewItem | UnreadBuffer));
+ fmt.merge(_listItemFormats.value(fmtType | UnreadBuffer));
+ }
+ else if (activity & BufferInfo::OtherActivity) {
+ fmt.merge(_listItemFormats.value(BufferViewItem | ActiveBuffer));
+ fmt.merge(_listItemFormats.value(fmtType | ActiveBuffer));
+ }
+ else if (!isActive) {
+ fmt.merge(_listItemFormats.value(BufferViewItem | InactiveBuffer));
+ fmt.merge(_listItemFormats.value(fmtType | InactiveBuffer));
+ }
+ else if (index.data(NetworkModel::UserAwayRole).toBool()) {
+ fmt.merge(_listItemFormats.value(BufferViewItem | UserAway));
+ fmt.merge(_listItemFormats.value(fmtType | UserAway));
+ }
+
+ return itemData(role, fmt);
+}
+
+
+QVariant UiStyle::nickViewItemData(const QModelIndex &index, int role) const
+{
+ NetworkModel::ItemType type = (NetworkModel::ItemType)index.data(NetworkModel::ItemTypeRole).toInt();
+
+ if (role == Qt::DecorationRole) {
+ if (!_showNickViewIcons)
+ return QVariant();
+
+ switch (type) {
+ case NetworkModel::UserCategoryItemType:
+ {
+ int categoryId = index.data(TreeModel::SortRole).toInt();
+ if (categoryId <= _opIconLimit)
+ return _categoryOpIcon;
+ if (categoryId <= _voiceIconLimit)
+ return _categoryVoiceIcon;
+ return _userOnlineIcon;
+ }
+ case NetworkModel::IrcUserItemType:
+ if (index.data(NetworkModel::ItemActiveRole).toBool())
+ return _userOnlineIcon;
+ else
+ return _userAwayIcon;
+ default:
+ return QVariant();
+ }
+ }
+
+ QTextCharFormat fmt = _listItemFormats.value(NickViewItem);
+
+ switch (type) {
+ case NetworkModel::IrcUserItemType:
+ fmt.merge(_listItemFormats.value(NickViewItem | IrcUserItem));
+ if (!index.data(NetworkModel::ItemActiveRole).toBool()) {
+ fmt.merge(_listItemFormats.value(NickViewItem | UserAway));
+ fmt.merge(_listItemFormats.value(NickViewItem | IrcUserItem | UserAway));
+ }
+ break;
+ case NetworkModel::UserCategoryItemType:
+ fmt.merge(_listItemFormats.value(NickViewItem | UserCategoryItem));
+ break;
+ default:
+ return QVariant();
+ }
+
+ return itemData(role, fmt);
+}
+
+
+QVariant UiStyle::itemData(int role, const QTextCharFormat &format) const
+{
+ switch (role) {
+ case Qt::FontRole:
+ return format.font();
+ case Qt::ForegroundRole:
+ return format.property(QTextFormat::ForegroundBrush);
+ case Qt::BackgroundRole:
+ return format.property(QTextFormat::BackgroundBrush);
+ default:
+ return QVariant();
+ }
+}
+
+
+/******** Caching *******/
+
+QTextCharFormat UiStyle::format(quint64 key) const
+{
+ return _formats.value(key, QTextCharFormat());
+}
+
+
+QTextCharFormat UiStyle::cachedFormat(quint32 formatType, quint32 messageLabel) const
+{
+ return _formatCache.value(formatType | ((quint64)messageLabel << 32), QTextCharFormat());
+}
+
+
+void UiStyle::setCachedFormat(const QTextCharFormat &format, quint32 formatType, quint32 messageLabel) const
+{
+ _formatCache[formatType | ((quint64)messageLabel << 32)] = format;
+}
+
+
+QFontMetricsF *UiStyle::fontMetrics(quint32 ftype, quint32 label) const
+{
+ // QFontMetricsF is not assignable, so we need to store pointers :/
+ quint64 key = ftype | ((quint64)label << 32);
+
+ if (_metricsCache.contains(key))
+ return _metricsCache.value(key);
+
+ return (_metricsCache[key] = new QFontMetricsF(format(ftype, label).font()));
+}
+
+
+/******** Generate formats ********/
+
+// NOTE: This and the following functions are intimately tied to the values in FormatType. Don't change this