X-Git-Url: https://git.quassel-irc.org/?a=blobdiff_plain;f=src%2Fuisupport%2Fmultilineedit.cpp;h=1001ba97aa7a6190cce038f0281e75851258daaf;hb=745f33ace5377addd6fec7991db594dd8c64984d;hp=699d4ec825eac48b46b122340146da61df96e0a9;hpb=09d19fe065125ccc6e406fa325d675387788ab3c;p=quassel.git diff --git a/src/uisupport/multilineedit.cpp b/src/uisupport/multilineedit.cpp index 699d4ec8..1001ba97 100644 --- a/src/uisupport/multilineedit.cpp +++ b/src/uisupport/multilineedit.cpp @@ -59,6 +59,24 @@ MultiLineEdit::MultiLineEdit(QWidget *parent) reset(); connect(this, SIGNAL(textChanged()), this, SLOT(on_textChanged())); + + _mircColorMap["00"] = "#ffffff"; + _mircColorMap["01"] = "#000000"; + _mircColorMap["02"] = "#000080"; + _mircColorMap["03"] = "#008000"; + _mircColorMap["04"] = "#ff0000"; + _mircColorMap["05"] = "#800000"; + _mircColorMap["06"] = "#800080"; + _mircColorMap["07"] = "#ffa500"; + _mircColorMap["08"] = "#ffff00"; + _mircColorMap["09"] = "#00ff00"; + _mircColorMap["10"] = "#008080"; + _mircColorMap["11"] = "#00ffff"; + _mircColorMap["12"] = "#4169e1"; + _mircColorMap["13"] = "#ff00ff"; + _mircColorMap["14"] = "#808080"; + _mircColorMap["15"] = "#c0c0c0"; + } MultiLineEdit::~MultiLineEdit() { @@ -171,7 +189,7 @@ void MultiLineEdit::setPasteProtectionEnabled(bool enable, QWidget *) { } void MultiLineEdit::historyMoveBack() { - addToHistory(text(), true); + addToHistory(convertHtmlToMircCodes(html()), true); if(idx > 0) { idx--; @@ -180,7 +198,7 @@ void MultiLineEdit::historyMoveBack() { } void MultiLineEdit::historyMoveForward() { - addToHistory(text(), true); + addToHistory(convertHtmlToMircCodes(html()), true); if(idx < history.count()) { idx++; @@ -189,7 +207,7 @@ void MultiLineEdit::historyMoveForward() { else reset(); // equals clear() in this case } else { - addToHistory(text()); + addToHistory(convertHtmlToMircCodes(html())); reset(); } } @@ -231,8 +249,11 @@ void MultiLineEdit::keyPressEvent(QKeyEvent *event) { # endif #endif - if(_mode == SingleLine) + if(_mode == SingleLine) { + event->accept(); + on_returnPressed(); return; + } #ifdef HAVE_KDE KTextEdit::keyPressEvent(event); #else @@ -296,8 +317,177 @@ void MultiLineEdit::keyPressEvent(QKeyEvent *event) { #endif } +QString MultiLineEdit::convertHtmlToMircCodes(const QString &text) { + qWarning() << text; + + QRegExp regexHtmlContent = QRegExp("(.*)

", Qt::CaseInsensitive); + regexHtmlContent.setMinimal(true); + + QRegExp regexLines = QRegExp("(.*)(?:
)?", Qt::CaseInsensitive); + regexLines.setMinimal(true); + + QRegExp regexStyles = QRegExp("(?:(()(.*)))", Qt::CaseInsensitive); + regexStyles.setMinimal(true); + + QRegExp regexColors = QRegExp("((?:background-)?color):(#[0-9a-f]{6})", Qt::CaseInsensitive); + regexStyles.setMinimal(true); + + QStringList result; + QString htmlContent, line, line2, styleText, style, content; + + if (regexHtmlContent.indexIn((text)) > -1) { + htmlContent = regexHtmlContent.cap(1); + qWarning() << htmlContent; + QStringList lines = htmlContent.split("
"); + for (int i=0; i < lines.count(); i++) { + line = line2 = lines[i]; + int posStyles = 0; + while ((posStyles = regexStyles.indexIn(line2, posStyles)) != -1) { + styleText = regexStyles.cap(1); + style = regexStyles.cap(2); + content = regexStyles.cap(3); + + if (style.contains("font-weight:600;")) { + content.prepend('\x02'); + content.append('\x02'); + } + if (style.contains("font-style:italic;")) { + content.prepend('\x1d'); + content.append('\x1d'); + } + if (style.contains("text-decoration: underline;")) { + content.prepend('\x1f'); + content.append('\x1f'); + } + if (style.contains("color:#")) { // we have either foreground or background color or both + int posColors = 0; + QString mircFgColor, mircBgColor; + while ((posColors = regexColors.indexIn(style, posColors)) != -1) { + QString colorType = regexColors.cap(1); + QString color = regexColors.cap(2); + + if (colorType == "color") + mircFgColor = _mircColorMap.key(color); + + if (colorType == "background-color") + mircBgColor = _mircColorMap.key(color); + + posColors += regexColors.matchedLength(); + } + if (!mircBgColor.isEmpty()) + content.prepend("," + mircBgColor); + + // we need a fg color to be able to use a bg color + if (mircFgColor.isEmpty()) { + //FIXME try to use the current forecolor + mircFgColor = _mircColorMap.key(textColor().name()); + if (mircFgColor.isEmpty()) + mircFgColor = "01"; //use black if the current foreground color can't be converted + } + + content.prepend(mircFgColor); + content.prepend('\x03'); + content.append('\x03'); + } + + line.replace(styleText, content); + posStyles += regexStyles.matchedLength(); + } + + // get rid of all remaining html tags + QRegExp regexTags = QRegExp("<.*>",Qt::CaseInsensitive); + regexTags.setMinimal(true); + line.replace(regexTags, ""); + + line.replace("&","&"); + line.replace("<","<"); + line.replace(">",">"); + line.replace(""","\""); + + qWarning() << line; + qWarning() << line2; + result << line; + } + } + return result.join("\n"); +} + +QString MultiLineEdit::convertMircCodesToHtml(const QString &text) { + QStringList words; + QRegExp mircCode = QRegExp("(|||)", Qt::CaseSensitive); + + int posLeft = 0; + int posRight = 0; + + for(;;) { + posRight = mircCode.indexIn(text, posLeft); + + if(posRight < 0) { + words << text.mid(posLeft); + break; // no more mirc color codes + } + + if (posLeft < posRight) { + words << text.mid(posLeft, posRight - posLeft); + posLeft = posRight; + } + + posRight = text.indexOf(mircCode.cap(), posRight + 1); + words << text.mid(posLeft, posRight + 1 - posLeft); + posLeft = posRight + 1; + } + + for (int i = 0; i < words.count(); i++) { + QString style; + if (words[i].contains('\x02')) { + style.append(" font-weight:600;"); + words[i].replace('\x02',""); + } + if (words[i].contains('\x1d')) { + style.append(" font-style:italic;"); + words[i].replace('\x1d',""); + } + if (words[i].contains('\x1f')) { + style.append(" text-decoration: underline;"); + words[i].replace('\x1f',""); + } + if (words[i].contains('\x03')) { + int pos = words[i].indexOf('\x03'); + int len = 3; + QString fg = words[i].mid(pos + 1,2); + QString bg; + if (words[i][pos+3] == ',') + bg = words[i].mid(pos+4,2); + + style.append(" color:"); + style.append(_mircColorMap[fg]); + style.append(";"); + + if (!bg.isEmpty()) { + style.append(" background-color:"); + style.append(_mircColorMap[bg]); + style.append(";"); + len = 6; + } + words[i].replace(pos, len, ""); + words[i].replace('\x03',""); + } + words[i].replace("&","&"); + words[i].replace("<", "<"); + words[i].replace(">", ">"); + words[i].replace("\"", """); + if (style.isEmpty()) { + words[i] = "" + words[i] + ""; + } + else { + words[i] = "" + words[i] + ""; + } + } + return words.join(""); +} + void MultiLineEdit::on_returnPressed() { - on_returnPressed(text()); + on_returnPressed(convertHtmlToMircCodes(html())); } void MultiLineEdit::on_returnPressed(const QString & text) { @@ -310,6 +500,8 @@ void MultiLineEdit::on_returnPressed(const QString & text) { } reset(); tempHistory.clear(); + } else { + emit noTextEntered(); } } @@ -358,6 +550,7 @@ void MultiLineEdit::on_textChanged() { on_documentHeightChanged(_lastDocumentHeight); } updateSizeHint(); + ensureCursorVisible(); } void MultiLineEdit::on_documentHeightChanged(qreal) { @@ -376,7 +569,8 @@ void MultiLineEdit::reset() { void MultiLineEdit::showHistoryEntry() { // if the user changed the history, display the changed line - setPlainText(tempHistory.contains(idx) ? tempHistory[idx] : history[idx]); + setHtml(convertMircCodesToHtml(tempHistory.contains(idx) ? tempHistory[idx] : history[idx])); + //setPlainText(tempHistory.contains(idx) ? tempHistory[idx] : history[idx]); QTextCursor cursor = textCursor(); QTextBlockFormat format = cursor.blockFormat(); format.setLeftMargin(leftMargin); // we want a little space between the frame and the contents