-#include "uistylesettings.h"
-
-UiStyle::UiStyle(const QString &settingsKey) : _settingsKey(settingsKey) {
- // Default format
- QTextCharFormat def;
- def.setForeground(QBrush("#000000"));
- def.setFont(QFont("Mono",12));
- _defaultFormats = QVector<QTextCharFormat>(NumFormatTypes, def);
- _customFormats = QVector<QTextCharFormat>(NumFormatTypes, QTextFormat().toCharFormat());
-
- // Load saved custom formats
- UiStyleSettings s(_settingsKey);
- foreach(FormatType type, s.availableFormats()) {
- _customFormats[type] = s.customFormat(type);
- }
-
- // 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";
-
- // 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;
-
- // 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);
- _formatCodes[QString("%Dcb%1").arg(idx)] = (FormatType)(BgCol00 + i);
- QTextCharFormat fgf, bgf;
- fgf.setForeground(QBrush(QColor(colors[i]))); setFormat((FormatType)(FgCol00 + i), fgf, Settings::Default);
- bgf.setBackground(QBrush(QColor(colors[i]))); setFormat((FormatType)(BgCol00 + i), 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() {
-
-}
-
-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[ftype] = QTextFormat().toCharFormat();
- s.removeCustomFormat(ftype);
- }
- }
-}
-
-QTextCharFormat UiStyle::format(FormatType ftype, Settings::Mode mode) const {
- if(mode == Settings::Custom && _customFormats[ftype].isValid()) return _customFormats[ftype];
- else return _defaultFormats[ftype];
-}
-
-UiStyle::FormatType UiStyle::formatType(const QString & code) const {
- if(_formatCodes.contains(code)) return _formatCodes.value(code);
- return Invalid;
-}
-
-QString UiStyle::formatCode(FormatType ftype) const {
- return _formatCodes.key(ftype);
-}
-
-UiStyle::StyledText UiStyle::styleString(QString s) {
- StyledText result;
- QList<FormatType> fmtList;
- fmtList.append(None);
- QTextLayout::FormatRange curFmtRng;
- curFmtRng.format = format(None);
- curFmtRng.start = 0;
- result.formats.append(curFmtRng);
- int pos = 0; int length;
- int fgCol = -1, bgCol = -1; // marks current mIRC color
- for(;;) {
- pos = s.indexOf('%', pos);
- if(pos < 0) break;
- if(s[pos+1] == '%') { // escaped %, just remove one and continue
- s.remove(pos, 1);
- pos++;
- continue;
- } else if(s[pos+1] == 'D' && s[pos+2] == 'c') { // color code
- if(s[pos+3] == '-') { // color off
- if(fgCol >= 0) {
- fmtList.removeAll((FormatType)(FgCol00 + fgCol));
- fgCol = -1;
- }
- if(bgCol >= 0) {
- fmtList.removeAll((FormatType)(BgCol00 + bgCol));
- bgCol = -1;
- }
- curFmtRng.format = mergedFormat(fmtList);
- length = 4;
- } else {
- int color = 10 * s[pos+4].digitValue() + s[pos+5].digitValue();
- //TODO: use 99 as transparent color (re mirc color "standard")
- color &= 0x0f;
- int *colptr; FormatType coltype;
- if(s[pos+3] == 'f') { // foreground
- colptr = &fgCol; coltype = FgCol00;
- } else { // background
- Q_ASSERT(s[pos+3] == 'b');
- colptr = &bgCol; coltype = BgCol00;
- }
- if(*colptr >= 0) {
- // color already set, remove format code and add new one
- Q_ASSERT(fmtList.contains((FormatType)(coltype + *colptr)));
- fmtList.removeAll((FormatType)(coltype + *colptr));
- fmtList.append((FormatType)(coltype + color));
- curFmtRng.format = mergedFormat(fmtList);
+#include "uisettings.h"
+#include "util.h"
+
+QHash<QString, UiStyle::FormatType> UiStyle::_formatCodes;
+QString UiStyle::_timestampFormatString;
+
+UiStyle::UiStyle(QObject *parent)
+ : QObject(parent),
+ _channelJoinedIcon(QIcon::fromTheme("irc-channel-joined", QIcon(":/icons/irc-channel-joined.png"))),
+ _channelPartedIcon(QIcon::fromTheme("irc-channel-parted", QIcon(":/icons/irc-channel-parted.png"))),
+ _userOfflineIcon(QIcon::fromTheme("im-user-offline", QIcon::fromTheme("user-offline", QIcon(":/icons/im-user-offline.png")))),
+ _userOnlineIcon(QIcon::fromTheme("im-user", QIcon::fromTheme("user-available", QIcon(":/icons/im-user.png")))), // im-user-* are non-standard oxygen extensions
+ _userAwayIcon(QIcon::fromTheme("im-user-away", QIcon::fromTheme("user-away", QIcon(":/icons/im-user-away.png")))),
+ _categoryOpIcon(QIcon::fromTheme("irc-operator")),
+ _categoryVoiceIcon(QIcon::fromTheme("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()) {
+ QString customSheetPath(s.value("CustomStyleSheetPath").toString());
+ QString customSheet = loadStyleSheet("file:///" + customSheetPath, true);
+ if (customSheet.isEmpty()) {
+ // MIGRATION: changed default install path for data from /usr/share/apps to /usr/share
+ if (customSheetPath.startsWith("/usr/share/apps/quassel")) {
+ customSheetPath.replace(QRegExp("^/usr/share/apps"), "/usr/share");
+ customSheet = loadStyleSheet("file:///" + customSheetPath, true);
+ if (!customSheet.isEmpty()) {
+ s.setValue("CustomStyleSheetPath", customSheetPath);
+ qDebug() << "Custom stylesheet path migrated to" << customSheetPath;
+ }
+ }
+ }
+ styleSheet += customSheet;
+ }
+ 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
+// until you _really_ know what you do!
+QTextCharFormat UiStyle::format(quint32 ftype, quint32 label_) const
+{
+ if (ftype == Invalid)
+ return QTextCharFormat();
+
+ quint64 label = (quint64)label_ << 32;
+
+ // check if we have exactly this format readily cached already
+ QTextCharFormat fmt = cachedFormat(ftype, label_);
+ if (fmt.properties().count())
+ return fmt;
+
+ mergeFormat(fmt, ftype, label & Q_UINT64_C(0xffff000000000000));
+
+ for (quint64 mask = Q_UINT64_C(0x0000000100000000); mask <= (quint64)Selected << 32; mask <<= 1) {
+ if (label & mask)
+ mergeFormat(fmt, ftype, mask | Q_UINT64_C(0xffff000000000000));
+ }
+
+ setCachedFormat(fmt, ftype, label_);
+ return fmt;
+}
+
+
+void UiStyle::mergeFormat(QTextCharFormat &fmt, quint32 ftype, quint64 label) const
+{
+ mergeSubElementFormat(fmt, ftype & 0x00ff, label);
+
+ // TODO: allow combinations for mirc formats and colors (each), e.g. setting a special format for "bold and italic"
+ // or "foreground 01 and background 03"
+ if ((ftype & 0xfff00)) { // element format
+ for (quint32 mask = 0x00100; mask <= 0x40000; mask <<= 1) {
+ if (ftype & mask) {
+ mergeSubElementFormat(fmt, ftype & (mask | 0xff), label);
+ }
+ }
+ }
+
+ // Now we handle color codes
+ // We assume that those can't be combined with subelement and message types.
+ if (_allowMircColors) {
+ if (ftype & 0x00400000)
+ mergeSubElementFormat(fmt, ftype & 0x0f400000, label); // foreground
+ if (ftype & 0x00800000)
+ mergeSubElementFormat(fmt, ftype & 0xf0800000, label); // background
+ if ((ftype & 0x00c00000) == 0x00c00000)
+ mergeSubElementFormat(fmt, ftype & 0xffc00000, label); // combination
+ }
+
+ // URL
+ if (ftype & Url)
+ mergeSubElementFormat(fmt, ftype & (Url | 0x000000ff), label);
+}
+
+
+// Merge a subelement format into an existing message format
+void UiStyle::mergeSubElementFormat(QTextCharFormat &fmt, quint32 ftype, quint64 label) const
+{
+ quint64 key = ftype | label;
+ fmt.merge(format(key & Q_UINT64_C(0x0000ffffffffff00))); // label + subelement
+ fmt.merge(format(key & Q_UINT64_C(0x0000ffffffffffff))); // label + subelement + msgtype
+ fmt.merge(format(key & Q_UINT64_C(0xffffffffffffff00))); // label + subelement + nickhash
+ fmt.merge(format(key & Q_UINT64_C(0xffffffffffffffff))); // label + subelement + nickhash + msgtype
+}
+
+
+UiStyle::FormatType UiStyle::formatType(Message::Type msgType)
+{
+ switch (msgType) {
+ case Message::Plain:
+ return PlainMsg;
+ case Message::Notice:
+ return NoticeMsg;
+ case Message::Action:
+ return ActionMsg;
+ case Message::Nick:
+ return NickMsg;
+ case Message::Mode:
+ return ModeMsg;
+ case Message::Join:
+ return JoinMsg;
+ case Message::Part:
+ return PartMsg;
+ case Message::Quit:
+ return QuitMsg;
+ case Message::Kick:
+ return KickMsg;
+ case Message::Kill:
+ return KillMsg;
+ case Message::Server:
+ return ServerMsg;
+ case Message::Info:
+ return InfoMsg;
+ case Message::Error:
+ return ErrorMsg;
+ case Message::DayChange:
+ return DayChangeMsg;
+ case Message::Topic:
+ return TopicMsg;
+ case Message::NetsplitJoin:
+ return NetsplitJoinMsg;
+ case Message::NetsplitQuit:
+ return NetsplitQuitMsg;
+ case Message::Invite:
+ return InviteMsg;
+ }
+ //Q_ASSERT(false); // we need to handle all message types
+ qWarning() << Q_FUNC_INFO << "Unknown message type:" << msgType;
+ return ErrorMsg;
+}
+
+
+UiStyle::FormatType UiStyle::formatType(const QString &code)
+{
+ if (_formatCodes.contains(code)) return _formatCodes.value(code);
+ return Invalid;
+}
+
+
+QString UiStyle::formatCode(FormatType ftype)
+{
+ return _formatCodes.key(ftype);
+}
+
+
+QList<QTextLayout::FormatRange> UiStyle::toTextLayoutList(const FormatList &formatList, int textLength, quint32 messageLabel) const
+{
+ QList<QTextLayout::FormatRange> formatRanges;
+ QTextLayout::FormatRange range;
+ int i = 0;
+ for (i = 0; i < formatList.count(); i++) {
+ range.format = format(formatList.at(i).second, messageLabel);
+ range.start = formatList.at(i).first;
+ if (i > 0) formatRanges.last().length = range.start - formatRanges.last().start;
+ formatRanges.append(range);
+ }
+ if (i > 0) formatRanges.last().length = textLength - formatRanges.last().start;
+ return formatRanges;
+}
+
+
+// This method expects a well-formatted string, there is no error checking!
+// Since we create those ourselves, we should be pretty safe that nobody does something crappy here.
+UiStyle::StyledString UiStyle::styleString(const QString &s_, quint32 baseFormat)
+{
+ QString s = s_;
+ StyledString result;
+ result.formatList.append(qMakePair((quint16)0, baseFormat));
+
+ if (s.length() > 65535) {
+ // We use quint16 for indexes
+ qWarning() << QString("String too long to be styled: %1").arg(s);
+ result.plainText = s;
+ return result;
+ }
+
+ quint32 curfmt = baseFormat;
+ int pos = 0; quint16 length = 0;
+ for (;;) {
+ pos = s.indexOf('%', pos);
+ if (pos < 0) break;
+ if (s[pos+1] == '%') { // escaped %, we just remove one and continue
+ s.remove(pos, 1);
+ pos++;
+ continue;
+ }
+ if (s[pos+1] == 'D' && s[pos+2] == 'c') { // color code
+ if (s[pos+3] == '-') { // color off
+ curfmt &= 0x003fffff;
+ length = 4;
+ }
+ else {
+ int color = 10 * s[pos+4].digitValue() + s[pos+5].digitValue();
+ //TODO: use 99 as transparent color (re mirc color "standard")
+ color &= 0x0f;
+ if (s[pos+3] == 'f') {
+ curfmt &= 0xf0ffffff;
+ curfmt |= (quint32)(color << 24) | 0x00400000;
+ }
+ else {
+ curfmt &= 0x0fffffff;
+ curfmt |= (quint32)(color << 28) | 0x00800000;
+ }
+ length = 6;
+ }
+ }
+ else if (s[pos+1] == 'O') { // reset formatting
+ curfmt &= 0x000000ff; // we keep message type-specific formatting
+ length = 2;
+ }
+ else if (s[pos+1] == 'R') { // reverse
+ // TODO: implement reverse formatting
+
+ length = 2;
+ }
+ else { // all others are toggles
+ QString code = QString("%") + s[pos+1];
+ if (s[pos+1] == 'D') code += s[pos+2];
+ FormatType ftype = formatType(code);
+ if (ftype == Invalid) {
+ pos++;
+ qWarning() << (QString("Invalid format code in string: %1").arg(s));
+ continue;
+ }
+ curfmt ^= ftype;
+ length = code.length();
+ }
+ s.remove(pos, length);
+ if (pos == result.formatList.last().first)
+ result.formatList.last().second = curfmt;
+ else
+ result.formatList.append(qMakePair((quint16)pos, curfmt));
+ }
+ result.plainText = s;
+ return result;
+}
+
+
+QString UiStyle::mircToInternal(const QString &mirc_)
+{
+ QString mirc;
+ mirc.reserve(mirc_.size());
+ foreach (const QChar &c, mirc_) {
+ if ((c < '\x20' || c == '\x7f') && c != '\x03') {
+ switch (c.unicode()) {
+ case '\x02':
+ mirc += "%B";
+ break;
+ case '\x0f':
+ mirc += "%O";
+ break;
+ case '\x09':
+ mirc += " ";
+ break;
+ case '\x12':
+ case '\x16':
+ mirc += "%R";
+ break;
+ case '\x1d':
+ mirc += "%S";
+ break;
+ case '\x1f':
+ mirc += "%U";
+ break;
+ case '\x7f':
+ mirc += QChar(0x2421);
+ break;
+ default:
+ mirc += QChar(0x2400 + c.unicode());
+ }