This commit is contained in:
DavidXanatos 2024-11-20 11:03:25 +01:00
parent 0cf5d8dc90
commit ca894bbd2a
10 changed files with 506 additions and 11 deletions

View File

@ -0,0 +1,135 @@
#include "stdafx.h"
#include "CodeEdit.h"
#define TAB_SPACES " "
CCodeEdit::CCodeEdit(QSyntaxHighlighter* pHighlighter, QWidget* pParent)
: QWidget(pParent)
{
m_pMainLayout = new QGridLayout(this);
m_pMainLayout->setContentsMargins(0,0,0,0);
setLayout(m_pMainLayout);
m_pSourceCode = new QTextEdit();
QFont Font = m_pSourceCode->font();
Font.setFamily("Courier New");
Font.setPointSize(10);
m_pSourceCode->setFont(Font);
m_pSourceCode->setLineWrapMode(QTextEdit::NoWrap);
if(pHighlighter)
pHighlighter->setDocument(m_pSourceCode->document());
//m_pSourceCode->setTabStopWidth (QFontMetrics(Font).width(TAB_SPACES));
m_pMainLayout->addWidget(m_pSourceCode, 0, 0);
connect(m_pSourceCode, SIGNAL(textChanged()), this, SIGNAL(textChanged()));
// hot keys
m_pFind = new QAction(tr("Find"),this);
m_pFind->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_F));
connect(m_pFind, SIGNAL(triggered()), this, SLOT(OnFind()));
m_pSourceCode->addAction(m_pFind);
m_pFindNext = new QAction(tr("FindNext"),this);
QList<QKeySequence> Finds;
Finds << QKeySequence(Qt::Key_F3);
Finds << QKeySequence(Qt::SHIFT | Qt::Key_F3) << QKeySequence(Qt::CTRL | Qt::Key_F3) << QKeySequence(Qt::ALT | Qt::Key_F3);
Finds << QKeySequence(Qt::CTRL | Qt::ALT | Qt::Key_F3) << QKeySequence(Qt::SHIFT | Qt::ALT | Qt::Key_F3) << QKeySequence(Qt::SHIFT | Qt::CTRL | Qt::Key_F3);
Finds << QKeySequence(Qt::SHIFT | Qt::CTRL | Qt::ALT | Qt::Key_F3);
m_pFindNext->setShortcuts(Finds);
connect(m_pFindNext, SIGNAL(triggered()), this, SLOT(OnFindNext()));
m_pSourceCode->addAction(m_pFindNext);
m_pGoTo = new QAction(tr("GoTo"),this);
m_pGoTo->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_G));
connect(m_pGoTo, SIGNAL(triggered()), this, SLOT(OnGoTo()));
m_pSourceCode->addAction(m_pGoTo);
/*m_pComment = new QAction(tr("Comment"),this);
m_pComment->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_R));
connect(m_pComment, SIGNAL(triggered()), this, SLOT(OnComment()));
m_pSourceCode->addAction(m_pComment);
m_pUnComment = new QAction(tr("UnComment"),this);
m_pUnComment->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_T));
connect(m_pUnComment, SIGNAL(triggered()), this, SLOT(OnUnComment()));
m_pSourceCode->addAction(m_pUnComment);*/
}
#define ADD_HISTORY(list,entry) \
list.removeAll(entry); \
list.prepend(entry); \
while(list.size() > 10) \
list.removeLast();
void CCodeEdit::OnFind()
{
static QStringList Finds;
bool bOK = false;
m_CurFind = QInputDialog::getItem (this, tr("Find"),tr("F3: Find Next\n+ Shift: Backward\n+ Ctrl: Case Sensitively\n+ Alt: Whole Words\n\nFind String:") + QString(160,' '), Finds, 0, true, &bOK);
if (!bOK)
return;
ADD_HISTORY(Finds, m_CurFind);
OnFindNext();
}
void CCodeEdit::OnFindNext()
{
if(m_CurFind.isEmpty())
return;
QTextDocument::FindFlags Flags = QTextDocument::FindFlags();
Qt::KeyboardModifiers Mods = QApplication::keyboardModifiers();
if(Mods & Qt::ShiftModifier)
Flags |= QTextDocument::FindBackward;
if(Mods & Qt::ControlModifier)
Flags |= QTextDocument::FindCaseSensitively;
if(Mods & Qt::AltModifier)
Flags |= QTextDocument::FindWholeWords;
m_pSourceCode->find(m_CurFind, Flags);
}
void CCodeEdit::OnGoTo()
{
int iLine = QInputDialog::getText(this, tr("Go to Line:"),tr(""), QLineEdit::Normal, "").toInt();
if(!iLine)
return;
QTextCursor Cursor = m_pSourceCode->textCursor();
Cursor.movePosition(QTextCursor::Start);
while(iLine-- > 1)
Cursor.movePosition(QTextCursor::Down, QTextCursor::MoveAnchor);
//Cursor.select(QTextCursor::LineUnderCursor);
m_pSourceCode->setTextCursor(Cursor);
}
/*void CCodeEdit::OnComment()
{
QTextCursor Cursor = m_pSourceCode->textCursor();
int Start = Cursor.selectionStart();
int End = Cursor.selectionEnd();
QString Text = m_pSourceCode->toPlainText();
QString Fragment = Text.mid(Start, End - Start);
Fragment.replace(QRegExp("^"),"'");
Fragment.replace(QRegExp("\r?\n"),"\r\n'");
m_pSourceCode->insertPlainText(Fragment);
}
void CCodeEdit::OnUnComment()
{
QTextCursor Cursor = m_pSourceCode->textCursor();
int Start = Cursor.selectionStart();
int End = Cursor.selectionEnd();
QString Text = m_pSourceCode->toPlainText();
QString Fragment = Text.mid(Start, End - Start);
Fragment.replace(QRegExp("^[ \t]*'"),"");
Fragment.replace(QRegExp("\r?\n[ \t]*'"),"\r\n");
m_pSourceCode->insertPlainText(Fragment);
}*/

View File

@ -0,0 +1,40 @@
#pragma once
#include "../mischelpers_global.h"
#include <QSyntaxHighlighter>
class MISCHELPERS_EXPORT CCodeEdit : public QWidget
{
Q_OBJECT
public:
CCodeEdit(QSyntaxHighlighter* pHighlighter, QWidget* pParent = 0);
void SetCode(const QString& Code) {m_pSourceCode->setPlainText(Code);}
QString GetCode() {return m_pSourceCode->toPlainText();}
signals:
void textChanged();
private slots:
void OnFind();
void OnFindNext();
void OnGoTo();
/*void OnComment();
void OnUnComment();*/
private:
QGridLayout* m_pMainLayout;
QTextEdit* m_pSourceCode;
QAction* m_pFind;
QAction* m_pFindNext;
QAction* m_pGoTo;
/*QAction* m_pComment;
QAction* m_pUnComment;*/
QString m_CurFind;
};

View File

@ -37,6 +37,7 @@ HEADERS += ./MiscHelpers.h \
./Common/MT/ThreadLock.h \ ./Common/MT/ThreadLock.h \
./Common/MultiErrorDialog.h \ ./Common/MultiErrorDialog.h \
./Common/CheckableComboBox.h \ ./Common/CheckableComboBox.h \
./Common/CodeEdit.h \
./Archive/Archive.h \ ./Archive/Archive.h \
./Archive/ArchiveFS.h \ ./Archive/ArchiveFS.h \
./Archive/ArchiveExtractor.h \ ./Archive/ArchiveExtractor.h \
@ -74,6 +75,7 @@ SOURCES += ./MiscHelpers.cpp \
./Common/MT/ThreadLock.cpp \ ./Common/MT/ThreadLock.cpp \
./Common/MultiErrorDialog.cpp \ ./Common/MultiErrorDialog.cpp \
./Common/CheckableComboBox.cpp \ ./Common/CheckableComboBox.cpp \
./Common/CodeEdit.cpp \
./Archive/Archive.cpp \ ./Archive/Archive.cpp \
./Archive/ArchiveFS.cpp \ ./Archive/ArchiveFS.cpp \
./Archive/ArchiveExtractor.cpp \ ./Archive/ArchiveExtractor.cpp \

View File

@ -0,0 +1,246 @@
#include "stdafx.h"
#include "IniHighlighter.h"
CIniHighlighter::CIniHighlighter(QTextDocument *parent)
: QSyntaxHighlighter(parent)
{
HighlightRule rule;
// Section headers: [Section]
sectionFormat.setForeground(QColor("#0000FF")); // Blue
sectionFormat.setFontWeight(QFont::Bold);
rule.pattern = QRegularExpression("^\\s*\\[.*\\]\\s*$");
rule.format = sectionFormat;
highlightRules.append(rule);
// Comments: ; comment or # comment
commentFormat.setForeground(QColor("#008000")); // Green
rule.pattern = QRegularExpression("^\\s*[;#].*");
rule.format = commentFormat;
highlightRules.append(rule);
// Keys: key=
keyFormat.setForeground(QColor("#800000")); // Dark Red
rule.pattern = QRegularExpression("^[\\w\\.]+(?=\\s*=)");
rule.format = keyFormat;
highlightRules.append(rule);
// Equals sign: =
equalsFormat.setForeground(QColor("#FF0000")); // Red
rule.pattern = QRegularExpression("=");
rule.format = equalsFormat;
highlightRules.append(rule);
// Values: =value
valueFormat.setForeground(QColor("#000000")); // Black
rule.pattern = QRegularExpression("(?<=\\=).*");
rule.format = valueFormat;
highlightRules.append(rule);
// Initialize formats for value prefix and first comma
valuePrefixFormat.setForeground(QColor("#0000FF")); // Blue
firstCommaFormat.setForeground(QColor("#FF0000")); // Red
#ifdef INI_WITH_JSON
// Initialize JSON formats
jsonKeyFormat.setForeground(QColor("#A52A2A")); // Brown
jsonStringFormat.setForeground(QColor("#000000")); // Black
jsonNumberFormat.setForeground(QColor("#0000FF")); // Blue
jsonBoolNullFormat.setForeground(QColor("#800080")); // Purple
jsonBracesFormat.setForeground(QColor("#808080")); // Gray
jsonColonFormat.setForeground(QColor("#FF0000")); // Red
jsonCommaFormat.setForeground(QColor("#FF0000")); // Red
// 1. JSON Colon: Match colons not preceded by backslash
HighlightRule jsonRule;
jsonRule.pattern = QRegularExpression(R"((?<!\\):)");
jsonRule.format = jsonColonFormat;
jsonHighlightRules.append(jsonRule);
// 2. JSON Comma: Match commas not preceded by backslash
jsonRule.pattern = QRegularExpression(R"((?<!\\),)");
jsonRule.format = jsonCommaFormat;
jsonHighlightRules.append(jsonRule);
// 3. JSON Keys: "key":
jsonRule.pattern = QRegularExpression(R"("(?:(?:\\.)|[^"\\])*"(?=\s*:))");
jsonRule.format = jsonKeyFormat;
jsonHighlightRules.append(jsonRule);
// 4. JSON Strings: "value" (excluding keys)
jsonRule.pattern = QRegularExpression(R"("(?:(?:\\.)|[^"\\])*"(?!\s*:))");
jsonRule.format = jsonStringFormat;
jsonHighlightRules.append(jsonRule);
// 5. JSON Numbers: 123, 45.67
jsonRule.pattern = QRegularExpression(R"(\b-?\d+(\.\d+)?\b)");
jsonRule.format = jsonNumberFormat;
jsonHighlightRules.append(jsonRule);
// 6. JSON Booleans and Null: true, false, null
jsonRule.pattern = QRegularExpression(R"(\b(true|false|null)\b)", QRegularExpression::CaseInsensitiveOption);
jsonRule.format = jsonBoolNullFormat;
jsonHighlightRules.append(jsonRule);
// 7. JSON Braces and Brackets: { }, [ ]
jsonRule.pattern = QRegularExpression(R"([\{\}\[\]])");
jsonRule.format = jsonBracesFormat;
jsonHighlightRules.append(jsonRule);
#endif
}
CIniHighlighter::~CIniHighlighter()
{
}
void CIniHighlighter::highlightBlock(const QString &text)
{
// First, reset all formatting
setFormat(0, text.length(), QTextCharFormat());
// 1. Check if the entire line is a comment
QRegularExpression commentRegex(R"(^\s*[;#].*)");
QRegularExpressionMatch commentMatch = commentRegex.match(text);
if (commentMatch.hasMatch()) {
setFormat(0, text.length(), commentFormat);
return; // Skip other rules
}
// 2. Apply INI highlighting rules (section, key, equals, value)
for (const HighlightRule &rule : qAsConst(highlightRules)) {
// Skip the comment rule as it's already handled
if (rule.format.foreground() == commentFormat.foreground())
continue;
QRegularExpressionMatchIterator matchIterator = rule.pattern.globalMatch(text);
while (matchIterator.hasNext()) {
QRegularExpressionMatch match = matchIterator.next();
int start = match.capturedStart();
int length = match.capturedLength();
setFormat(start, length, rule.format);
}
}
// 3. Process the value part for value prefixes and first comma
// Find the position of '=' to identify the start of the value
int equalsIndex = text.indexOf('=');
if (equalsIndex != -1) {
// Start position of the value (after '=')
int valueStart = equalsIndex + 1;
QString valueText = text.mid(valueStart).trimmed();
// Iterate through the value to find the first comma outside of {}
int braceLevel = 0;
int firstRelevantCommaIndex = -1;
for (int i = 0; i < valueText.length(); ++i) {
QChar currentChar = valueText[i];
if (currentChar == '{') {
braceLevel++;
} else if (currentChar == '}') {
if (braceLevel > 0)
braceLevel--;
} else if (currentChar == ',' && braceLevel == 0) {
firstRelevantCommaIndex = i;
break;
}
}
if (firstRelevantCommaIndex != -1) {
// Position of the first comma relative to the entire line
int commaPos = valueStart + firstRelevantCommaIndex;
// Highlight text before the first comma in blue
if (firstRelevantCommaIndex > 0) {
setFormat(valueStart, firstRelevantCommaIndex, valuePrefixFormat);
}
// Highlight the first comma in red
setFormat(commaPos, 1, firstCommaFormat);
}
#ifdef INI_WITH_JSON
bool inString = false;
int stringStart = -1;
for (int i = firstRelevantCommaIndex != -1 ? firstRelevantCommaIndex + 1 : 0; i < valueText.length(); ++i) {
QChar currentChar = valueText[i];
if (currentChar == '\"') {
// Check if the quote is escaped
bool escaped = false;
int backslashCount = 0;
int j = i - 1;
while (j >= 0 && valueText[j] == '\\') {
backslashCount++;
j--;
}
if (backslashCount % 2 == 1)
escaped = true;
if (!escaped) {
if (!inString) {
inString = true;
stringStart = valueStart + i;
}
else {
inString = false;
// Apply string formatting from stringStart to current position
int length = (valueStart + i + 1) - stringStart;
setFormat(stringStart, length, jsonStringFormat);
}
}
}
// Apply colon and comma formatting only if not inside a string
if (!inString) {
if (currentChar == ':') {
setFormat(valueStart + i, 1, jsonColonFormat);
}
else if (currentChar == ',') {
setFormat(valueStart + i, 1, jsonCommaFormat);
}
}
}
// If still inside a string (unclosed), format till end
if (inString && stringStart != -1) {
int length = text.length() - stringStart;
setFormat(stringStart, length, jsonStringFormat);
}
// 4. Apply JSON Key Formatting within JSON Substrings
// Find all JSON substrings and apply key formatting
int current = 0;
while (current < valueText.length()) {
int startBrace = valueText.indexOf('{', current);
if (startBrace == -1)
break;
int braceCounter = 1;
int endBrace = startBrace + 1;
while (endBrace < valueText.length() && braceCounter > 0) {
if (valueText[endBrace] == '{')
braceCounter++;
else if (valueText[endBrace] == '}')
braceCounter--;
endBrace++;
}
if (braceCounter == 0) {
// Found a JSON substring from startBrace to endBrace-1
QString jsonString = valueText.mid(startBrace, endBrace - startBrace);
QRegularExpression keyRegex(R"("(?:(?:\\.)|[^"\\])*"(?=\s*:))");
QRegularExpressionMatchIterator keyMatches = keyRegex.globalMatch(jsonString);
while (keyMatches.hasNext()) {
QRegularExpressionMatch keyMatch = keyMatches.next();
int keyStart = valueStart + startBrace + keyMatch.capturedStart();
int keyLength = keyMatch.capturedLength();
setFormat(keyStart, keyLength, jsonKeyFormat);
}
current = endBrace;
}
else {
break;
}
}
#endif
}
}

View File

@ -0,0 +1,46 @@
#pragma once
#include <QSyntaxHighlighter>
#define INI_WITH_JSON
class CIniHighlighter : public QSyntaxHighlighter
{
Q_OBJECT
public:
explicit CIniHighlighter(QTextDocument *parent = nullptr);
~CIniHighlighter();
protected:
void highlightBlock(const QString &text) override;
private:
struct HighlightRule {
QRegularExpression pattern;
QTextCharFormat format;
};
QVector<HighlightRule> highlightRules;
QTextCharFormat sectionFormat;
QTextCharFormat keyFormat;
QTextCharFormat valueFormat;
QTextCharFormat commentFormat;
QTextCharFormat equalsFormat; // New format for '=' character
QTextCharFormat valuePrefixFormat; // Format for text before first comma in value
QTextCharFormat firstCommaFormat; // Format for the first comma in value
#ifdef INI_WITH_JSON
// JSON-specific formats
QTextCharFormat jsonKeyFormat;
QTextCharFormat jsonStringFormat;
QTextCharFormat jsonNumberFormat;
QTextCharFormat jsonBoolNullFormat;
QTextCharFormat jsonBracesFormat;
QTextCharFormat jsonColonFormat;
QTextCharFormat jsonCommaFormat;
// JSON highlighting rules
QVector<HighlightRule> jsonHighlightRules;
#endif
};

View File

@ -24,6 +24,7 @@ HEADERS += ./stdafx.h \
./Helpers/ReadDirectoryChanges.h \ ./Helpers/ReadDirectoryChanges.h \
./Helpers/ReadDirectoryChangesPrivate.h \ ./Helpers/ReadDirectoryChangesPrivate.h \
./Helpers/TabOrder.h \ ./Helpers/TabOrder.h \
./Helpers/IniHighlighter.h \
./Windows/RecoveryWindow.h \ ./Windows/RecoveryWindow.h \
./Windows/PopUpWindow.h \ ./Windows/PopUpWindow.h \
./Windows/SnapshotsWindow.h \ ./Windows/SnapshotsWindow.h \
@ -74,6 +75,7 @@ SOURCES += ./main.cpp \
./Helpers/StorageInfo.cpp \ ./Helpers/StorageInfo.cpp \
./Helpers/ReadDirectoryChanges.cpp \ ./Helpers/ReadDirectoryChanges.cpp \
./Helpers/ReadDirectoryChangesPrivate.cpp \ ./Helpers/ReadDirectoryChangesPrivate.cpp \
./Helpers/IniHighlighter.cpp \
./Helpers/WindowFromPointEx.cpp \ ./Helpers/WindowFromPointEx.cpp \
./Helpers/TabOrder.cpp \ ./Helpers/TabOrder.cpp \
./Windows/OptionsWindow.cpp \ ./Windows/OptionsWindow.cpp \

View File

@ -9,6 +9,8 @@
#include "Helpers/WinAdmin.h" #include "Helpers/WinAdmin.h"
#include "../Wizards/TemplateWizard.h" #include "../Wizards/TemplateWizard.h"
#include "Helpers/TabOrder.h" #include "Helpers/TabOrder.h"
#include "../MiscHelpers/Common/CodeEdit.h"
#include "Helpers/IniHighlighter.h"
class NoEditDelegate : public QStyledItemDelegate { class NoEditDelegate : public QStyledItemDelegate {
@ -575,14 +577,17 @@ COptionsWindow::COptionsWindow(const QSharedPointer<CSbieIni>& pBox, const QStri
connect(ui.tabs, SIGNAL(currentChanged(int)), this, SLOT(OnTab())); connect(ui.tabs, SIGNAL(currentChanged(int)), this, SLOT(OnTab()));
// edit // edit
m_pCodeEdit = new CCodeEdit(new CIniHighlighter);
ui.txtIniSection->parentWidget()->layout()->replaceWidget(ui.txtIniSection, m_pCodeEdit);
ui.txtIniSection->deleteLater();
connect(m_pCodeEdit, SIGNAL(textChanged()), this, SLOT(OnIniChanged()));
ApplyIniEditFont(); ApplyIniEditFont();
connect(ui.btnEditIni, SIGNAL(clicked(bool)), this, SLOT(OnEditIni())); connect(ui.btnEditIni, SIGNAL(clicked(bool)), this, SLOT(OnEditIni()));
connect(ui.btnSaveIni, SIGNAL(clicked(bool)), this, SLOT(OnSaveIni())); connect(ui.btnSaveIni, SIGNAL(clicked(bool)), this, SLOT(OnSaveIni()));
connect(ui.btnCancelEdit, SIGNAL(clicked(bool)), this, SLOT(OnCancelEdit())); connect(ui.btnCancelEdit, SIGNAL(clicked(bool)), this, SLOT(OnCancelEdit()));
connect(ui.txtIniSection, SIGNAL(textChanged()), this, SLOT(OnIniChanged())); //connect(ui.txtIniSection, SIGNAL(textChanged()), this, SLOT(OnIniChanged()));
//
connect(ui.buttonBox->button(QDialogButtonBox::Ok), SIGNAL(clicked(bool)), this, SLOT(ok())); connect(ui.buttonBox->button(QDialogButtonBox::Ok), SIGNAL(clicked(bool)), this, SLOT(ok()));
connect(ui.buttonBox->button(QDialogButtonBox::Apply), SIGNAL(clicked(bool)), this, SLOT(apply())); connect(ui.buttonBox->button(QDialogButtonBox::Apply), SIGNAL(clicked(bool)), this, SLOT(apply()));
@ -679,7 +684,8 @@ void COptionsWindow::ApplyIniEditFont()
QFont font; // defaults to application font QFont font; // defaults to application font
auto fontName = theConf->GetString("UIConfig/IniFont", "").trimmed(); auto fontName = theConf->GetString("UIConfig/IniFont", "").trimmed();
if (!fontName.isEmpty()) bool dummy = font.fromString(fontName); // ignore fromString() fail if (!fontName.isEmpty()) bool dummy = font.fromString(fontName); // ignore fromString() fail
ui.txtIniSection->setFont(font); //ui.txtIniSection->setFont(font);
m_pCodeEdit->setFont(font);
} }
void COptionsWindow::OnSetTree() void COptionsWindow::OnSetTree()
@ -1300,7 +1306,8 @@ void COptionsWindow::LoadIniSection()
Section = m_pBox->GetAPI()->SbieIniGetEx(m_pBox->GetName(), ""); Section = m_pBox->GetAPI()->SbieIniGetEx(m_pBox->GetName(), "");
m_HoldChange = true; m_HoldChange = true;
ui.txtIniSection->setPlainText(Section); //ui.txtIniSection->setPlainText(Section);
m_pCodeEdit->SetCode(Section);
m_HoldChange = false; m_HoldChange = false;
} }
@ -1341,7 +1348,8 @@ void COptionsWindow::SaveIniSection()
m_pBox->SetRefreshOnChange(true); m_pBox->SetRefreshOnChange(true);
m_pBox->GetAPI()->CommitIniChanges();*/ m_pBox->GetAPI()->CommitIniChanges();*/
m_pBox->GetAPI()->SbieIniSet(m_pBox->GetName(), "", ui.txtIniSection->toPlainText()); //m_pBox->GetAPI()->SbieIniSet(m_pBox->GetName(), "", ui.txtIniSection->toPlainText());
m_pBox->GetAPI()->SbieIniSet(m_pBox->GetName(), "", m_pCodeEdit->GetCode());
LoadIniSection(); LoadIniSection();
} }

View File

@ -639,5 +639,7 @@ private:
QMap<QCheckBox*, SDbgOpt> m_DebugOptions; QMap<QCheckBox*, SDbgOpt> m_DebugOptions;
void InitLangID(); void InitLangID();
class CCodeEdit* m_pCodeEdit;
}; };

View File

@ -18,6 +18,8 @@
#include <QJsonDocument> #include <QJsonDocument>
#include <QJsonObject> #include <QJsonObject>
#include "Helpers/TabOrder.h" #include "Helpers/TabOrder.h"
#include "../MiscHelpers/Common/CodeEdit.h"
#include "Helpers/IniHighlighter.h"
#include <windows.h> #include <windows.h>
@ -597,6 +599,12 @@ CSettingsWindow::CSettingsWindow(QWidget* parent)
ui.btnSelectIniFont->setToolTip(tr("Select font")); ui.btnSelectIniFont->setToolTip(tr("Select font"));
ui.btnResetIniFont->setIcon(CSandMan::GetIcon("ResetFont")); ui.btnResetIniFont->setIcon(CSandMan::GetIcon("ResetFont"));
ui.btnResetIniFont->setToolTip(tr("Reset font")); ui.btnResetIniFont->setToolTip(tr("Reset font"));
m_pCodeEdit = new CCodeEdit(new CIniHighlighter);
ui.txtIniSection->parentWidget()->layout()->replaceWidget(ui.txtIniSection, m_pCodeEdit);
ui.txtIniSection->deleteLater();
connect(m_pCodeEdit, SIGNAL(textChanged()), this, SLOT(OnIniChanged()));
ApplyIniEditFont(); ApplyIniEditFont();
connect(ui.btnSelectIniFont, SIGNAL(clicked(bool)), this, SLOT(OnSelectIniEditFont())); connect(ui.btnSelectIniFont, SIGNAL(clicked(bool)), this, SLOT(OnSelectIniEditFont()));
@ -604,7 +612,7 @@ CSettingsWindow::CSettingsWindow(QWidget* parent)
connect(ui.btnEditIni, SIGNAL(clicked(bool)), this, SLOT(OnEditIni())); connect(ui.btnEditIni, SIGNAL(clicked(bool)), this, SLOT(OnEditIni()));
connect(ui.btnSaveIni, SIGNAL(clicked(bool)), this, SLOT(OnSaveIni())); connect(ui.btnSaveIni, SIGNAL(clicked(bool)), this, SLOT(OnSaveIni()));
connect(ui.btnCancelEdit, SIGNAL(clicked(bool)), this, SLOT(OnCancelEdit())); connect(ui.btnCancelEdit, SIGNAL(clicked(bool)), this, SLOT(OnCancelEdit()));
connect(ui.txtIniSection, SIGNAL(textChanged()), this, SLOT(OnIniChanged())); //connect(ui.txtIniSection, SIGNAL(textChanged()), this, SLOT(OnIniChanged()));
// //
connect(ui.buttonBox->button(QDialogButtonBox::Ok), SIGNAL(clicked(bool)), this, SLOT(ok())); connect(ui.buttonBox->button(QDialogButtonBox::Ok), SIGNAL(clicked(bool)), this, SLOT(ok()));
@ -657,14 +665,16 @@ void CSettingsWindow::ApplyIniEditFont()
QFont font; // defaults to application font QFont font; // defaults to application font
auto fontName = theConf->GetString("UIConfig/IniFont", "").trimmed(); auto fontName = theConf->GetString("UIConfig/IniFont", "").trimmed();
if (!fontName.isEmpty()) bool dummy = font.fromString(fontName); // ignore fromString() fail if (!fontName.isEmpty()) bool dummy = font.fromString(fontName); // ignore fromString() fail
ui.txtIniSection->setFont(font); //ui.txtIniSection->setFont(font);
m_pCodeEdit->setFont(font);
ui.lblIniEditFont->setText(tr("%0, %1 pt").arg(font.family()).arg(font.pointSizeF())); // tr: example: "Calibri, 9.5 pt" ui.lblIniEditFont->setText(tr("%0, %1 pt").arg(font.family()).arg(font.pointSizeF())); // tr: example: "Calibri, 9.5 pt"
} }
void CSettingsWindow::OnSelectIniEditFont() void CSettingsWindow::OnSelectIniEditFont()
{ {
bool ok; bool ok;
auto newFont = QFontDialog::getFont(&ok, ui.txtIniSection->font(), this); //auto newFont = QFontDialog::getFont(&ok, ui.txtIniSection->font(), this);
auto newFont = QFontDialog::getFont(&ok, m_pCodeEdit->font(), this);
if (!ok) return; if (!ok) return;
theConf->SetValue("UIConfig/IniFont", newFont.toString()); theConf->SetValue("UIConfig/IniFont", newFont.toString());
ApplyIniEditFont(); ApplyIniEditFont();
@ -2531,14 +2541,16 @@ void CSettingsWindow::LoadIniSection()
Section = theAPI->SbieIniGetEx("GlobalSettings", ""); Section = theAPI->SbieIniGetEx("GlobalSettings", "");
m_HoldChange = true; m_HoldChange = true;
ui.txtIniSection->setPlainText(Section); //ui.txtIniSection->setPlainText(Section);
m_pCodeEdit->SetCode(Section);
m_HoldChange = false; m_HoldChange = false;
} }
void CSettingsWindow::SaveIniSection() void CSettingsWindow::SaveIniSection()
{ {
if(theAPI->IsConnected()) if(theAPI->IsConnected())
theAPI->SbieIniSet("GlobalSettings", "", ui.txtIniSection->toPlainText()); //theAPI->SbieIniSet("GlobalSettings", "", ui.txtIniSection->toPlainText());
theAPI->SbieIniSet("GlobalSettings", "", m_pCodeEdit->GetCode());
LoadIniSection(); LoadIniSection();
} }

View File

@ -195,6 +195,8 @@ private:
void WriteTextList(const QString& Setting, const QStringList& List); void WriteTextList(const QString& Setting, const QStringList& List);
Ui::SettingsWindow ui; Ui::SettingsWindow ui;
class CCodeEdit* m_pCodeEdit;
}; };
QVariantMap GetRunEntry(const QString& sEntry); QVariantMap GetRunEntry(const QString& sEntry);