File indexing completed on 2025-10-19 05:04:31

0001 /*  view/formtextinput.h
0002 
0003     This file is part of Kleopatra, the KDE keymanager
0004     SPDX-FileCopyrightText: 2022 g10 Code GmbH
0005     SPDX-FileContributor: Ingo Klöcker <dev@ingo-kloecker.de>
0006 
0007     SPDX-License-Identifier: GPL-2.0-or-later
0008 */
0009 #pragma once
0010 
0011 #include <QString>
0012 
0013 #include <memory>
0014 
0015 class QLabel;
0016 class QLineEdit;
0017 class QValidator;
0018 class QWidget;
0019 
0020 namespace Kleo
0021 {
0022 class ErrorLabel;
0023 
0024 namespace _detail
0025 {
0026 class FormTextInputBase
0027 {
0028 protected:
0029     FormTextInputBase();
0030 
0031 public:
0032     virtual ~FormTextInputBase();
0033     FormTextInputBase(const FormTextInputBase &) = delete;
0034     FormTextInputBase &operator=(const FormTextInputBase &) = delete;
0035     FormTextInputBase(FormTextInputBase &&) = delete;
0036     FormTextInputBase &operator=(FormTextInputBase &&) = delete;
0037 
0038     /**
0039      * Returns the label associated to the controlled widget. Use it to add
0040      * the label to a layout, but do not use it to set properties of the label
0041      * for which this class provides setters.
0042      */
0043     QLabel *label() const;
0044 
0045     /**
0046      * Returns the hint label associated to the controlled widget.
0047      */
0048     QLabel *hintLabel() const;
0049 
0050     /**
0051      * Returns the error label associated to the controlled widget.
0052      */
0053     ErrorLabel *errorLabel() const;
0054 
0055     /**
0056      * Sets \p text as text of the label and \p accessibleName as alternative
0057      * text for assistive tools. If \p accessibleName is empty, then \p text is
0058      * used instead. Both texts must be plain text.
0059      *
0060      * Note: If input is required, then the label is annotated appropriately.
0061      */
0062     void setLabelText(const QString &text, const QString &accessibleName = {});
0063 
0064     /**
0065      * Sets \p text as hint text for this input field and \p accessibleDescription
0066      * as alternative text for assistive tools. If \p accessibleDescription is
0067      * empty, then \p text is used instead.  Both texts must be plain text.
0068      */
0069     void setHint(const QString &text, const QString &accessibleDescription = {});
0070 
0071     /**
0072      * Marks this input field as required.
0073      */
0074     void setIsRequired(bool required);
0075 
0076     /**
0077      * Returns \c true, if this field needs to be filled out.
0078      */
0079     bool isRequired() const;
0080 
0081     /**
0082      * Sets the validator to use for validating the input.
0083      *
0084      * Note: If you wrap a QLineEdit, then do not set a validator (or an input mask)
0085      *       on it because this will break the correct displaying of the error message.
0086      */
0087     void setValidator(const std::shared_ptr<QValidator> &validator);
0088 
0089     /**
0090      * Sets \p text as error message to display if a value is required for the
0091      * input field, but if no value has been entered. If \p text is empty, then
0092      * a default message will be used. Both texts must be plain text.
0093      * The optional \p accessibleText is used as alternative text for assistive
0094      * tools.
0095      */
0096     void setValueRequiredErrorMessage(const QString &text, const QString &accessibleText = {});
0097 
0098     /**
0099      * Sets \p text as error message to display if the entered value is not accepted
0100      * by the validator. If \p text is empty, then a default message will be used.
0101      * The optional \p accessibleText is used as alternative text for assistive
0102      * tools. Both texts must be plain text.
0103      */
0104     void setInvalidEntryErrorMessage(const QString &text, const QString &accessibleText = {});
0105 
0106     /**
0107      * Sets the tool tip of the controlled widget and its associated label.
0108      */
0109     void setToolTip(const QString &toolTip);
0110 
0111     /**
0112      * Enables or disables the controlled widget and its associated label.
0113      * If the widget is disables, then the error label is hidden. Otherwise,
0114      * the error label is shown if there is an error.
0115      */
0116     void setEnabled(bool enabled);
0117 
0118     /**
0119      * Returns the currently shown error message for this input field.
0120      */
0121     QString currentError() const;
0122 
0123     /**
0124      * Returns \c true, if the input has a value. This function is used to
0125      * check required input fields for non-empty user input.
0126      * Needs to be implemented for concrete widget classes.
0127      * \sa validate
0128      */
0129     virtual bool hasValue() const = 0;
0130 
0131     /**
0132      * Returns \c true, if the input satisfies the validator.
0133      * Needs to be implemented for concrete widget classes.
0134      * \sa validate
0135      */
0136     virtual bool hasAcceptableInput() const = 0;
0137 
0138 protected:
0139     /**
0140      * Connects the slots \ref onTextChanged and \ref onEditingFinished to the
0141      * corresponding signal of the controlled widget.
0142      * Needs to be implemented for concrete widget classes.
0143      */
0144     virtual void connectWidget() = 0;
0145 
0146     /**
0147      * Sets the controlled widget and creates the associated labels.
0148      */
0149     void setWidget(QWidget *widget);
0150 
0151     /**
0152      * Returns the controlled widget.
0153      */
0154     QWidget *widget() const;
0155 
0156     /**
0157      * Validates \p text with the validator. Should be used when implementing
0158      * \ref hasAcceptableInput.
0159      */
0160     bool validate(const QString &text, int pos) const;
0161 
0162     /**
0163      * This slot needs to be connected to a signal of the controlled widget
0164      * that is emitted when the text changes like \ref QLineEdit::textChanged.
0165      * \sa connectWidget
0166      */
0167     void onTextChanged();
0168 
0169     /**
0170      * This slot needs to be connected to a signal of the controlled widget
0171      * that is emitted when the widget loses focus (or some user interaction
0172      * signals that they want to commit the entered text) like
0173      * \ref QLineEdit::editingFinished.
0174      * \sa connectWidget
0175      */
0176     void onEditingFinished();
0177 
0178 private:
0179     class Private;
0180     const std::unique_ptr<Private> d;
0181 };
0182 }
0183 
0184 /**
0185  * FormTextInput is a class for simplifying the management of text input widgets
0186  * like QLineEdit or QTextEdit with associated label and error message for usage
0187  * in form-like dialogs.
0188  *
0189  * Usage hints:
0190  * * If you wrap a QLineEdit, then do not set a validator (or an input mask)
0191  *   on it. Instead set the validator on this class.
0192  *   If you set a validator on the QLineEdit, then showing the error message
0193  *   when editing is finished does not work because QLineEdit doesn't emit the
0194  *   editingFinished() signal if the input is not acceptable.
0195  */
0196 template<class Widget>
0197 class FormTextInput : public _detail::FormTextInputBase
0198 {
0199     /**
0200      * Use \ref create to create a new instance.
0201      */
0202     FormTextInput() = default;
0203 
0204 public:
0205     /**
0206      * Creates a new instance of this class with a new instance of \p Widget.
0207      */
0208     static auto create(QWidget *parent)
0209     {
0210         std::unique_ptr<FormTextInput> self{new FormTextInput};
0211         self->setWidget(new Widget{parent});
0212         return self;
0213     }
0214 
0215     /**
0216      * Returns the controlled widget.
0217      */
0218     Widget *widget() const
0219     {
0220         return static_cast<Widget *>(FormTextInputBase::widget());
0221     }
0222 
0223     bool hasValue() const override;
0224 
0225     bool hasAcceptableInput() const override;
0226 
0227 private:
0228     void connectWidget() override;
0229 };
0230 
0231 template<>
0232 bool FormTextInput<QLineEdit>::hasValue() const;
0233 
0234 template<>
0235 bool FormTextInput<QLineEdit>::hasAcceptableInput() const;
0236 
0237 template<>
0238 void FormTextInput<QLineEdit>::connectWidget();
0239 
0240 }