// Copyright (C) 2018 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "aspects.h" #include "algorithm.h" #include "checkablemessagebox.h" #include "environment.h" #include "fancylineedit.h" #include "guard.h" #include "layoutbuilder.h" #include "macroexpander.h" #include "passworddialog.h" #include "pathchooser.h" #include "pathlisteditor.h" #include "qtcassert.h" #include "qtcolorbutton.h" #include "qtcsettings.h" #include "qtcwidgets.h" #include "stylehelper.h" #include "utilsicons.h" #include "utilstr.h" #include "variablechooser.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Layouting; namespace Utils { static QtcSettings *theSettings = nullptr; void BaseAspect::setQtcSettings(QtcSettings *settings) { theSettings = settings; } QtcSettings *BaseAspect::qtcSettings() { return theSettings; } BaseAspect::Changes::Changes() { memset(this, 0, sizeof(*this)); } class Internal::BaseAspectPrivate { public: explicit BaseAspectPrivate(AspectContainer *container) : m_container(container) {} MacroExpander *macroExpander() { if (!m_expander) { m_expander = std::make_unique(); m_expander->setDisplayName("Variables"); if (m_container) m_expander->registerSubProvider([this] { return m_container->macroExpander(); }); } return m_expander.get(); } void setContainer(AspectContainer *container) { m_container = container; if (m_expander) m_expander->registerSubProvider([this] { return m_container->macroExpander(); }); } Id m_id; std::function m_toSettings; std::function m_fromSettings; QString m_displayName; Key m_settingsKey; // Name of data in settings. QString m_tooltip; QString m_labelText; QPixmap m_labelPixmap; QIcon m_icon; QPointer m_action; // Owned by us. AspectContainer *m_container = nullptr; // Not owned by us. bool m_visible = true; bool m_readOnly = false; bool m_autoApply = true; bool m_saveAlways = false; // if true, also empty keys will be written QPointer m_enabler; bool m_enabled = true; int m_spanX = 1; int m_spanY = 1; BaseAspect::ConfigWidgetCreator m_configWidgetCreator; QList> m_subWidgets; BaseAspect::DataCreator m_dataCreator; BaseAspect::DataCloner m_dataCloner; QList m_dataExtractors; QUndoStack *m_undoStack = nullptr; private: std::unique_ptr m_expander; }; /*! \class Utils::BaseAspect \inmodule QtCreator \brief The \c BaseAspect class provides a common base for classes implementing aspects. An \e aspect is a hunk of data like a property or collection of related properties of some object, together with a description of its behavior for common operations like visualizing or persisting. Simple aspects are, for example, a boolean property represented by a QCheckBox in the user interface, or a string property represented by a PathChooser, for selecting directories in the filesystem. While aspects implementations usually can visualize and persist their data, or use an ID, neither of these is mandatory. The derived classes can implement addToLayout() to create a UI. Implement \c guiToBuffer() and \c bufferToGui() to synchronize the UI with the internal data. */ /*! \enum Utils::BaseAspect::Announcement Whether to emit a signal when a value changes. \value DoEmit Emit a signal. \value BeQuiet Don't emit a signal. */ /*! Constructs a base aspect. If \a container is non-null, the aspect is made known to the container. */ BaseAspect::BaseAspect(AspectContainer *container) : d(new Internal::BaseAspectPrivate(container)) { if (container) container->registerAspect(this); addDataExtractor(this, &BaseAspect::variantValue, &Data::value); } /*! Destructs a BaseAspect. */ BaseAspect::~BaseAspect() { delete d->m_action; } Id BaseAspect::id() const { return d->m_id; } void BaseAspect::setId(Id id) { d->m_id = id; } QVariant BaseAspect::volatileVariantValue() const { return {}; } QVariant BaseAspect::variantValue() const { return {}; } /*! Sets \a value. If \a howToAnnounce is set to \c DoEmit, emits the \c valueChanged signal. Prefer the typed \c setValue() of the derived classes. */ void BaseAspect::setVariantValue(const QVariant &value, Announcement howToAnnounce) { Q_UNUSED(value) Q_UNUSED(howToAnnounce) QTC_CHECK(false); } void BaseAspect::setDefaultVariantValue(const QVariant &value) { Q_UNUSED(value) QTC_CHECK(false); } bool BaseAspect::isDefaultValue() const { return defaultVariantValue() == variantValue(); } QVariant BaseAspect::defaultVariantValue() const { return {}; } /*! \class Utils::TypedAspect \inheaderfile utils/aspects.h \inmodule QtCreator \brief The \c TypedAspect class is a helper class for implementing a simple aspect. A typed aspect contains a single piece of data that is of the type \c ValueType. */ /*! \fn template void Utils::TypedAspect::setDefaultValue(const ValueType &value) Sets a default \a value and the current value for this aspect. \note The current value will be set silently to the same value. It is reasonable to only set default values in the setup phase of the aspect. Default values will not be stored in settings. */ void BaseAspect::setDisplayName(const QString &displayName) { d->m_displayName = displayName; } bool BaseAspect::isVisible() const { return d->m_visible; } /*! Shows or hides the visual representation of this aspect depending on the value of \a visible. By default, it is visible. */ void BaseAspect::setVisible(bool visible) { d->m_visible = visible; for (QWidget *w : std::as_const(d->m_subWidgets)) { QTC_ASSERT(w, continue); // This may happen during layout building. Explicit setting visibility here // may create a show a toplevel widget for a moment until it is parented // to some non-shown widget. if (!visible || w->parentWidget()) w->setVisible(visible); } } QLabel *BaseAspect::createLabel() { if (d->m_labelText.isEmpty() && d->m_labelPixmap.isNull()) return nullptr; auto label = new QLabel(d->m_labelText); label->setTextInteractionFlags(label->textInteractionFlags() | Qt::TextSelectableByMouse); connect(label, &QLabel::linkActivated, this, [this](const QString &link) { emit labelLinkActivated(link); }); if (!d->m_labelPixmap.isNull()) label->setPixmap(d->m_labelPixmap); registerSubWidget(label); connect(this, &BaseAspect::labelTextChanged, label, [label, this] { label->setText(d->m_labelText); }); connect(this, &BaseAspect::labelPixmapChanged, label, [label, this] { label->setPixmap(d->m_labelPixmap); }); return label; } void BaseAspect::addLabeledItem(Layout &parent, QWidget *widget) { if (QLabel *l = createLabel()) { l->setBuddy(widget); parent.addItem(l); parent.addItem(Span(std::max(d->m_spanX - 1, 1), widget)); } else { parent.addItem(widget); } } void BaseAspect::addLabeledItems(Layouting::Layout &parent, const QList &widgets) { if (QLabel *l = createLabel()) parent.addItem(l); for (auto widget : widgets) parent.addItem(widget); } /*! Sets \a labelText as text for the separate label in the visual representation of this aspect. */ void BaseAspect::setLabelText(const QString &labelText) { d->m_labelText = labelText; emit labelTextChanged(); } /*! Sets \a labelPixmap as pixmap for the separate label in the visual representation of this aspect. */ void BaseAspect::setLabelPixmap(const QPixmap &labelPixmap) { d->m_labelPixmap = labelPixmap; emit labelPixmapChanged(); } void BaseAspect::setIcon(const QIcon &icon) { d->m_icon = icon; if (d->m_action) d->m_action->setIcon(icon); } QIcon BaseAspect::icon() const { return d->m_icon; } /*! Returns the current text for the separate label in the visual representation of this aspect. */ QString BaseAspect::labelText() const { return d->m_labelText; } QString BaseAspect::toolTip() const { return d->m_tooltip; } /*! Sets \a tooltip as tool tip for the visual representation of this aspect. */ void BaseAspect::setToolTip(const QString &tooltip) { d->m_tooltip = tooltip; for (QWidget *w : std::as_const(d->m_subWidgets)) { QTC_ASSERT(w, continue); w->setToolTip(tooltip); } } void BaseAspect::setUndoStack(QUndoStack *undoStack) { d->m_undoStack = undoStack; } QUndoStack *BaseAspect::undoStack() const { return d->m_undoStack; } bool BaseAspect::isEnabled() const { if (d->m_enabler) return d->m_enabler->isEnabled() && d->m_enabler->volatileValue(); return d->m_enabled; } void BaseAspect::setEnabled(bool enabled) { for (QWidget *w : std::as_const(d->m_subWidgets)) { QTC_ASSERT(w, continue); w->setEnabled(enabled); } if (enabled == d->m_enabled) return; d->m_enabled = enabled; emit enabledChanged(); } /*! Makes the enabled state of this aspect depend on the checked state of \a checker. */ void BaseAspect::setEnabler(BoolAspect *checker) { QTC_ASSERT(checker, return); d->m_enabler = checker; auto update = [this] { BaseAspect::setEnabled(isEnabled()); }; connect(checker, &BoolAspect::volatileValueChanged, this, update); connect(checker, &BoolAspect::changed, this, update); connect(checker, &BaseAspect::enabledChanged, this, update); update(); } bool BaseAspect::isReadOnly() const { return d->m_readOnly; } void BaseAspect::setReadOnly(bool readOnly) { d->m_readOnly = readOnly; for (QWidget *w : std::as_const(d->m_subWidgets)) { QTC_ASSERT(w, continue); if (auto lineEdit = qobject_cast(w)) lineEdit->setReadOnly(readOnly); else if (auto textEdit = qobject_cast(w)) textEdit->setReadOnly(readOnly); else if (auto pathChooser = qobject_cast(w)) pathChooser->setReadOnly(readOnly); } } void BaseAspect::setSpan(int x, int y) { d->m_spanX = x; d->m_spanY = y; } bool BaseAspect::isSaveAlways() const { return d->m_saveAlways; } void BaseAspect::setSaveAlways(bool saveAlways) { d->m_saveAlways = saveAlways; } bool BaseAspect::isAutoApply() const { return d->m_autoApply; } /*! Sets auto-apply mode. When auto-apply mode is \a on, user interaction to this aspect's widget will not modify the \c value of the aspect until \c apply() is called programmatically. \sa setSettingsKey() */ void BaseAspect::setAutoApply(bool on) { d->m_autoApply = on; } /*! \internal */ void BaseAspect::setConfigWidgetCreator(const ConfigWidgetCreator &configWidgetCreator) { d->m_configWidgetCreator = configWidgetCreator; } /*! Returns the key to be used when accessing the settings. \sa setSettingsKey() */ Key BaseAspect::settingsKey() const { return d->m_settingsKey; } /*! Sets the \a key to be used when accessing the settings. \sa settingsKey() */ void BaseAspect::setSettingsKey(const Key &key) { d->m_settingsKey = key; } /*! Sets the \a key and \a group to be used when accessing the settings. \sa settingsKey() */ void BaseAspect::setSettingsKey(const Key &group, const Key &key) { d->m_settingsKey = group + "/" + key; } /*! Immediately writes the value of this aspect into its specified settings, taking a potential container's settings group specification into account. \note This is expensive, so it should only be used with good reason. */ void BaseAspect::writeToSettingsImmediatly() const { QStringList groups; if (d->m_container) groups = d->m_container->settingsGroups(); const SettingsGroupNester nester(groups); writeSettings(); } /*! Returns the string that should be used when this action appears in menus or other places that are typically used with Book style capitalization. If no display name is set, the label text will be used as fallback. */ QString BaseAspect::displayName() const { return d->m_displayName.isEmpty() ? d->m_labelText : d->m_displayName; } /*! \internal */ QWidget *BaseAspect::createConfigWidget() const { return d->m_configWidgetCreator ? d->m_configWidgetCreator() : nullptr; } QAction *BaseAspect::action() { if (!d->m_action) { d->m_action = new QAction(labelText()); d->m_action->setIcon(d->m_icon); } return d->m_action; } AspectContainer *BaseAspect::container() const { return d->m_container; } /*! Adds the visual representation of this aspect to the layout with the specified \a parent using a layout builder. */ void BaseAspect::addToLayoutImpl(Layout &) { } void addToLayout(Layouting::Layout *iface, const BaseAspect &aspect) { aspect.addToLayout(*iface); } void addToLayout(Layouting::Layout *item, const BaseAspect *aspect) { aspect->addToLayout(*item); } /*! Updates this aspect's value from user-initiated changes in the widget. Emits changed() if the value changed. */ void BaseAspect::apply() { // We assume m_buffer to reflect current gui state as invariant after signalling settled down. // It's an aspect (-subclass) implementation problem if this doesn't hold. Fix it up and bark. QTC_CHECK(!guiToBuffer()); if (!bufferToInternal()) // Nothing to do. return; Changes changes; changes.internalFromBuffer = true; announceChanges(changes); } /*! Discard user changes in the widget and restore widget contents from aspect's value. This has only an effect if \c isAutoApply is false. */ void BaseAspect::cancel() { internalToBuffer(); bufferToGui(); } void BaseAspect::finish() { // No qDeleteAll() possible as long as the connect in registerSubWidget() exist. while (d->m_subWidgets.size()) delete d->m_subWidgets.takeLast(); } bool BaseAspect::hasAction() const { return d->m_action != nullptr; } void BaseAspect::announceChanges(Changes changes, Announcement howToAnnounce) { if (howToAnnounce == BeQuiet) return; if (changes.bufferFromInternal || changes.bufferFromOutside || changes.bufferFromGui) emit volatileValueChanged(); if (changes.internalFromOutside || changes.internalFromBuffer) { emit changed(); if (hasAction()) emit action()->triggered(variantValue().toBool()); } } bool BaseAspect::isDirty() { return false; } void BaseAspect::registerSubWidget(QWidget *widget) { d->m_subWidgets.append(widget); // FIXME: This interferes with qDeleteAll() in finish() and destructor, // it would not be needed when all users actually deleted their subwidgets, // e.g. the SettingsPage::finish() base implementation, but this still // leaves the cases where no such base functionality is available, e.g. // in the run/build config aspects. connect(widget, &QObject::destroyed, this, [this, widget] { d->m_subWidgets.removeAll(widget); }); widget->setEnabled(isEnabled()); widget->setToolTip(d->m_tooltip); // Visible is on by default. Not setting it explicitly avoid popping // it up when the parent is not set yet, the normal case. if (!d->m_visible) widget->setVisible(d->m_visible); } void BaseAspect::forEachSubWidget(const std::function &func) { for (const QPointer &w : std::as_const(d->m_subWidgets)) func(w); } void BaseAspect::setContainer(AspectContainer *container) { d->setContainer(container); } void BaseAspect::saveToMap(Store &data, const QVariant &value, const QVariant &defaultValue, const Key &key) const { if (key.isEmpty() && !d->m_saveAlways) return; if (value == defaultValue) data.remove(key); else data.insert(key, value); } bool BaseAspect::skipSave() const { return settingsKey().isEmpty() && !d->m_saveAlways; } /*! Retrieves the internal value of this BaseAspect from the Store \a map. */ void BaseAspect::fromMap(const Store &map) { if (skipSave()) return; const QVariant val = map.value(settingsKey(), toSettingsValue(defaultVariantValue())); setVariantValue(fromSettingsValue(val), BeQuiet); } /*! Stores the internal value of this BaseAspect into the Store \a map. */ void BaseAspect::toMap(Store &map) const { saveToMap(map, toSettingsValue(variantValue()), toSettingsValue(defaultVariantValue()), settingsKey()); } void BaseAspect::volatileToMap(Store &map) const { saveToMap(map, toSettingsValue(volatileVariantValue()), toSettingsValue(defaultVariantValue()), settingsKey()); } void BaseAspect::addToLayout(Layouting::Layout &parent) const { const_cast(this)->addToLayoutImpl(parent); } void BaseAspect::readSettings() { if (skipSave()) return; QTC_ASSERT(theSettings, return); const QVariant val = theSettings->value(settingsKey()); setVariantValue(val.isValid() ? fromSettingsValue(val) : defaultVariantValue(), BeQuiet); } void BaseAspect::writeSettings() const { if (skipSave()) return; QTC_ASSERT(theSettings, return); theSettings->setValueWithDefault(settingsKey(), toSettingsValue(variantValue()), toSettingsValue(defaultVariantValue())); } void BaseAspect::setFromSettingsTransformation(const SavedValueTransformation &transform) { d->m_fromSettings = transform; } void BaseAspect::setToSettingsTransformation(const SavedValueTransformation &transform) { d->m_toSettings = transform; } QVariant BaseAspect::toSettingsValue(const QVariant &val) const { return d->m_toSettings ? d->m_toSettings(val) : val; } QVariant BaseAspect::fromSettingsValue(const QVariant &val) const { return d->m_fromSettings ? d->m_fromSettings(val) : val; } void BaseAspect::setMacroExpander(MacroExpander *expander) { d->macroExpander()->clearSubProviders(); if (expander) d->macroExpander()->registerSubProvider([expander] { return expander; }); } MacroExpander *BaseAspect::macroExpander() const { return d->macroExpander(); } void BaseAspect::addOnChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::changed, guard, callback); } void BaseAspect::addOnVolatileValueChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::volatileValueChanged, guard, callback); } void BaseAspect::addOnCheckedChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::checkedChanged, guard, callback); } void BaseAspect::addOnEnabledChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::enabledChanged, guard, callback); } void BaseAspect::addOnLabelTextChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::labelTextChanged, guard, callback); } void BaseAspect::addOnLabelPixmapChanged(QObject *guard, const Callback &callback) { connect(this, &BaseAspect::labelPixmapChanged, guard, callback); } void BaseAspect::addMacroExpansion(QWidget *w) { const auto chooser = new VariableChooser(w); chooser->addSupportedWidget(w); chooser->addMacroExpanderProvider([this] { return d->macroExpander(); }); if (auto pathChooser = qobject_cast(w)) pathChooser->setMacroExpander(d->macroExpander()); } namespace Internal { class BoolAspectPrivate { public: BoolAspect::LabelPlacement m_labelPlacement = BoolAspect::LabelPlacement::AtCheckBox; UndoableValue m_undoable; }; class ToggleAspectPrivate { public: struct Data { QIcon icon; QString tooltip; QString text; } on, off; }; class ColorAspectPrivate { public: QPointer m_colorButton; // Owned by configuration widget QSize m_size; }; class SelectionAspectPrivate { public: ~SelectionAspectPrivate() { delete m_buttonGroup; } SelectionAspect::DisplayStyle m_displayStyle = SelectionAspect::DisplayStyle::RadioButtons; QList m_options; // These are all owned by the configuration widget. QList> m_buttons; QPointer m_comboBox; QPointer m_buttonGroup; }; class MultiSelectionAspectPrivate { public: explicit MultiSelectionAspectPrivate(MultiSelectionAspect *q) : q(q) {} bool setValueSelectedHelper(const QString &value, bool on); MultiSelectionAspect *q; QStringList m_allValues; MultiSelectionAspect::DisplayStyle m_displayStyle = MultiSelectionAspect::DisplayStyle::ListView; // These are all owned by the configuration widget. QPointer m_listView; }; class CheckableAspectImplementation { public: void fromMap(const Store &map) { if (m_checked) m_checked->fromMap(map); } void toMap(Store &map) { if (m_checked) m_checked->toMap(map); } void volatileToMap(Store &map) { if (m_checked) m_checked->volatileToMap(map); } template void updateWidgetFromCheckStatus(BaseAspect *aspect, Widget *w) { const bool enabled = !m_checked || m_checked->volatileValue(); if (m_uncheckedSemantics == UncheckedSemantics::Disabled) w->setEnabled(enabled && aspect->isEnabled()); else w->setReadOnly(!enabled || aspect->isReadOnly()); } void setUncheckedSemantics(UncheckedSemantics semantics) { m_uncheckedSemantics = semantics; } bool isChecked() const { QTC_ASSERT(m_checked, return false); return m_checked->value(); } void setChecked(bool checked) { QTC_ASSERT(m_checked, return); m_checked->setValue(checked); } void makeCheckable(CheckBoxPlacement checkBoxPlacement, const QString &checkerLabel, const Key &checkerKey, BaseAspect *aspect) { QTC_ASSERT(!m_checked, return); m_checkBoxPlacement = checkBoxPlacement; m_checked.reset(new BoolAspect); m_checked->setLabel(checkerLabel, checkBoxPlacement == CheckBoxPlacement::Top ? BoolAspect::LabelPlacement::InExtraLabel : BoolAspect::LabelPlacement::AtCheckBox); m_checked->setSettingsKey(checkerKey); m_checked->addOnChanged(aspect, [aspect] { // FIXME: Check. aspect->internalToBuffer(); aspect->bufferToGui(); emit aspect->changed(); aspect->checkedChanged(); }); m_checked->addOnVolatileValueChanged(aspect, [aspect] { // FIXME: Check. aspect->internalToBuffer(); aspect->bufferToGui(); aspect->checkedChanged(); }); aspect->internalToBuffer(); aspect->bufferToGui(); } void addToLayoutFirst(Layout &parent) { if (m_checked && m_checkBoxPlacement == CheckBoxPlacement::Top) { m_checked->addToLayoutImpl(parent); parent.flush(); } } void addToLayoutLast(Layout &parent) { if (m_checked && m_checkBoxPlacement == CheckBoxPlacement::Right) m_checked->addToLayoutImpl(parent); } CheckBoxPlacement m_checkBoxPlacement = CheckBoxPlacement::Right; UncheckedSemantics m_uncheckedSemantics = UncheckedSemantics::Disabled; std::unique_ptr m_checked; }; class StringAspectPrivate { public: StringAspect::DisplayStyle m_displayStyle = StringAspect::LabelDisplay; std::function m_displayFilter; Qt::TextElideMode m_elideMode = Qt::ElideNone; QString m_placeHolderText; Key m_historyCompleterKey; StringAspect::ValueAcceptor m_valueAcceptor; std::optional m_validator; std::function m_validatorFactory; CheckableAspectImplementation m_checkerImpl; bool m_undoRedoEnabled = true; bool m_acceptRichText = false; bool m_showToolTipOnLabel = false; bool m_useResetButton = false; bool m_autoApplyOnEditingFinished = false; bool m_validatePlaceHolder = false; FilePath m_rightSideIconPath; int m_minimumHeight = 0; QPointer m_completer; UndoableValue undoable; }; class IntegerAspectPrivate { public: std::optional m_minimumValue; std::optional m_maximumValue; int m_displayIntegerBase = 10; qint64 m_displayScaleFactor = 1; QString m_prefix; QString m_suffix; QString m_specialValueText; int m_singleStep = 1; QPointer m_spinBox; // Owned by configuration widget }; class DoubleAspectPrivate { public: std::optional m_minimumValue; std::optional m_maximumValue; QString m_prefix; QString m_suffix; QString m_specialValueText; double m_singleStep = 1; QPointer m_spinBox; // Owned by configuration widget }; class StringListAspectPrivate { public: UndoableValue undoable; bool allowAdding{true}; bool allowRemoving{true}; bool allowEditing{true}; }; class FilePathListAspectPrivate { public: UndoableValue undoable; QString placeHolderText; }; class TextDisplayPrivate { public: QString m_message; InfoLabel::InfoType m_type; QPointer m_label; }; } // Internal /*! \enum Utils::StringAspect::DisplayStyle \inmodule QtCreator The DisplayStyle enum describes the main visual characteristics of a string aspect. \value LabelDisplay Based on QLabel, used for text that cannot be changed by the user in this place, for example names of executables that are defined in the build system. \value LineEditDisplay Based on QLineEdit, used for user-editable strings that usually fit on a line. \value TextEditDisplay Based on QTextEdit, used for user-editable strings that often do not fit on a line. \value PasswordLineEditDisplay Based on QLineEdit, used for password strings \sa Utils::PathChooser */ /*! \class Utils::StringAspect \inmodule QtCreator \brief A string aspect is a string-like property of some object, together with a description of its behavior for common operations like visualizing or persisting. String aspects can represent for example a parameter for an external commands, paths in a file system, or simply strings. The string can be displayed using a QLabel, QLineEdit, QTextEdit or Utils::PathChooser. The visual representation often contains a label in front of the display of the actual value. */ /*! Constructs the string aspect \a container. */ StringAspect::StringAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::StringAspectPrivate) { setSpan(2, 1); // Default: Label + something } /*! \internal */ StringAspect::~StringAspect() = default; /*! \internal */ void StringAspect::setValueAcceptor(StringAspect::ValueAcceptor &&acceptor) { d->m_valueAcceptor = std::move(acceptor); } /*! \reimp */ void StringAspect::fromMap(const Store &map) { if (!skipSave()) setValue(map.value(settingsKey(), defaultValue()).toString(), BeQuiet); d->m_checkerImpl.fromMap(map); } /*! \reimp */ void StringAspect::toMap(Store &map) const { saveToMap(map, value(), defaultValue(), settingsKey()); d->m_checkerImpl.toMap(map); } void StringAspect::volatileToMap(Store &map) const { saveToMap(map, volatileValue(), defaultValue(), settingsKey()); d->m_checkerImpl.volatileToMap(map); } /*! \internal */ void StringAspect::setShowToolTipOnLabel(bool show) { d->m_showToolTipOnLabel = show; bufferToGui(); } /*! Sets a \a displayFilter for fine-tuning the visual appearance of the value of this string aspect. */ void StringAspect::setDisplayFilter(const std::function &displayFilter) { d->m_displayFilter = displayFilter; } /*! Selects the main display characteristics of the aspect according to \a displayStyle. \note Not all StringAspect features are available with all display styles. \sa Utils::StringAspect::DisplayStyle */ void StringAspect::setDisplayStyle(DisplayStyle displayStyle) { d->m_displayStyle = displayStyle; } /*! Sets \a placeHolderText as place holder for line and text displays. */ void StringAspect::setPlaceHolderText(const QString &placeHolderText) { if (d->m_placeHolderText == placeHolderText) return; d->m_placeHolderText = placeHolderText; emit placeholderTextChanged(placeHolderText); } /*! Sets \a elideMode as label elide mode. */ void StringAspect::setElideMode(Qt::TextElideMode elideMode) { if (d->m_elideMode == elideMode) return; d->m_elideMode = elideMode; emit elideModeChanged(elideMode); } /*! Sets \a historyCompleterKey as key for the history completer settings for line edits and path chooser displays. \sa Utils::PathChooser::setExpectedKind() */ void StringAspect::setHistoryCompleter(const Key &historyCompleterKey) { d->m_historyCompleterKey = historyCompleterKey; emit historyCompleterKeyChanged(historyCompleterKey); } void StringAspect::setAcceptRichText(bool acceptRichText) { d->m_acceptRichText = acceptRichText; emit acceptRichTextChanged(acceptRichText); } void StringAspect::setUseResetButton() { d->m_useResetButton = true; } void StringAspect::setValidationFunction(const FancyLineEdit::ValidationFunction &validator) { d->m_validator = validator; emit validationFunctionChanged(validator); } void StringAspect::setValidatorFactory( const std::function &validatorFactory) { d->m_validatorFactory = validatorFactory; } void StringAspect::setAutoApplyOnEditingFinished(bool applyOnEditingFinished) { d->m_autoApplyOnEditingFinished = applyOnEditingFinished; } void StringAspect::addToLayoutImpl(Layout &parent) { d->m_checkerImpl.addToLayoutFirst(parent); const QString displayedString = d->m_displayFilter ? d->m_displayFilter(volatileValue()) : volatileValue(); switch (d->m_displayStyle) { case PasswordLineEditDisplay: case LineEditDisplay: { auto lineEditDisplay = createSubWidget(); addMacroExpansion(lineEditDisplay); lineEditDisplay->setPlaceholderText(d->m_placeHolderText); lineEditDisplay->setMinimumHeight(d->m_minimumHeight); if (d->m_completer) lineEditDisplay->setSpecialCompleter(d->m_completer); if (!d->m_rightSideIconPath.isEmpty()) { QIcon icon(d->m_rightSideIconPath.toFSPathString()); QTC_CHECK(!icon.isNull()); lineEditDisplay->setButtonIcon(FancyLineEdit::Right, icon); lineEditDisplay->setButtonVisible(FancyLineEdit::Right, true); connect(lineEditDisplay, &FancyLineEdit::rightButtonClicked, this, &StringAspect::rightSideIconClicked); } if (!d->m_historyCompleterKey.isEmpty()) lineEditDisplay->setHistoryCompleter(d->m_historyCompleterKey); connect(this, &StringAspect::historyCompleterKeyChanged, lineEditDisplay, [lineEditDisplay](const Key &historyCompleterKey) { lineEditDisplay->setHistoryCompleter(historyCompleterKey); }); connect(this, &StringAspect::placeholderTextChanged, lineEditDisplay, &FancyLineEdit::setPlaceholderText); if (d->m_validator) lineEditDisplay->setValidationFunction(*d->m_validator); else if (d->m_validatorFactory) lineEditDisplay->setValidator(d->m_validatorFactory(lineEditDisplay)); lineEditDisplay->setTextKeepingActiveCursor(displayedString); lineEditDisplay->setReadOnly(isReadOnly()); lineEditDisplay->setValidatePlaceHolder(d->m_validatePlaceHolder); d->m_checkerImpl.updateWidgetFromCheckStatus(this, lineEditDisplay); if (d->m_checkerImpl.m_checked.get()) { connect(d->m_checkerImpl.m_checked.get(), &BoolAspect::volatileValueChanged, lineEditDisplay, [this, lineEditDisplay] { d->m_checkerImpl.updateWidgetFromCheckStatus(this, lineEditDisplay); }); } addLabeledItem(parent, lineEditDisplay); if (d->m_useResetButton) { auto resetButton = createSubWidget(Tr::tr("Reset")); resetButton->setEnabled(lineEditDisplay->text() != defaultValue()); connect(resetButton, &QPushButton::clicked, lineEditDisplay, [this, lineEditDisplay] { lineEditDisplay->setText(defaultValue()); }); connect(lineEditDisplay, &QLineEdit::textChanged, resetButton, [this, lineEditDisplay, resetButton] { resetButton->setEnabled(lineEditDisplay->text() != defaultValue()); }); parent.addItem(resetButton); } connect(lineEditDisplay, &FancyLineEdit::validChanged, this, &StringAspect::validChanged); bufferToGui(); if (isAutoApply() && d->m_autoApplyOnEditingFinished) { connect(lineEditDisplay, &FancyLineEdit::editingFinished, this, [this, lineEditDisplay] { if (lineEditDisplay->text() != d->undoable.get()) { d->undoable.set(undoStack(), lineEditDisplay->text()); handleGuiChanged(); } }); } else { connect(lineEditDisplay, &QLineEdit::textChanged, this, [this, lineEditDisplay] { d->undoable.set(undoStack(), lineEditDisplay->text()); handleGuiChanged(); }); } if (d->m_displayStyle == PasswordLineEditDisplay) { auto showPasswordButton = createSubWidget(); lineEditDisplay->setEchoMode(QLineEdit::PasswordEchoOnEdit); parent.addItem(showPasswordButton); connect(showPasswordButton, &ShowPasswordButton::toggled, lineEditDisplay, [showPasswordButton, lineEditDisplay] { lineEditDisplay->setEchoMode(showPasswordButton->isChecked() ? QLineEdit::Normal : QLineEdit::PasswordEchoOnEdit); }); } connect(&d->undoable.m_signal, &UndoSignaller::changed, lineEditDisplay, [this, lineEditDisplay] { if (lineEditDisplay->text() != d->undoable.get()) lineEditDisplay->setTextKeepingActiveCursor(d->undoable.get()); lineEditDisplay->validate(); }); break; } case TextEditDisplay: { auto textEditDisplay = createSubWidget(); addMacroExpansion(textEditDisplay); textEditDisplay->setPlaceholderText(d->m_placeHolderText); textEditDisplay->setUndoRedoEnabled(false); textEditDisplay->setAcceptRichText(d->m_acceptRichText); textEditDisplay->setTextInteractionFlags(Qt::TextEditorInteraction); textEditDisplay->setText(displayedString); textEditDisplay->setReadOnly(isReadOnly()); d->m_checkerImpl.updateWidgetFromCheckStatus(this, textEditDisplay); if (d->m_checkerImpl.m_checked) { connect(d->m_checkerImpl.m_checked.get(), &BoolAspect::volatileValueChanged, textEditDisplay, [this, textEditDisplay] { d->m_checkerImpl.updateWidgetFromCheckStatus(this, textEditDisplay); }); } addLabeledItem(parent, textEditDisplay); bufferToGui(); connect(this, &StringAspect::acceptRichTextChanged, textEditDisplay, &QTextEdit::setAcceptRichText); connect(this, &StringAspect::placeholderTextChanged, textEditDisplay, &QTextEdit::setPlaceholderText); connect(textEditDisplay, &QTextEdit::textChanged, this, [this, textEditDisplay] { if (textEditDisplay->toPlainText() != d->undoable.get()) { d->undoable.set(undoStack(), textEditDisplay->toPlainText()); handleGuiChanged(); } }); connect(&d->undoable.m_signal, &UndoSignaller::changed, textEditDisplay, [this, textEditDisplay] { if (textEditDisplay->toPlainText() != d->undoable.get()) textEditDisplay->setText(d->undoable.get()); }); break; } case LabelDisplay: { auto labelDisplay = createSubWidget(); labelDisplay->setElideMode(d->m_elideMode); labelDisplay->setTextInteractionFlags(Qt::TextSelectableByMouse); labelDisplay->setText(displayedString); labelDisplay->setToolTip(d->m_showToolTipOnLabel ? displayedString : toolTip()); connect(this, &StringAspect::elideModeChanged, labelDisplay, &ElidingLabel::setElideMode); addLabeledItem(parent, labelDisplay); connect(&d->undoable.m_signal, &UndoSignaller::changed, labelDisplay, [this, labelDisplay] { labelDisplay->setText(d->undoable.get()); labelDisplay->setToolTip(d->m_showToolTipOnLabel ? d->undoable.get() : toolTip()); }); break; } } d->m_checkerImpl.addToLayoutLast(parent); } QString StringAspect::expandedValue() const { if (!m_internal.isEmpty()) { if (auto expander = macroExpander()) return expander->expand(m_internal); } return m_internal; } bool StringAspect::guiToBuffer() { return updateStorage(m_buffer, d->undoable.get()); } bool StringAspect::bufferToInternal() { if (d->m_valueAcceptor) { if (const std::optional tmp = d->m_valueAcceptor(m_internal, m_buffer)) return updateStorage(m_internal, *tmp); return false; } return updateStorage(m_internal, m_buffer); } bool StringAspect::internalToBuffer() { const QString val = d->m_displayFilter ? d->m_displayFilter(m_internal) : m_internal; return updateStorage(m_buffer, val); } void StringAspect::bufferToGui() { d->undoable.setWithoutUndo(m_buffer); } /*! Adds a check box with a \a checkerLabel according to \a checkBoxPlacement to the line edit. The state of the check box is made persistent when using a non-emtpy \a checkerKey. */ void StringAspect::makeCheckable(CheckBoxPlacement checkBoxPlacement, const QString &checkerLabel, const Key &checkerKey) { d->m_checkerImpl.makeCheckable(checkBoxPlacement, checkerLabel, checkerKey, this); } bool StringAspect::isChecked() const { return d->m_checkerImpl.isChecked(); } void StringAspect::setChecked(bool checked) { return d->m_checkerImpl.setChecked(checked); } void StringAspect::addOnRightSideIconClicked(QObject *guard, const std::function &callback) { connect(this, &StringAspect::rightSideIconClicked, guard, callback); } void StringAspect::setMinimumHeight(int height) { d->m_minimumHeight = height; } void StringAspect::setCompleter(QCompleter *completer) { d->m_completer = completer; } void StringAspect::setRightSideIconPath(const FilePath &path) { d->m_rightSideIconPath = path; } /*! \class Utils::FilePathAspect \inmodule QtCreator \brief A file path aspect is shallow wrapper around a Utils::StringAspect that represents a file in the file system. It is displayed by default using Utils::PathChooser. The visual representation often contains a label in front of the display of the actual value. \sa Utils::StringAspect */ class Internal::FilePathAspectPrivate { public: std::function m_displayFilter; QString m_placeHolderText; QString m_prompDialogFilter; QString m_prompDialogTitle; QStringList m_commandVersionArguments; Key m_historyCompleterKey; PathChooser::Kind m_expectedKind = PathChooser::File; Environment m_environment; QPointer m_pathChooserDisplay; FilePath m_baseFileName; StringAspect::ValueAcceptor m_valueAcceptor; std::optional m_validator; std::optional m_effectiveBinary; std::function m_openTerminal; CheckableAspectImplementation m_checkerImpl; bool m_showToolTipOnLabel = false; bool m_fileDialogOnly = false; bool m_autoApplyOnEditingFinished = false; bool m_allowPathFromDevice = true; bool m_validatePlaceHolder = false; Guard m_editFinishedGuard; }; FilePathAspect::FilePathAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::FilePathAspectPrivate) { setSpan(2, 1); // Default: Label + something addDataExtractor(this, &FilePathAspect::value, &Data::value); addDataExtractor(this, &FilePathAspect::operator(), &Data::filePath); connect(this, &BaseAspect::changed, this, [this] { d->m_effectiveBinary.reset(); }); } FilePathAspect::~FilePathAspect() = default; /*! Returns the value of this aspect as \c Utils::FilePath. \note This simply uses \c FilePath::fromUserInput() for the conversion. It does not use any check that the value is actually a valid file path. */ FilePath FilePathAspect::operator()() const { return FilePath::fromUserInput(TypedAspect::value()); } FilePath FilePathAspect::expandedValue() const { const auto value = TypedAspect::value(); if (!value.isEmpty()) { if (auto expander = macroExpander()) return FilePath::fromUserInput(expander->expand(value)); } return FilePath::fromUserInput(value); } /*! Returns the full path of the set command. Only makes a difference if expected kind is \c Command or \c ExistingCommand and the current file path is an executable provided without its path. Performs a lookup in PATH if necessary. */ FilePath FilePathAspect::effectiveBinary() const { if (d->m_effectiveBinary) return *d->m_effectiveBinary; const FilePath current = expandedValue(); const PathChooser::Kind kind = d->m_expectedKind; if (kind != PathChooser::ExistingCommand && kind != PathChooser::Command) return current; if (!current.isLocal()) return current; d->m_effectiveBinary.emplace(current.searchInPath()); return *d->m_effectiveBinary; } QString FilePathAspect::value() const { return TypedAspect::value(); } /*! Sets the value of this file path aspect to \a filePath. If \a howToAnnounce is set to \c DoEmit, emits the \c valueChanged signal. \note This does not use any check that the value is actually a file path. */ void FilePathAspect::setValue(const FilePath &filePath, Announcement howToAnnounce) { TypedAspect::setValue(filePath.toUserOutput(), howToAnnounce); } void FilePathAspect::setValue(const QString &filePath, Announcement howToAnnounce) { TypedAspect::setValue(filePath, howToAnnounce); } void FilePathAspect::setDefaultValue(const QString &filePath) { TypedAspect::setDefaultValue(filePath); } void FilePathAspect::setDefaultPathValue(const FilePath &filePath) { TypedAspect::setDefaultValue(filePath.toUserOutput()); } /*! Adds a check box with a \a checkerLabel according to \a checkBoxPlacement to the line edit. The state of the check box is made persistent when using a non-emtpy \a checkerKey. */ void FilePathAspect::makeCheckable(CheckBoxPlacement checkBoxPlacement, const QString &checkerLabel, const Key &checkerKey) { d->m_checkerImpl.makeCheckable(checkBoxPlacement, checkerLabel, checkerKey, this); } bool FilePathAspect::isChecked() const { return d->m_checkerImpl.isChecked(); } void FilePathAspect::setChecked(bool checked) { return d->m_checkerImpl.setChecked(checked); } void FilePathAspect::setValueAcceptor(ValueAcceptor &&acceptor) { d->m_valueAcceptor = std::move(acceptor); } bool FilePathAspect::guiToBuffer() { if (d->m_pathChooserDisplay) return updateStorage(m_buffer, d->m_pathChooserDisplay->lineEdit()->text()); return false; } bool FilePathAspect::bufferToInternal() { if (d->m_valueAcceptor) { if (const std::optional tmp = d->m_valueAcceptor(m_internal, m_buffer)) return updateStorage(m_internal, *tmp); return false; } return updateStorage(m_internal, m_buffer); } bool FilePathAspect::internalToBuffer() { const QString val = d->m_displayFilter ? d->m_displayFilter(m_internal) : m_internal; return updateStorage(m_buffer, val); } void FilePathAspect::bufferToGui() { if (d->m_pathChooserDisplay) { d->m_pathChooserDisplay->lineEdit()->setText(m_buffer); d->m_checkerImpl.updateWidgetFromCheckStatus(this, d->m_pathChooserDisplay.data()); } validateInput(); } PathChooser *FilePathAspect::pathChooser() const { return d->m_pathChooserDisplay.data(); } void FilePathAspect::addToLayoutImpl(Layouting::Layout &parent) { d->m_checkerImpl.addToLayoutFirst(parent); const QString displayedString = d->m_displayFilter ? d->m_displayFilter(value()) : value(); d->m_pathChooserDisplay = createSubWidget(); addMacroExpansion(d->m_pathChooserDisplay->lineEdit()); d->m_pathChooserDisplay->setExpectedKind(d->m_expectedKind); if (!d->m_historyCompleterKey.isEmpty()) d->m_pathChooserDisplay->setHistoryCompleter(d->m_historyCompleterKey); if (d->m_validator) d->m_pathChooserDisplay->setValidationFunction(*d->m_validator); d->m_pathChooserDisplay->setEnvironment(d->m_environment); d->m_pathChooserDisplay->setBaseDirectory(d->m_baseFileName); d->m_pathChooserDisplay->setOpenTerminalHandler(d->m_openTerminal); d->m_pathChooserDisplay->setPromptDialogFilter(d->m_prompDialogFilter); d->m_pathChooserDisplay->setPromptDialogTitle(d->m_prompDialogTitle); d->m_pathChooserDisplay->setCommandVersionArguments(d->m_commandVersionArguments); d->m_pathChooserDisplay->setAllowPathFromDevice(d->m_allowPathFromDevice); d->m_pathChooserDisplay->setReadOnly(isReadOnly()); d->m_pathChooserDisplay->lineEdit()->setValidatePlaceHolder(d->m_validatePlaceHolder); if (defaultValue() == value()) d->m_pathChooserDisplay->setDefaultValue(defaultValue()); else d->m_pathChooserDisplay->setFilePath(FilePath::fromUserInput(displayedString)); // do not override default value with placeholder, but use placeholder if default is empty if (d->m_pathChooserDisplay->lineEdit()->placeholderText().isEmpty()) d->m_pathChooserDisplay->lineEdit()->setPlaceholderText(d->m_placeHolderText); d->m_checkerImpl.updateWidgetFromCheckStatus(this, d->m_pathChooserDisplay.data()); addLabeledItem(parent, d->m_pathChooserDisplay); connect(d->m_pathChooserDisplay, &PathChooser::validChanged, this, &FilePathAspect::validChanged); bufferToGui(); if (isAutoApply() && d->m_autoApplyOnEditingFinished) { connect(d->m_pathChooserDisplay, &PathChooser::editingFinished, this, [this] { if (d->m_editFinishedGuard.isLocked()) return; GuardLocker lk(d->m_editFinishedGuard); handleGuiChanged(); }); connect(d->m_pathChooserDisplay, &PathChooser::browsingFinished, this, &FilePathAspect::handleGuiChanged); } else { connect(d->m_pathChooserDisplay, &PathChooser::textChanged, this, &FilePathAspect::handleGuiChanged); } d->m_checkerImpl.addToLayoutLast(parent); } /*! \reimp */ void FilePathAspect::fromMap(const Store &map) { if (!skipSave()) setValue(map.value(settingsKey(), defaultValue()).toString(), BeQuiet); d->m_checkerImpl.fromMap(map); } /*! \reimp */ void FilePathAspect::toMap(Store &map) const { saveToMap(map, value(), defaultValue(), settingsKey()); d->m_checkerImpl.toMap(map); } void FilePathAspect::volatileToMap(Store &map) const { saveToMap(map, volatileValue(), defaultValue(), settingsKey()); d->m_checkerImpl.volatileToMap(map); } void FilePathAspect::setPromptDialogFilter(const QString &filter) { d->m_prompDialogFilter = filter; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setPromptDialogFilter(filter); } void FilePathAspect::setPromptDialogTitle(const QString &title) { d->m_prompDialogTitle = title; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setPromptDialogTitle(title); } void FilePathAspect::setCommandVersionArguments(const QStringList &arguments) { d->m_commandVersionArguments = arguments; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setCommandVersionArguments(arguments); } void FilePathAspect::setAllowPathFromDevice(bool allowPathFromDevice) { d->m_allowPathFromDevice = allowPathFromDevice; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setAllowPathFromDevice(allowPathFromDevice); } void FilePathAspect::setValidatePlaceHolder(bool validatePlaceHolder) { d->m_validatePlaceHolder = validatePlaceHolder; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->lineEdit()->setValidatePlaceHolder(validatePlaceHolder); } void FilePathAspect::setShowToolTipOnLabel(bool show) { d->m_showToolTipOnLabel = show; bufferToGui(); } void FilePathAspect::setAutoApplyOnEditingFinished(bool applyOnEditingFinished) { d->m_autoApplyOnEditingFinished = applyOnEditingFinished; } /*! Sets \a expectedKind as expected kind for path chooser displays. \sa Utils::PathChooser::setExpectedKind() */ void FilePathAspect::setExpectedKind(const PathChooser::Kind expectedKind) { if (d->m_expectedKind != expectedKind) { d->m_expectedKind = expectedKind; d->m_effectiveBinary.reset(); if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setExpectedKind(expectedKind); } } void FilePathAspect::setEnvironment(const Environment &env) { d->m_environment = env; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setEnvironment(env); } void FilePathAspect::setBaseFileName(const FilePath &baseFileName) { d->m_baseFileName = baseFileName; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setBaseDirectory(baseFileName); } void FilePathAspect::setPlaceHolderText(const QString &placeHolderText) { d->m_placeHolderText = placeHolderText; } void FilePathAspect::setValidationFunction(const FancyLineEdit::ValidationFunction &validator) { d->m_validator = validator; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setValidationFunction(*d->m_validator); } void FilePathAspect::setDisplayFilter(const std::function &displayFilter) { d->m_displayFilter = displayFilter; } void FilePathAspect::setHistoryCompleter(const Key &historyCompleterKey) { d->m_historyCompleterKey = historyCompleterKey; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setHistoryCompleter(historyCompleterKey); } void FilePathAspect::validateInput() { if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->triggerChanged(); } void FilePathAspect::setOpenTerminalHandler(const std::function &openTerminal) { d->m_openTerminal = openTerminal; if (d->m_pathChooserDisplay) d->m_pathChooserDisplay->setOpenTerminalHandler(openTerminal); } /*! \class Utils::ColorAspect \inmodule QtCreator \brief A color aspect is a color property of some object, together with a description of its behavior for common operations like visualizing or persisting. The color aspect is displayed using a QtColorButton. */ ColorAspect::ColorAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::ColorAspectPrivate) { setDefaultValue(QColor::fromRgb(0, 0, 0)); setSpan(1, 1); } ColorAspect::~ColorAspect() = default; void ColorAspect::addToLayoutImpl(Layouting::Layout &parent) { QTC_CHECK(!d->m_colorButton); d->m_colorButton = createSubWidget(); if (d->m_size.isValid()) d->m_colorButton->setMinimumSize(d->m_size); parent.addItem(d->m_colorButton.data()); bufferToGui(); connect(d->m_colorButton.data(), &QtColorButton::colorChanged, this, &ColorAspect::handleGuiChanged); } void ColorAspect::setMinimumSize(const QSize &size) { d->m_size = size; } bool ColorAspect::guiToBuffer() { if (d->m_colorButton) return updateStorage(m_buffer, d->m_colorButton->color()); return false; } void ColorAspect::bufferToGui() { if (d->m_colorButton) d->m_colorButton->setColor(m_buffer); } static void updateToggleAction(ToggleAspect &aspect, const std::unique_ptr &d) { if (!aspect.hasAction()) return; QAction *action = aspect.action(); Internal::ToggleAspectPrivate::Data data = aspect.value() ? d->on : d->off; if (data.icon.isNull()) data.icon = aspect() ? aspect.icon() : d->on.icon; if (data.text.isEmpty()) data.text = aspect() ? aspect.toolTip() : d->on.text; if (data.tooltip.isEmpty()) data.tooltip = aspect() ? aspect.toolTip() : d->on.tooltip; action->setIcon(data.icon); action->setText(data.text); action->setToolTip(data.tooltip); } /*! \class Utils::ToggleAspect \inmodule QtCreator \brief A toggle aspect is a boolean property of some object, together with a description of its behavior for common operations like visualizing or persisting. It also contains independent tooltips, icons and text for the action() according to the on / off state of the aspect. The aspect is displayed using a QCheckBox. The visual representation often contains a label in front or after the display of the actual checkmark. */ ToggleAspect::ToggleAspect(AspectContainer *container) : BoolAspect(container) , d(std::make_unique()) {} ToggleAspect::~ToggleAspect() {} void ToggleAspect::setOffIcon(const QIcon &icon) { d->off.icon = icon; updateToggleAction(*this, d); } void ToggleAspect::setOffTooltip(const QString &tooltip) { d->off.tooltip = tooltip; updateToggleAction(*this, d); } void ToggleAspect::setOnTooltip(const QString &tooltip) { d->on.tooltip = tooltip; updateToggleAction(*this, d); } void ToggleAspect::setOnIcon(const QIcon &icon) { d->on.icon = icon; updateToggleAction(*this, d); } QString ToggleAspect::onTooltip() const { return d->on.tooltip; } QIcon ToggleAspect::onIcon() const { return d->on.icon; } QString ToggleAspect::offTooltip() const { return d->off.tooltip; } QIcon ToggleAspect::offIcon() const { return d->off.icon; } void ToggleAspect::setOnText(const QString &text) { d->on.text = text; } QString ToggleAspect::onText() const { return d->on.text; } void ToggleAspect::setOffText(const QString &text) { d->off.text = text; } QString ToggleAspect::offText() const { return d->off.text; } void ToggleAspect::announceChanges(Changes changes, Announcement howToAnnounce) { if (changes.internalFromBuffer || changes.internalFromOutside) updateToggleAction(*this, d); BoolAspect::announceChanges(changes, howToAnnounce); } QAction *ToggleAspect::action() { if (hasAction()) return BoolAspect::action(); QAction *a = BoolAspect::action(); updateToggleAction(*this, d); return a; } /*! \class Utils::BoolAspect \inmodule QtCreator \brief A boolean aspect is a boolean property of some object, together with a description of its behavior for common operations like visualizing or persisting. The boolean aspect is displayed using a QCheckBox. The visual representation often contains a label in front or after the display of the actual checkmark. */ BoolAspect::BoolAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::BoolAspectPrivate) { setDefaultValue(false); setSpan(2, 1); d->m_undoable.setSilently(false); } /*! \internal */ BoolAspect::~BoolAspect() = default; void BoolAspect::addToLayoutHelper(Layouting::Layout &parent, QAbstractButton *button) { switch (d->m_labelPlacement) { case LabelPlacement::Compact: button->setText(labelText()); parent.addItem(button); break; case LabelPlacement::AtCheckBox: button->setText(labelText()); parent.addItem(empty); parent.addItem(button); break; case LabelPlacement::InExtraLabel: addLabeledItem(parent, button); break; case LabelPlacement::ShowTip: { parent.addItem(empty); button->setText(labelText()); auto ttLabel = new QLabel(toolTip()); ttLabel->setFont(StyleHelper::uiFont(StyleHelper::UiElementLabelSmall)); auto lt = new QVBoxLayout; lt->setContentsMargins({}); lt->setSpacing(StyleHelper::SpacingTokens::VGapXxs); lt->addWidget(button); lt->addWidget(ttLabel); parent.addItem(lt); break; } } connect(button, &QAbstractButton::clicked, this, [button, this] { d->m_undoable.set(undoStack(), button->isChecked()); }); connect(&d->m_undoable.m_signal, &UndoSignaller::changed, button, [button, this] { button->setChecked(d->m_undoable.get()); handleGuiChanged(); }); } std::function BoolAspect::adoptButton(QAbstractButton *button) { return [this, button](Layouting::Layout *layout) { addToLayoutHelper(*layout, button); bufferToGui(); }; } /*! \reimp */ void BoolAspect::addToLayoutImpl(Layouting::Layout &parent) { QCheckBox *checkBox = createSubWidget(); addToLayoutHelper(parent, checkBox); bufferToGui(); } std::function BoolAspect::groupChecker() { return [this](QObject *target) { auto groupBox = qobject_cast(target); QTC_ASSERT(groupBox, return); registerSubWidget(groupBox); groupBox->setCheckable(true); groupBox->setChecked(value()); connect(groupBox, &QGroupBox::clicked, this, [groupBox, this] { d->m_undoable.set(undoStack(), groupBox->isChecked()); }); connect(&d->m_undoable.m_signal, &UndoSignaller::changed, groupBox, [groupBox, this] { groupBox->setChecked(d->m_undoable.get()); handleGuiChanged(); }); bufferToGui(); }; } QAction *BoolAspect::action() { if (hasAction()) return TypedAspect::action(); auto act = TypedAspect::action(); // Creates it. act->setCheckable(true); act->setChecked(m_internal); act->setToolTip(toolTip()); connect(act, &QAction::triggered, this, [this](bool newValue) { setValue(newValue); }); connect(this, &BoolAspect::changed, act, [act, this] { act->setChecked(m_internal); }); return act; } bool BoolAspect::guiToBuffer() { return updateStorage(m_buffer, d->m_undoable.get()); } void BoolAspect::bufferToGui() { d->m_undoable.setWithoutUndo(m_buffer); } void BoolAspect::setLabel(const QString &labelText, LabelPlacement labelPlacement) { TypedAspect::setLabelText(labelText); d->m_labelPlacement = labelPlacement; } void BoolAspect::setLabelPlacement(BoolAspect::LabelPlacement labelPlacement) { d->m_labelPlacement = labelPlacement; } CheckableDecider BoolAspect::askAgainCheckableDecider() { return CheckableDecider( [this] { return value(); }, [this] { setValue(false); } ); } CheckableDecider BoolAspect::doNotAskAgainCheckableDecider() { return CheckableDecider( [this] { return !value(); }, [this] { setValue(true); } ); } /*! \class Utils::SelectionAspect \inmodule QtCreator \brief A selection aspect represents a specific choice out of several. The selection aspect is displayed using a QComboBox or QRadioButtons in a QButtonGroup. */ SelectionAspect::SelectionAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::SelectionAspectPrivate) { setSpan(2, 1); } /*! \internal */ SelectionAspect::~SelectionAspect() = default; /*! \reimp */ void SelectionAspect::addToLayoutImpl(Layouting::Layout &parent) { QTC_CHECK(d->m_buttonGroup == nullptr); QTC_CHECK(!d->m_comboBox); QTC_ASSERT(d->m_buttons.isEmpty(), d->m_buttons.clear()); switch (d->m_displayStyle) { case DisplayStyle::RadioButtons: d->m_buttonGroup = new QButtonGroup(); d->m_buttonGroup->setExclusive(true); for (int i = 0, n = d->m_options.size(); i < n; ++i) { const Option &option = d->m_options.at(i); auto button = createSubWidget(option.displayName); button->setChecked(i == value()); button->setEnabled(option.enabled); button->setToolTip(option.tooltip); parent.addItem(button); d->m_buttons.append(button); d->m_buttonGroup->addButton(button, i); } bufferToGui(); connect(d->m_buttonGroup, &QButtonGroup::idClicked, this, &SelectionAspect::handleGuiChanged); break; case DisplayStyle::ComboBox: setLabelText(displayName()); d->m_comboBox = createSubWidget(); for (int i = 0, n = d->m_options.size(); i < n; ++i) d->m_comboBox->addItem(d->m_options.at(i).displayName); d->m_comboBox->setCurrentIndex(value()); addLabeledItem(parent, d->m_comboBox); bufferToGui(); connect(d->m_comboBox.data(), &QComboBox::activated, this, &SelectionAspect::handleGuiChanged); break; } } bool SelectionAspect::guiToBuffer() { const int old = m_buffer; switch (d->m_displayStyle) { case DisplayStyle::RadioButtons: if (d->m_buttonGroup) m_buffer = d->m_buttonGroup->checkedId(); break; case DisplayStyle::ComboBox: if (d->m_comboBox) m_buffer = d->m_comboBox->currentIndex(); break; } return m_buffer != old; } void SelectionAspect::bufferToGui() { if (d->m_buttonGroup) { QAbstractButton *button = d->m_buttonGroup->button(m_buffer); QTC_ASSERT(button, return); button->setChecked(true); } else if (d->m_comboBox) { d->m_comboBox->setCurrentIndex(m_buffer); } } void SelectionAspect::finish() { delete d->m_buttonGroup; d->m_buttonGroup = nullptr; BaseAspect::finish(); d->m_buttons.clear(); } void SelectionAspect::setDisplayStyle(SelectionAspect::DisplayStyle style) { d->m_displayStyle = style; } void SelectionAspect::setUseDataAsSavedValue() { setFromSettingsTransformation([this](const QVariant &savedValue) { const int index = indexForItemValue(savedValue); return index >= 0 ? index : defaultVariantValue(); }); setToSettingsTransformation([this](const QVariant &valueToSave) { return itemValueForIndex(valueToSave.toInt()); }); } void SelectionAspect::setStringValue(const QString &val) { const int index = indexForDisplay(val); QTC_ASSERT(index >= 0, return); setValue(index); } void SelectionAspect::setDefaultValue(int val) { TypedAspect::setDefaultValue(val); } // Note: This needs to be set after all options are added. void SelectionAspect::setDefaultValue(const QString &val) { TypedAspect::setDefaultValue(indexForDisplay(val)); } QString SelectionAspect::stringValue() const { const int idx = value(); return idx >= 0 && idx < d->m_options.size() ? d->m_options.at(idx).displayName : QString(); } QVariant SelectionAspect::itemValue() const { const int idx = value(); return idx >= 0 && idx < d->m_options.size() ? d->m_options.at(idx).itemData : QVariant(); } void SelectionAspect::addOption(const QString &displayName, const QString &toolTip) { d->m_options.append(Option(displayName, toolTip, {})); } void SelectionAspect::addOption(const Option &option) { d->m_options.append(option); } int SelectionAspect::indexForDisplay(const QString &displayName) const { for (int i = 0, n = d->m_options.size(); i < n; ++i) { if (d->m_options.at(i).displayName == displayName) return i; } return -1; } QString SelectionAspect::displayForIndex(int index) const { QTC_ASSERT(index >= 0 && index < d->m_options.size(), return {}); return d->m_options.at(index).displayName; } int SelectionAspect::indexForItemValue(const QVariant &value) const { for (int i = 0, n = d->m_options.size(); i < n; ++i) { if (d->m_options.at(i).itemData == value) return i; } return -1; } QVariant SelectionAspect::itemValueForIndex(int index) const { QTC_ASSERT(index >= 0 && index < d->m_options.size(), return {}); return d->m_options.at(index).itemData; } /*! \class Utils::MultiSelectionAspect \inmodule QtCreator \brief A multi-selection aspect represents one or more choices out of several. The multi-selection aspect is displayed using a QListWidget with checkable items. */ MultiSelectionAspect::MultiSelectionAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::MultiSelectionAspectPrivate(this)) { setDefaultValue(QStringList()); setSpan(2, 1); } /*! \internal */ MultiSelectionAspect::~MultiSelectionAspect() = default; /*! \reimp */ void MultiSelectionAspect::addToLayoutImpl(Layout &builder) { QTC_CHECK(d->m_listView == nullptr); if (d->m_allValues.isEmpty()) return; switch (d->m_displayStyle) { case DisplayStyle::ListView: d->m_listView = createSubWidget(); for (const QString &val : std::as_const(d->m_allValues)) (void) new QListWidgetItem(val, d->m_listView); addLabeledItem(builder, d->m_listView); bufferToGui(); connect(d->m_listView, &QListWidget::itemChanged, this, &MultiSelectionAspect::handleGuiChanged); } } bool Internal::MultiSelectionAspectPrivate::setValueSelectedHelper(const QString &val, bool on) { QStringList list = q->value(); if (on && !list.contains(val)) { list.append(val); q->setValue(list); return true; } if (!on && list.contains(val)) { list.removeOne(val); q->setValue(list); return true; } return false; } QStringList MultiSelectionAspect::allValues() const { return d->m_allValues; } void MultiSelectionAspect::setAllValues(const QStringList &val) { d->m_allValues = val; } void MultiSelectionAspect::setDisplayStyle(MultiSelectionAspect::DisplayStyle style) { d->m_displayStyle = style; } void MultiSelectionAspect::bufferToGui() { if (d->m_listView) { const int n = d->m_listView->count(); QTC_CHECK(n == d->m_allValues.size()); for (int i = 0; i != n; ++i) { auto item = d->m_listView->item(i); item->setCheckState(m_buffer.contains(item->text()) ? Qt::Checked : Qt::Unchecked); } } } bool MultiSelectionAspect::guiToBuffer() { if (d->m_listView) { QStringList val; const int n = d->m_listView->count(); QTC_CHECK(n == d->m_allValues.size()); for (int i = 0; i != n; ++i) { auto item = d->m_listView->item(i); if (item->checkState() == Qt::Checked) val.append(item->text()); } return updateStorage(m_buffer, val); } return false; } /*! \class Utils::IntegerAspect \inmodule QtCreator \brief An integer aspect is a integral property of some object, together with a description of its behavior for common operations like visualizing or persisting. The integer aspect is displayed using a \c QSpinBox. The visual representation often contains a label in front the display of the spin box. */ // IntegerAspect IntegerAspect::IntegerAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::IntegerAspectPrivate) { setSpan(2, 1); } /*! \internal */ IntegerAspect::~IntegerAspect() = default; /*! \reimp */ void IntegerAspect::addToLayoutImpl(Layouting::Layout &parent) { QTC_CHECK(!d->m_spinBox); d->m_spinBox = createSubWidget(); d->m_spinBox->setDisplayIntegerBase(d->m_displayIntegerBase); d->m_spinBox->setPrefix(d->m_prefix); d->m_spinBox->setSuffix(d->m_suffix); d->m_spinBox->setSingleStep(d->m_singleStep); d->m_spinBox->setSpecialValueText(d->m_specialValueText); if (d->m_minimumValue && d->m_maximumValue) d->m_spinBox->setRange(int(d->m_minimumValue.value() / d->m_displayScaleFactor), int(d->m_maximumValue.value() / d->m_displayScaleFactor)); bufferToGui(); addLabeledItem(parent, d->m_spinBox); connect(d->m_spinBox.data(), &QSpinBox::valueChanged, this, &IntegerAspect::handleGuiChanged); } bool IntegerAspect::guiToBuffer() { if (d->m_spinBox) return updateStorage(m_buffer, d->m_spinBox->value() * d->m_displayScaleFactor); return false; } void IntegerAspect::bufferToGui() { if (d->m_spinBox) d->m_spinBox->setValue(m_buffer / d->m_displayScaleFactor); } void IntegerAspect::setRange(qint64 min, qint64 max) { d->m_minimumValue = min; d->m_maximumValue = max; } void IntegerAspect::setLabel(const QString &label) { setLabelText(label); } void IntegerAspect::setPrefix(const QString &prefix) { d->m_prefix = prefix; } void IntegerAspect::setSuffix(const QString &suffix) { d->m_suffix = suffix; } void IntegerAspect::setDisplayIntegerBase(int base) { d->m_displayIntegerBase = base; } void IntegerAspect::setDisplayScaleFactor(qint64 factor) { d->m_displayScaleFactor = factor; } void IntegerAspect::setSpecialValueText(const QString &specialText) { d->m_specialValueText = specialText; } void IntegerAspect::setSingleStep(qint64 step) { d->m_singleStep = step; } /*! \class Utils::DoubleAspect \inmodule QtCreator \brief An double aspect is a numerical property of some object, together with a description of its behavior for common operations like visualizing or persisting. The double aspect is displayed using a \c QDoubleSpinBox. The visual representation often contains a label in front the display of the spin box. */ DoubleAspect::DoubleAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::DoubleAspectPrivate) { setDefaultValue(double(0)); setSpan(2, 1); } /*! \internal */ DoubleAspect::~DoubleAspect() = default; /*! \reimp */ void DoubleAspect::addToLayoutImpl(Layout &builder) { QTC_CHECK(!d->m_spinBox); d->m_spinBox = createSubWidget(); d->m_spinBox->setPrefix(d->m_prefix); d->m_spinBox->setSuffix(d->m_suffix); d->m_spinBox->setSingleStep(d->m_singleStep); d->m_spinBox->setSpecialValueText(d->m_specialValueText); if (d->m_minimumValue && d->m_maximumValue) d->m_spinBox->setRange(d->m_minimumValue.value(), d->m_maximumValue.value()); bufferToGui(); // Must happen after setRange()! addLabeledItem(builder, d->m_spinBox); connect(d->m_spinBox.data(), &QDoubleSpinBox::valueChanged, this, &DoubleAspect::handleGuiChanged); } bool DoubleAspect::guiToBuffer() { if (d->m_spinBox) return updateStorage(m_buffer, d->m_spinBox->value()); return false; } void DoubleAspect::bufferToGui() { if (d->m_spinBox) d->m_spinBox->setValue(m_buffer); } void DoubleAspect::setRange(double min, double max) { d->m_minimumValue = min; d->m_maximumValue = max; } void DoubleAspect::setPrefix(const QString &prefix) { d->m_prefix = prefix; } void DoubleAspect::setSuffix(const QString &suffix) { d->m_suffix = suffix; } void DoubleAspect::setSpecialValueText(const QString &specialText) { d->m_specialValueText = specialText; } void DoubleAspect::setSingleStep(double step) { d->m_singleStep = step; } /*! \class Utils::TriStateAspect \inmodule QtCreator \brief A tristate aspect is a property of some object that can have three values: enabled, disabled, and unspecified. Its visual representation is a QComboBox with three items. */ TriStateAspect::TriStateAspect(AspectContainer *container, const QString &enabledDisplay, const QString &disabledDisplay, const QString &defaultDisplay) : SelectionAspect(container) { setDisplayStyle(DisplayStyle::ComboBox); setDefaultValue(TriState::Default); SelectionAspect::addOption({}); SelectionAspect::addOption({}); SelectionAspect::addOption({}); setOptionText(TriState::EnabledValue, enabledDisplay); setOptionText(TriState::DisabledValue, disabledDisplay); setOptionText(TriState::DefaultValue, defaultDisplay); } static QString defaultTristateDisplay(TriState::Value tristate) { switch (tristate) { case TriState::EnabledValue: return Tr::tr("Enable"); case TriState::DisabledValue: return Tr::tr("Disable"); case TriState::DefaultValue: return Tr::tr("Default"); } QTC_CHECK(false); return {}; } void TriStateAspect::setOptionText(const TriState::Value tristate, const QString &display) { d->m_options[tristate].displayName = display.isEmpty() ? defaultTristateDisplay(tristate) : display; } TriState TriStateAspect::value() const { return TriState::fromInt(SelectionAspect::value()); } void TriStateAspect::setValue(TriState value) { SelectionAspect::setValue(value.toInt()); } TriState TriStateAspect::defaultValue() const { return TriState::fromInt(SelectionAspect::defaultValue()); } void TriStateAspect::setDefaultValue(TriState value) { SelectionAspect::setDefaultValue(value.toInt()); } const TriState TriState::Enabled{TriState::EnabledValue}; const TriState TriState::Disabled{TriState::DisabledValue}; const TriState TriState::Default{TriState::DefaultValue}; TriState TriState::fromVariant(const QVariant &variant) { return fromInt(variant.toInt()); } TriState TriState::fromInt(int v) { QTC_ASSERT(v == EnabledValue || v == DisabledValue || v == DefaultValue, v = DefaultValue); return TriState(Value(v)); } /*! \class Utils::StringListAspect \inmodule QtCreator \brief A string list aspect represents a property of some object that is a list of strings. */ StringListAspect::StringListAspect(AspectContainer *container) : TypedAspect(container), d(new Internal::StringListAspectPrivate) { setDefaultValue(QStringList()); } /*! \internal */ StringListAspect::~StringListAspect() = default; bool StringListAspect::guiToBuffer() { const QStringList newValue = d->undoable.get(); if (newValue != m_buffer) { m_buffer = newValue; return true; } return false; } void StringListAspect::bufferToGui() { d->undoable.setWithoutUndo(m_buffer); } void StringListAspect::addToLayoutImpl(Layout &parent) { d->undoable.setSilently(value()); auto editor = new QTreeWidget(); editor->setHeaderHidden(true); editor->setRootIsDecorated(false); editor->setEditTriggers( d->allowEditing ? QAbstractItemView::AllEditTriggers : QAbstractItemView::NoEditTriggers); QPushButton *add = d->allowAdding ? new QPushButton(Tr::tr("Add")) : nullptr; QPushButton *remove = d->allowRemoving ? new QPushButton(Tr::tr("Remove")) : nullptr; auto itemsToStringList = [editor] { QStringList items; const QTreeWidgetItem *rootItem = editor->invisibleRootItem(); for (int i = 0, count = rootItem->childCount(); i < count; ++i) { auto expr = rootItem->child(i)->data(0, Qt::DisplayRole).toString(); items.append(expr); } return items; }; auto populate = [editor, this] { editor->clear(); for (const QString &entry : d->undoable.get()) { auto item = new QTreeWidgetItem(editor, {entry}); item->setData(0, Qt::ToolTipRole, entry); item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable); } }; if (add) { connect(add, &QPushButton::clicked, this, [this, populate, editor] { d->undoable.setSilently(d->undoable.get() << ""); populate(); const QTreeWidgetItem *root = editor->invisibleRootItem(); QTreeWidgetItem *lastChild = root->child(root->childCount() - 1); const QModelIndex index = editor->indexFromItem(lastChild, 0); editor->edit(index); }); } if (remove) { connect(remove, &QPushButton::clicked, this, [this, editor, itemsToStringList] { const QList selected = editor->selectedItems(); QTC_ASSERT(selected.size() == 1, return); editor->invisibleRootItem()->removeChild(selected.first()); delete selected.first(); d->undoable.set(undoStack(), itemsToStringList()); }); } connect( &d->undoable.m_signal, &UndoSignaller::changed, editor, [this, populate, itemsToStringList] { if (itemsToStringList() != d->undoable.get()) populate(); handleGuiChanged(); }); connect( editor->model(), &QAbstractItemModel::dataChanged, this, [this, itemsToStringList](const QModelIndex &tl, const QModelIndex &br, const QList &roles) { if (!roles.contains(Qt::DisplayRole)) return; if (tl != br) return; d->undoable.set(undoStack(), itemsToStringList()); }); populate(); parent.addItem( // clang-format off Column { createLabel(), Row { editor, If { d->allowAdding || d->allowRemoving, { Column { If { d->allowAdding, {add}, {}}, If { d->allowRemoving, {remove}, {}}, st, } }, {}}, } } // clang-format on ); registerSubWidget(editor); if (d->allowAdding) registerSubWidget(add); if (d->allowRemoving) registerSubWidget(remove); } void StringListAspect::appendValue(const QString &s, bool allowDuplicates) { QStringList val = value(); if (allowDuplicates || !val.contains(s)) val.append(s); setValue(val); } void StringListAspect::removeValue(const QString &s) { QStringList val = value(); val.removeAll(s); setValue(val); } void StringListAspect::appendValues(const QStringList &values, bool allowDuplicates) { QStringList val = value(); for (const QString &s : values) { if (allowDuplicates || !val.contains(s)) val.append(s); } setValue(val); } void StringListAspect::removeValues(const QStringList &values) { QStringList val = value(); for (const QString &s : values) val.removeAll(s); setValue(val); } void StringListAspect::setUiAllowAdding(bool allowAdding) { d->allowAdding = allowAdding; } void StringListAspect::setUiAllowRemoving(bool allowRemoving) { d->allowRemoving = allowRemoving; } void StringListAspect::setUiAllowEditing(bool allowEditing) { d->allowEditing = allowEditing; } bool StringListAspect::uiAllowAdding() const { return d->allowAdding; } bool StringListAspect::uiAllowRemoving() const { return d->allowRemoving; } bool StringListAspect::uiAllowEditing() const { return d->allowEditing; } /*! \class Utils::FilePathListAspect \inmodule QtCreator \brief A filepath list aspect represents a property of some object that is a list of filepathList. */ FilePathListAspect::FilePathListAspect(AspectContainer *container) : TypedAspect(container) , d(new Internal::FilePathListAspectPrivate) { setDefaultValue(QStringList()); } FilePathListAspect::~FilePathListAspect() = default; FilePaths FilePathListAspect::operator()() const { return Utils::transform(m_internal, &FilePath::fromUserInput); } bool FilePathListAspect::guiToBuffer() { const QStringList newValue = d->undoable.get(); if (newValue != m_buffer) { m_buffer = newValue; return true; } return false; } void FilePathListAspect::bufferToGui() { d->undoable.setWithoutUndo(m_buffer); } void FilePathListAspect::addToLayoutImpl(Layout &parent) { d->undoable.setSilently(value()); PathListEditor *editor = new PathListEditor; editor->setPathList(value()); connect(editor, &PathListEditor::changed, this, [this, editor] { d->undoable.set(undoStack(), editor->pathList()); }); connect(&d->undoable.m_signal, &UndoSignaller::changed, editor, [this, editor] { if (editor->pathList() != d->undoable.get()) editor->setPathList(d->undoable.get()); handleGuiChanged(); }); editor->setToolTip(toolTip()); editor->setMaximumHeight(100); editor->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); editor->setPlaceholderText(d->placeHolderText); registerSubWidget(editor); parent.addItem(editor); } void FilePathListAspect::setPlaceHolderText(const QString &placeHolderText) { d->placeHolderText = placeHolderText; forEachSubWidget([placeHolderText](QWidget *widget) { if (auto pathListEditor = qobject_cast(widget)) { pathListEditor->setPlaceholderText(placeHolderText); } }); } void FilePathListAspect::appendValue(const FilePath &path, bool allowDuplicates) { const QString asString = path.toUserOutput(); QStringList val = value(); if (allowDuplicates || !val.contains(asString)) val.append(asString); setValue(val); } void FilePathListAspect::removeValue(const FilePath &s) { QStringList val = value(); val.removeAll(s.toUserOutput()); setValue(val); } void FilePathListAspect::appendValues(const FilePaths &paths, bool allowDuplicates) { QStringList val = value(); for (const FilePath &path : paths) { const QString asString = path.toUserOutput(); if (allowDuplicates || !val.contains(asString)) val.append(asString); } setValue(val); } void FilePathListAspect::removeValues(const FilePaths &paths) { QStringList val = value(); for (const FilePath &path : paths) val.removeAll(path.toUserOutput()); setValue(val); } /*! \class Utils::IntegersAspect \internal \inmodule QtCreator \brief An integers aspect represents a property of some object that is a list of integers. */ IntegersAspect::IntegersAspect(AspectContainer *container) : TypedAspect(container) {} /*! \internal */ IntegersAspect::~IntegersAspect() = default; /*! \reimp */ void IntegersAspect::addToLayoutImpl(Layouting::Layout &parent) { Q_UNUSED(parent) // TODO - when needed. } /*! \class Utils::TextDisplay \inmodule QtCreator \brief A text display is a phony aspect with the sole purpose of providing some text display using an Utils::InfoLabel in places where otherwise more expensive Utils::StringAspect items would be used. A text display does not have a real value. */ /*! Constructs a text display with the parent \a container. The display shows \a message and an icon representing the type \a type. */ TextDisplay::TextDisplay(AspectContainer *container, const QString &message, InfoLabel::InfoType type) : BaseAspect(container), d(new Internal::TextDisplayPrivate) { d->m_message = message; d->m_type = type; } /*! \internal */ TextDisplay::~TextDisplay() = default; /*! \reimp */ void TextDisplay::addToLayoutImpl(Layout &parent) { if (!d->m_label) { d->m_label = createSubWidget(d->m_message, d->m_type); d->m_label->setTextInteractionFlags(Qt::TextSelectableByMouse); d->m_label->setElideMode(Qt::ElideNone); d->m_label->setWordWrap(true); // Do not use m_label->setVisible(isVisible()) unconditionally, it does not // have a QWidget parent yet when used in a LayoutBuilder. if (!isVisible()) d->m_label->setVisible(false); connect(this, &TextDisplay::changed, d->m_label, [this] { d->m_label->setText(d->m_message); }); } parent.addItem(d->m_label.data()); } /*! Sets \a t as the information label type for the visual representation of this aspect. */ void TextDisplay::setIconType(InfoLabel::InfoType t) { d->m_type = t; if (d->m_label) d->m_label->setType(t); } void TextDisplay::setText(const QString &message) { d->m_message = message; emit changed(); } /*! \class Utils::AspectContainer \inmodule QtCreator \brief The AspectContainer class wraps one or more aspects while providing the interface of a single aspect. Sub-aspects ownership can be declared using \a setOwnsSubAspects. */ class Internal::AspectContainerPrivate { public: QList m_items; // Both owned and non-owned. QList m_ownedItems; // Owned only. QStringList m_settingsGroup; std::function m_layouter; }; AspectContainer::AspectContainer() : d(new Internal::AspectContainerPrivate) {} /*! \internal */ AspectContainer::~AspectContainer() { qDeleteAll(d->m_ownedItems); } void AspectContainer::addToLayoutImpl(Layouting::Layout &parent) { parent.addItem(layouter()()); } /*! \internal */ void AspectContainer::registerAspect(BaseAspect *aspect, bool takeOwnership) { aspect->setContainer(this); aspect->setAutoApply(isAutoApply()); aspect->setEnabled(aspect->isEnabled() && isEnabled()); d->m_items.append(aspect); if (takeOwnership) d->m_ownedItems.append(aspect); connect(aspect, &BaseAspect::changed, this, [this]() { emit changed(); }); } void AspectContainer::registerAspects(const AspectContainer &aspects) { for (BaseAspect *aspect : std::as_const(aspects.d->m_items)) registerAspect(aspect); } /*! Retrieves a BaseAspect with a given \a id, or nullptr if no such aspect is contained. \sa BaseAspect */ BaseAspect *AspectContainer::aspect(Id id) const { return findOrDefault(d->m_items, equal(&BaseAspect::id, id)); } AspectContainer::const_iterator AspectContainer::begin() const { return d->m_items.cbegin(); } AspectContainer::const_iterator AspectContainer::end() const { return d->m_items.cend(); } void AspectContainer::setLayouter(const std::function &layouter) { d->m_layouter = layouter; } std::function AspectContainer::layouter() const { return d->m_layouter; } const QList &AspectContainer::aspects() const { return d->m_items; } void AspectContainer::fromMap(const Store &map) { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->fromMap(map); emit fromMapFinished(); } void AspectContainer::toMap(Store &map) const { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->toMap(map); } void AspectContainer::volatileToMap(Store &map) const { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->volatileToMap(map); } void AspectContainer::readSettings() { const SettingsGroupNester nester(d->m_settingsGroup); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->readSettings(); } void AspectContainer::writeSettings() const { const SettingsGroupNester nester(d->m_settingsGroup); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->writeSettings(); } void AspectContainer::setSettingsGroup(const QString &groupKey) { d->m_settingsGroup = QStringList{groupKey}; } void AspectContainer::setSettingsGroups(const QString &groupKey, const QString &subGroupKey) { d->m_settingsGroup = QStringList{groupKey, subGroupKey}; } QStringList AspectContainer::settingsGroups() const { return d->m_settingsGroup; } void AspectContainer::apply() { const bool willChange = isDirty(); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->apply(); emit applied(); if (willChange) emit changed(); } void AspectContainer::cancel() { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->cancel(); } void AspectContainer::finish() { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->finish(); } void AspectContainer::reset() { for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->setVariantValue(aspect->defaultVariantValue()); } void AspectContainer::setAutoApply(bool on) { BaseAspect::setAutoApply(on); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->setAutoApply(on); } bool AspectContainer::isDirty() { for (BaseAspect *aspect : std::as_const(d->m_items)) { if (aspect->isDirty()) return true; } return false; } void AspectContainer::setUndoStack(QUndoStack *undoStack) { BaseAspect::setUndoStack(undoStack); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->setUndoStack(undoStack); } void AspectContainer::setEnabled(bool enabled) { BaseAspect::setEnabled(enabled); for (BaseAspect *aspect : std::as_const(d->m_items)) aspect->setEnabled(enabled); } bool AspectContainer::equals(const AspectContainer &other) const { // FIXME: Expensive, but should not really be needed in a fully aspectified world. Store thisMap, thatMap; toMap(thisMap); other.toMap(thatMap); return thisMap == thatMap; } void AspectContainer::copyFrom(const AspectContainer &other) { Store map; other.toMap(map); fromMap(map); } void AspectContainer::forEachAspect(const std::function &run) const { for (BaseAspect *aspect : std::as_const(d->m_items)) { if (auto container = dynamic_cast(aspect)) container->forEachAspect(run); else run(aspect); } } BaseAspect::Data::Ptr BaseAspect::extractData() const { QTC_ASSERT(d->m_dataCreator, return {}); Data *data = d->m_dataCreator(); data->m_classId = metaObject(); data->m_id = id(); data->m_cloner = d->m_dataCloner; for (const DataExtractor &extractor : std::as_const(d->m_dataExtractors)) extractor(data); return Data::Ptr(data); } /* Mirrors the internal volatile value to the GUI element if they are already created. No-op otherwise. */ void BaseAspect::bufferToGui() { } /* Mirrors the data stored in GUI element if they are already created to the internal volatile value. No-op otherwise. \return true when the buffered volatile value changed. */ bool BaseAspect::guiToBuffer() { return false; } /* Mirrors buffered volatile value to the internal value. This function is used for \c apply(). \return true when the internal value changed. */ bool BaseAspect::bufferToInternal() { return false; } bool BaseAspect::internalToBuffer() { return false; } void BaseAspect::handleGuiChanged() { if (guiToBuffer()) volatileValueChanged(); if (isAutoApply()) apply(); } void BaseAspect::addDataExtractorHelper(const DataExtractor &extractor) const { d->m_dataExtractors.append(extractor); } void BaseAspect::setDataCreatorHelper(const DataCreator &creator) const { d->m_dataCreator = creator; } void BaseAspect::setDataClonerHelper(const DataCloner &cloner) const { d->m_dataCloner = cloner; } const BaseAspect::Data *AspectContainerData::aspect(Id instanceId) const { for (const BaseAspect::Data::Ptr &data : m_data) { if (data.get()->id() == instanceId) return data.get(); } return nullptr; } const BaseAspect::Data *AspectContainerData::aspect(BaseAspect::Data::ClassId classId) const { for (const BaseAspect::Data::Ptr &data : m_data) { if (data.get()->classId() == classId) return data.get(); } return nullptr; } void AspectContainerData::append(const BaseAspect::Data::Ptr &data) { m_data.append(data); } // BaseAspect::Data BaseAspect::Data::~Data() = default; void BaseAspect::Data::Ptr::operator=(const Ptr &other) { if (this == &other) return; delete m_data; m_data = other.m_data->clone(); } // SettingsGroupNester SettingsGroupNester::SettingsGroupNester(const QStringList &groups) : m_groupCount(groups.size()) { QTC_ASSERT(theSettings, return); for (const QString &group : groups) theSettings->beginGroup(keyFromString(group)); } SettingsGroupNester::~SettingsGroupNester() { QTC_ASSERT(theSettings, return); for (int i = 0; i != m_groupCount; ++i) theSettings->endGroup(); } class AddItemCommand : public QUndoCommand { public: AddItemCommand(AspectList *aspect, const std::shared_ptr &item) : m_aspect(aspect) , m_item(item) {} void undo() override { m_aspect->actualRemoveItem(m_item); } void redo() override { m_aspect->actualAddItem(m_item); } private: AspectList *m_aspect; std::shared_ptr m_item; }; class RemoveItemCommand : public QUndoCommand { public: RemoveItemCommand(AspectList *aspect, const std::shared_ptr &item) : m_aspect(aspect) , m_item(item) {} void undo() override { m_aspect->actualAddItem(m_item); } void redo() override { m_aspect->actualRemoveItem(m_item); } private: AspectList *m_aspect; std::shared_ptr m_item; }; class Internal::AspectListPrivate { public: QList> items; QList> volatileItems; AspectList::CreateItem createItem; AspectList::ItemCallback itemAdded; AspectList::ItemCallback itemRemoved; }; AspectList::AspectList(Utils::AspectContainer *container) : Utils::BaseAspect(container) , d(std::make_unique()) {} AspectList::~AspectList() = default; void AspectList::fromMap(const Utils::Store &map) { QTC_ASSERT(!settingsKey().isEmpty(), return); const QVariantList list = map[settingsKey()].toList(); d->volatileItems.clear(); for (const QVariant &entry : list) { auto item = d->createItem(); item->setAutoApply(isAutoApply()); item->setUndoStack(undoStack()); item->fromMap(Utils::storeFromVariant(entry)); d->volatileItems.append(item); } d->items = d->volatileItems; } QVariantList AspectList::toList(bool v) const { QVariantList list; const auto &items = v ? d->volatileItems : d->items; for (const std::shared_ptr &item : items) { Utils::Store childStore; if (v) item->volatileToMap(childStore); else item->toMap(childStore); list.append(Utils::variantFromStore(childStore)); } return list; } void AspectList::toMap(Utils::Store &map) const { QTC_ASSERT(!settingsKey().isEmpty(), return); const Utils::Key key = settingsKey(); map[key] = toList(false); } void AspectList::volatileToMap(Utils::Store &map) const { QTC_ASSERT(!settingsKey().isEmpty(), return); const Utils::Key key = settingsKey(); map[key] = toList(true); } std::shared_ptr AspectList::actualAddItem(const std::shared_ptr &item) { item->setAutoApply(isAutoApply()); item->setUndoStack(undoStack()); d->volatileItems.append(item); if (d->itemAdded) d->itemAdded(item); emit volatileValueChanged(); if (isAutoApply()) d->items = d->volatileItems; return item; } QList> AspectList::items() const { return d->items; } QList> AspectList::volatileItems() const { return d->volatileItems; } std::shared_ptr AspectList::createAndAddItem() { return addItem(d->createItem()); } std::shared_ptr AspectList::addItem(const std::shared_ptr &item) { if (undoStack()) undoStack()->push(new AddItemCommand(this, item)); else return actualAddItem(item); return item; } void AspectList::actualRemoveItem(const std::shared_ptr &item) { d->volatileItems.removeOne(item); if (d->itemRemoved) d->itemRemoved(item); emit volatileValueChanged(); if (isAutoApply()) d->items = d->volatileItems; } void AspectList::removeItem(const std::shared_ptr &item) { if (undoStack()) undoStack()->push(new RemoveItemCommand(this, item)); else actualRemoveItem(item); } void AspectList::clear() { if (undoStack()) { undoStack()->beginMacro("Clear"); for (const std::shared_ptr &item : volatileItems()) undoStack()->push(new RemoveItemCommand(this, item)); undoStack()->endMacro(); } else { for (const std::shared_ptr &item : volatileItems()) actualRemoveItem(item); } } void AspectList::apply() { d->items = d->volatileItems; forEachItem([](const std::shared_ptr &aspect) { aspect->apply(); }); emit changed(); } void AspectList::setCreateItemFunction(CreateItem createItem) { d->createItem = createItem; } void AspectList::setItemAddedCallback(const ItemCallback &callback) { d->itemAdded = callback; } void AspectList::setItemRemovedCallback(const ItemCallback &callback) { d->itemRemoved = callback; } qsizetype AspectList::size() const { return d->volatileItems.size(); } bool AspectList::isDirty() { if (d->items != d->volatileItems) return true; for (const std::shared_ptr &item : std::as_const(d->volatileItems)) { if (item->isDirty()) return true; } return false; } class ColoredRow : public QWidget { public: ColoredRow(int idx, QWidget *parent = nullptr) : QWidget(parent) , m_index(idx) {} void paintEvent(QPaintEvent *event) override { QPainter p(this); QPalette pal = palette(); if (m_index % 2 == 0) p.fillRect(event->rect(), pal.base()); else p.fillRect(event->rect(), pal.alternateBase()); } private: int m_index; }; void AspectList::addToLayoutImpl(Layouting::Layout &parent) { using namespace Layouting; using namespace Utils::QtcWidgets; auto fill = [this] { const auto createRow = [this](const std::shared_ptr &item) { // clang-format off return Row { *item, IconButton { ::icon(Utils::Icons::EDIT_CLEAR), sizePolicy(QSizePolicy{QSizePolicy::Fixed, QSizePolicy::Fixed}), onClicked(this, [this, item] { removeItem(item); }) }, spacing(5), noMargin, }; // clang-format on }; // clang-format off return Column { Utils::transform(volatileItems(), createRow), Row { noMargin, st, IconButton { ::icon(Utils::Icons::PLUS), onClicked(this, [this](){ addItem(d->createItem()); }) } } }; // clang-format on }; // clang-format off parent.addItem( Group { replaceLayoutOn(this, &AspectList::volatileValueChanged, fill) } ); // clang-format on } StringSelectionAspect::StringSelectionAspect(AspectContainer *container) : TypedAspect(container) {} QStandardItem *StringSelectionAspect::itemById(const QString &id) { for (int i = 0; i < m_model->rowCount(); ++i) { auto cur = m_model->item(i); if (cur->data() == id) return cur; } return nullptr; } void StringSelectionAspect::bufferToGui() { if (!m_model) { m_undoable.setSilently(m_buffer); return; } auto selected = itemById(m_buffer); if (selected) { m_undoable.setSilently(selected->data().toString()); m_selectionModel->setCurrentIndex(selected->index(), QItemSelectionModel::SelectionFlag::ClearAndSelect); return; } if (m_model->rowCount() > 0) { m_undoable.setSilently(m_model->item(0)->data().toString()); m_selectionModel->setCurrentIndex(m_model->item(0)->index(), QItemSelectionModel::SelectionFlag::ClearAndSelect); } else { m_undoable.setSilently(m_buffer); m_selectionModel->setCurrentIndex(QModelIndex(), QItemSelectionModel::SelectionFlag::Clear); } handleGuiChanged(); } bool StringSelectionAspect::guiToBuffer() { if (!m_model) return false; auto oldBuffer = m_buffer; m_buffer = m_undoable.get(); return oldBuffer != m_buffer; } void StringSelectionAspect::addToLayoutImpl(Layouting::Layout &parent) { QTC_ASSERT(m_fillCallback, return); auto cb = [this](const QList &items) { m_model->clear(); for (QStandardItem *item : items) m_model->appendRow(item); bufferToGui(); }; if (!m_model) { m_model = new QStandardItemModel(this); m_selectionModel = new QItemSelectionModel(m_model); connect(this, &StringSelectionAspect::refillRequested, this, [this, cb] { m_fillCallback(cb); }); m_fillCallback(cb); } QComboBox *comboBox = new QComboBox(); comboBox->setInsertPolicy(QComboBox::InsertPolicy::NoInsert); comboBox->setEditable(m_comboBoxEditable); if (m_comboBoxEditable) { comboBox->completer()->setCompletionMode(QCompleter::PopupCompletion); comboBox->completer()->setFilterMode(Qt::MatchContains); } comboBox->setSizeAdjustPolicy(QComboBox::AdjustToMinimumContentsLengthWithIcon); comboBox->setCurrentText(value()); comboBox->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Fixed); comboBox->setModel(m_model); setWheelScrollingWithoutFocusBlocked(comboBox); connect(m_selectionModel, &QItemSelectionModel::currentChanged, comboBox, [comboBox](QModelIndex currentIdx) { if (currentIdx.isValid() && comboBox->currentIndex() != currentIdx.row()) comboBox->setCurrentIndex(currentIdx.row()); }); connect(comboBox, &QComboBox::activated, this, [this](int idx) { QModelIndex modelIdx = m_model->index(idx, 0); if (!modelIdx.isValid()) return; QString newValue = m_model->index(idx, 0).data(Qt::UserRole + 1).toString(); if (newValue.isEmpty()) return; m_undoable.set(undoStack(), newValue); bufferToGui(); }); connect(&m_undoable.m_signal, &UndoSignaller::changed, comboBox, [this, comboBox] { auto item = itemById(m_undoable.get()); if (item) m_selectionModel->setCurrentIndex(item->index(), QItemSelectionModel::ClearAndSelect); else comboBox->setCurrentText(m_undoable.get()); handleGuiChanged(); }); if (m_selectionModel->currentIndex().isValid()) comboBox->setCurrentIndex(m_selectionModel->currentIndex().row()); return addLabeledItem(parent, comboBox); } } // namespace Utils