1 | /**************************************************************************** |
2 | ** |
3 | ** Copyright (C) 2016 The Qt Company Ltd. |
4 | ** Contact: https://www.qt.io/licensing/ |
5 | ** |
6 | ** This file is part of the Qt Designer of the Qt Toolkit. |
7 | ** |
8 | ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ |
9 | ** Commercial License Usage |
10 | ** Licensees holding valid commercial Qt licenses may use this file in |
11 | ** accordance with the commercial license agreement provided with the |
12 | ** Software or, alternatively, in accordance with the terms contained in |
13 | ** a written agreement between you and The Qt Company. For licensing terms |
14 | ** and conditions see https://www.qt.io/terms-conditions. For further |
15 | ** information use the contact form at https://www.qt.io/contact-us. |
16 | ** |
17 | ** GNU General Public License Usage |
18 | ** Alternatively, this file may be used under the terms of the GNU |
19 | ** General Public License version 3 as published by the Free Software |
20 | ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT |
21 | ** included in the packaging of this file. Please review the following |
22 | ** information to ensure the GNU General Public License requirements will |
23 | ** be met: https://www.gnu.org/licenses/gpl-3.0.html. |
24 | ** |
25 | ** $QT_END_LICENSE$ |
26 | ** |
27 | ****************************************************************************/ |
28 | |
29 | #include "stylesheeteditor_p.h" |
30 | #include "csshighlighter_p.h" |
31 | #include "iconselector_p.h" |
32 | #include "qtgradientmanager.h" |
33 | #include "qtgradientviewdialog.h" |
34 | #include "qtgradientutils.h" |
35 | #include "qdesigner_utils_p.h" |
36 | |
37 | #include <QtDesigner/abstractformwindow.h> |
38 | #include <QtDesigner/abstractformwindowcursor.h> |
39 | #include <QtDesigner/abstractformeditor.h> |
40 | #include <QtDesigner/propertysheet.h> |
41 | #include <QtDesigner/abstractintegration.h> |
42 | #include <QtDesigner/abstractsettings.h> |
43 | #include <QtDesigner/qextensionmanager.h> |
44 | |
45 | #include <texteditfindwidget.h> |
46 | |
47 | #include <QtWidgets/qaction.h> |
48 | #include <QtWidgets/qcolordialog.h> |
49 | #include <QtWidgets/qdialogbuttonbox.h> |
50 | #include <QtWidgets/qfontdialog.h> |
51 | #include <QtWidgets/qmenu.h> |
52 | #include <QtWidgets/qpushbutton.h> |
53 | #include <QtGui/qtextdocument.h> |
54 | #include <QtWidgets/qtoolbar.h> |
55 | #include <QtWidgets/qboxlayout.h> |
56 | #include <private/qcssparser_p.h> |
57 | |
58 | #include <QtGui/qevent.h> |
59 | |
60 | QT_BEGIN_NAMESPACE |
61 | |
62 | static const char *styleSheetProperty = "styleSheet" ; |
63 | static const char *StyleSheetDialogC = "StyleSheetDialog" ; |
64 | static const char *Geometry = "Geometry" ; |
65 | |
66 | namespace qdesigner_internal { |
67 | |
68 | StyleSheetEditor::StyleSheetEditor(QWidget *parent) |
69 | : QTextEdit(parent) |
70 | { |
71 | setTabStopDistance(fontMetrics().horizontalAdvance(QLatin1Char(' ')) * 4); |
72 | setAcceptRichText(false); |
73 | new CssHighlighter(document()); |
74 | } |
75 | |
76 | // --- StyleSheetEditorDialog |
77 | StyleSheetEditorDialog::StyleSheetEditorDialog(QDesignerFormEditorInterface *core, QWidget *parent, Mode mode): |
78 | QDialog(parent), |
79 | m_buttonBox(new QDialogButtonBox(QDialogButtonBox::Ok|QDialogButtonBox::Cancel|QDialogButtonBox::Help)), |
80 | m_editor(new StyleSheetEditor), |
81 | m_findWidget(new TextEditFindWidget), |
82 | m_validityLabel(new QLabel(tr(s: "Valid Style Sheet" ))), |
83 | m_core(core), |
84 | m_addResourceAction(new QAction(tr(s: "Add Resource..." ), this)), |
85 | m_addGradientAction(new QAction(tr(s: "Add Gradient..." ), this)), |
86 | m_addColorAction(new QAction(tr(s: "Add Color..." ), this)), |
87 | m_addFontAction(new QAction(tr(s: "Add Font..." ), this)) |
88 | { |
89 | setWindowTitle(tr(s: "Edit Style Sheet" )); |
90 | setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint); |
91 | |
92 | connect(sender: m_buttonBox, signal: &QDialogButtonBox::accepted, receiver: this, slot: &QDialog::accept); |
93 | connect(sender: m_buttonBox, signal: &QDialogButtonBox::rejected, receiver: this, slot: &QDialog::reject); |
94 | connect(sender: m_buttonBox, signal: &QDialogButtonBox::helpRequested, |
95 | receiver: this, slot: &StyleSheetEditorDialog::slotRequestHelp); |
96 | m_buttonBox->button(which: QDialogButtonBox::Help)->setShortcut(QKeySequence::HelpContents); |
97 | |
98 | connect(sender: m_editor, signal: &QTextEdit::textChanged, receiver: this, slot: &StyleSheetEditorDialog::validateStyleSheet); |
99 | m_findWidget->setTextEdit(m_editor); |
100 | |
101 | QToolBar *toolBar = new QToolBar; |
102 | |
103 | QGridLayout *layout = new QGridLayout; |
104 | layout->addWidget(toolBar, row: 0, column: 0, rowSpan: 1, columnSpan: 2); |
105 | layout->addWidget(m_editor, row: 1, column: 0, rowSpan: 1, columnSpan: 2); |
106 | layout->addWidget(m_findWidget, row: 2, column: 0, rowSpan: 1, columnSpan: 2); |
107 | layout->addWidget(m_validityLabel, row: 3, column: 0, rowSpan: 1, columnSpan: 1); |
108 | layout->addWidget(m_buttonBox, row: 3, column: 1, rowSpan: 1, columnSpan: 1); |
109 | setLayout(layout); |
110 | |
111 | m_editor->setContextMenuPolicy(Qt::CustomContextMenu); |
112 | connect(sender: m_editor, signal: &QWidget::customContextMenuRequested, |
113 | receiver: this, slot: &StyleSheetEditorDialog::slotContextMenuRequested); |
114 | |
115 | connect(sender: m_addResourceAction, signal: &QAction::triggered, |
116 | context: this, slot: [this] { this->slotAddResource(property: QString()); }); |
117 | connect(sender: m_addGradientAction, signal: &QAction::triggered, |
118 | context: this, slot: [this] { this->slotAddGradient(property: QString()); }); |
119 | connect(sender: m_addColorAction, signal: &QAction::triggered, |
120 | context: this, slot: [this] { this->slotAddColor(property: QString()); }); |
121 | connect(sender: m_addFontAction, signal: &QAction::triggered, receiver: this, slot: &StyleSheetEditorDialog::slotAddFont); |
122 | |
123 | m_addResourceAction->setEnabled(mode == ModePerForm); |
124 | |
125 | const char * const resourceProperties[] = { |
126 | "background-image" , |
127 | "border-image" , |
128 | "image" , |
129 | nullptr |
130 | }; |
131 | |
132 | const char * const colorProperties[] = { |
133 | "color" , |
134 | "background-color" , |
135 | "alternate-background-color" , |
136 | "border-color" , |
137 | "border-top-color" , |
138 | "border-right-color" , |
139 | "border-bottom-color" , |
140 | "border-left-color" , |
141 | "gridline-color" , |
142 | "selection-color" , |
143 | "selection-background-color" , |
144 | nullptr |
145 | }; |
146 | |
147 | QMenu * = new QMenu(this); |
148 | QMenu * = new QMenu(this); |
149 | QMenu * = new QMenu(this); |
150 | |
151 | for (int resourceProperty = 0; resourceProperties[resourceProperty]; ++resourceProperty) { |
152 | const QString resourcePropertyName = QLatin1String(resourceProperties[resourceProperty]); |
153 | resourceActionMenu->addAction(text: resourcePropertyName, |
154 | object: this, slot: [this, resourcePropertyName] { this->slotAddResource(property: resourcePropertyName); }); |
155 | } |
156 | |
157 | for (int colorProperty = 0; colorProperties[colorProperty]; ++colorProperty) { |
158 | const QString colorPropertyName = QLatin1String(colorProperties[colorProperty]); |
159 | colorActionMenu->addAction(text: colorPropertyName, |
160 | object: this, slot: [this, colorPropertyName] { this->slotAddColor(property: colorPropertyName); }); |
161 | gradientActionMenu->addAction(text: colorPropertyName, |
162 | object: this, slot: [this, colorPropertyName] { this->slotAddGradient(property: colorPropertyName); } ); |
163 | } |
164 | |
165 | m_addResourceAction->setMenu(resourceActionMenu); |
166 | m_addGradientAction->setMenu(gradientActionMenu); |
167 | m_addColorAction->setMenu(colorActionMenu); |
168 | |
169 | |
170 | toolBar->addAction(action: m_addResourceAction); |
171 | toolBar->addAction(action: m_addGradientAction); |
172 | toolBar->addAction(action: m_addColorAction); |
173 | toolBar->addAction(action: m_addFontAction); |
174 | m_findAction = m_findWidget->createFindAction(parent: toolBar); |
175 | toolBar->addAction(action: m_findAction); |
176 | |
177 | m_editor->setFocus(); |
178 | |
179 | QDesignerSettingsInterface *settings = core->settingsManager(); |
180 | settings->beginGroup(prefix: QLatin1String(StyleSheetDialogC)); |
181 | |
182 | if (settings->contains(key: QLatin1String(Geometry))) |
183 | restoreGeometry(geometry: settings->value(key: QLatin1String(Geometry)).toByteArray()); |
184 | |
185 | settings->endGroup(); |
186 | } |
187 | |
188 | StyleSheetEditorDialog::~StyleSheetEditorDialog() |
189 | { |
190 | QDesignerSettingsInterface *settings = m_core->settingsManager(); |
191 | settings->beginGroup(prefix: QLatin1String(StyleSheetDialogC)); |
192 | |
193 | settings->setValue(key: QLatin1String(Geometry), value: saveGeometry()); |
194 | settings->endGroup(); |
195 | } |
196 | |
197 | void StyleSheetEditorDialog::setOkButtonEnabled(bool v) |
198 | { |
199 | m_buttonBox->button(which: QDialogButtonBox::Ok)->setEnabled(v); |
200 | if (QPushButton *applyButton = m_buttonBox->button(which: QDialogButtonBox::Apply)) |
201 | applyButton->setEnabled(v); |
202 | } |
203 | |
204 | void StyleSheetEditorDialog::(const QPoint &pos) |
205 | { |
206 | QMenu * = m_editor->createStandardContextMenu(); |
207 | menu->addSeparator(); |
208 | menu->addAction(action: m_findAction); |
209 | menu->addSeparator(); |
210 | menu->addAction(action: m_addResourceAction); |
211 | menu->addAction(action: m_addGradientAction); |
212 | menu->exec(pos: mapToGlobal(pos)); |
213 | delete menu; |
214 | } |
215 | |
216 | void StyleSheetEditorDialog::slotAddResource(const QString &property) |
217 | { |
218 | const QString path = IconSelector::choosePixmapResource(core: m_core, resourceModel: m_core->resourceModel(), oldPath: QString(), parent: this); |
219 | if (!path.isEmpty()) |
220 | insertCssProperty(name: property, value: QString(QStringLiteral("url(%1)" )).arg(a: path)); |
221 | } |
222 | |
223 | void StyleSheetEditorDialog::slotAddGradient(const QString &property) |
224 | { |
225 | bool ok; |
226 | const QGradient grad = QtGradientViewDialog::getGradient(ok: &ok, manager: m_core->gradientManager(), parent: this); |
227 | if (ok) |
228 | insertCssProperty(name: property, value: QtGradientUtils::styleSheetCode(gradient: grad)); |
229 | } |
230 | |
231 | void StyleSheetEditorDialog::slotAddColor(const QString &property) |
232 | { |
233 | const QColor color = QColorDialog::getColor(initial: 0xffffffff, parent: this, title: QString(), options: QColorDialog::ShowAlphaChannel); |
234 | if (!color.isValid()) |
235 | return; |
236 | |
237 | QString colorStr; |
238 | |
239 | if (color.alpha() == 255) { |
240 | colorStr = QString(QStringLiteral("rgb(%1, %2, %3)" )).arg( |
241 | a: color.red()).arg(a: color.green()).arg(a: color.blue()); |
242 | } else { |
243 | colorStr = QString(QStringLiteral("rgba(%1, %2, %3, %4)" )).arg( |
244 | a: color.red()).arg(a: color.green()).arg(a: color.blue()).arg(a: color.alpha()); |
245 | } |
246 | |
247 | insertCssProperty(name: property, value: colorStr); |
248 | } |
249 | |
250 | void StyleSheetEditorDialog::slotAddFont() |
251 | { |
252 | bool ok; |
253 | QFont font = QFontDialog::getFont(ok: &ok, parent: this); |
254 | if (ok) { |
255 | QString fontStr; |
256 | if (font.weight() != QFont::Normal) { |
257 | fontStr += QString::number(font.weight()); |
258 | fontStr += QLatin1Char(' '); |
259 | } |
260 | |
261 | switch (font.style()) { |
262 | case QFont::StyleItalic: |
263 | fontStr += QStringLiteral("italic " ); |
264 | break; |
265 | case QFont::StyleOblique: |
266 | fontStr += QStringLiteral("oblique " ); |
267 | break; |
268 | default: |
269 | break; |
270 | } |
271 | fontStr += QString::number(font.pointSize()); |
272 | fontStr += QStringLiteral("pt \"" ); |
273 | fontStr += font.family(); |
274 | fontStr += QLatin1Char('"'); |
275 | |
276 | insertCssProperty(QStringLiteral("font" ), value: fontStr); |
277 | QString decoration; |
278 | if (font.underline()) |
279 | decoration += QStringLiteral("underline" ); |
280 | if (font.strikeOut()) { |
281 | if (!decoration.isEmpty()) |
282 | decoration += QLatin1Char(' '); |
283 | decoration += QStringLiteral("line-through" ); |
284 | } |
285 | insertCssProperty(QStringLiteral("text-decoration" ), value: decoration); |
286 | } |
287 | } |
288 | |
289 | void StyleSheetEditorDialog::insertCssProperty(const QString &name, const QString &value) |
290 | { |
291 | if (!value.isEmpty()) { |
292 | QTextCursor cursor = m_editor->textCursor(); |
293 | if (!name.isEmpty()) { |
294 | cursor.beginEditBlock(); |
295 | cursor.removeSelectedText(); |
296 | cursor.movePosition(op: QTextCursor::EndOfLine); |
297 | |
298 | // Simple check to see if we're in a selector scope |
299 | const QTextDocument *doc = m_editor->document(); |
300 | const QTextCursor closing = doc->find(QStringLiteral("}" ), cursor, options: QTextDocument::FindBackward); |
301 | const QTextCursor opening = doc->find(QStringLiteral("{" ), cursor, options: QTextDocument::FindBackward); |
302 | const bool inSelector = !opening.isNull() && (closing.isNull() || |
303 | closing.position() < opening.position()); |
304 | QString insertion; |
305 | if (m_editor->textCursor().block().length() != 1) |
306 | insertion += QLatin1Char('\n'); |
307 | if (inSelector) |
308 | insertion += QLatin1Char('\t'); |
309 | insertion += name; |
310 | insertion += QStringLiteral(": " ); |
311 | insertion += value; |
312 | insertion += QLatin1Char(';'); |
313 | cursor.insertText(text: insertion); |
314 | cursor.endEditBlock(); |
315 | } else { |
316 | cursor.insertText(text: value); |
317 | } |
318 | } |
319 | } |
320 | |
321 | void StyleSheetEditorDialog::slotRequestHelp() |
322 | { |
323 | m_core->integration()->emitHelpRequested(QStringLiteral("qtwidgets" ), |
324 | QStringLiteral("stylesheet-reference.html" )); |
325 | } |
326 | |
327 | // See QDialog::keyPressEvent() |
328 | static inline bool isEnter(const QKeyEvent *e) |
329 | { |
330 | const bool isEnter = e->key() == Qt::Key_Enter; |
331 | const bool isReturn = e->key() == Qt::Key_Return; |
332 | return (e->modifiers() == Qt::KeyboardModifiers() && (isEnter || isReturn)) |
333 | || (e->modifiers().testFlag(flag: Qt::KeypadModifier) && isEnter); |
334 | } |
335 | |
336 | void StyleSheetEditorDialog::keyPressEvent(QKeyEvent *e) |
337 | { |
338 | // As long as the find widget is visible, suppress the default button |
339 | // behavior (close on Enter) of QDialog. |
340 | if (!(m_findWidget->isVisible() && isEnter(e))) |
341 | QDialog::keyPressEvent(e); |
342 | } |
343 | |
344 | QDialogButtonBox * StyleSheetEditorDialog::buttonBox() const |
345 | { |
346 | return m_buttonBox; |
347 | } |
348 | |
349 | QString StyleSheetEditorDialog::text() const |
350 | { |
351 | return m_editor->toPlainText(); |
352 | } |
353 | |
354 | void StyleSheetEditorDialog::setText(const QString &t) |
355 | { |
356 | m_editor->setText(t); |
357 | } |
358 | |
359 | bool StyleSheetEditorDialog::isStyleSheetValid(const QString &styleSheet) |
360 | { |
361 | QCss::Parser parser(styleSheet); |
362 | QCss::StyleSheet sheet; |
363 | if (parser.parse(styleSheet: &sheet)) |
364 | return true; |
365 | QString fullSheet = QStringLiteral("* { " ); |
366 | fullSheet += styleSheet; |
367 | fullSheet += QLatin1Char('}'); |
368 | QCss::Parser parser2(fullSheet); |
369 | return parser2.parse(styleSheet: &sheet); |
370 | } |
371 | |
372 | void StyleSheetEditorDialog::validateStyleSheet() |
373 | { |
374 | const bool valid = isStyleSheetValid(styleSheet: m_editor->toPlainText()); |
375 | setOkButtonEnabled(valid); |
376 | if (valid) { |
377 | m_validityLabel->setText(tr(s: "Valid Style Sheet" )); |
378 | m_validityLabel->setStyleSheet(QStringLiteral("color: green" )); |
379 | } else { |
380 | m_validityLabel->setText(tr(s: "Invalid Style Sheet" )); |
381 | m_validityLabel->setStyleSheet(QStringLiteral("color: red" )); |
382 | } |
383 | } |
384 | |
385 | // --- StyleSheetPropertyEditorDialog |
386 | StyleSheetPropertyEditorDialog::StyleSheetPropertyEditorDialog(QWidget *parent, |
387 | QDesignerFormWindowInterface *fw, |
388 | QWidget *widget): |
389 | StyleSheetEditorDialog(fw->core(), parent), |
390 | m_fw(fw), |
391 | m_widget(widget) |
392 | { |
393 | Q_ASSERT(m_fw != nullptr); |
394 | |
395 | QPushButton *apply = buttonBox()->addButton(button: QDialogButtonBox::Apply); |
396 | QObject::connect(sender: apply, signal: &QAbstractButton::clicked, |
397 | receiver: this, slot: &StyleSheetPropertyEditorDialog::applyStyleSheet); |
398 | QObject::connect(sender: buttonBox(), signal: &QDialogButtonBox::accepted, |
399 | receiver: this, slot: &StyleSheetPropertyEditorDialog::applyStyleSheet); |
400 | |
401 | QDesignerPropertySheetExtension *sheet = |
402 | qt_extension<QDesignerPropertySheetExtension*>(manager: m_fw->core()->extensionManager(), object: m_widget); |
403 | Q_ASSERT(sheet != nullptr); |
404 | const int index = sheet->indexOf(name: QLatin1String(styleSheetProperty)); |
405 | const PropertySheetStringValue value = qvariant_cast<PropertySheetStringValue>(v: sheet->property(index)); |
406 | setText(value.value()); |
407 | } |
408 | |
409 | void StyleSheetPropertyEditorDialog::applyStyleSheet() |
410 | { |
411 | const PropertySheetStringValue value(text(), false); |
412 | m_fw->cursor()->setWidgetProperty(widget: m_widget, name: QLatin1String(styleSheetProperty), value: QVariant::fromValue(value)); |
413 | } |
414 | |
415 | } // namespace qdesigner_internal |
416 | |
417 | QT_END_NAMESPACE |
418 | |