Bug 442903 - "kdevcustomscript||GNU_indent_GNU" is stored by default if the user selects no source formatter
Summary: "kdevcustomscript||GNU_indent_GNU" is stored by default if the user selects n...
Status: RESOLVED FIXED
Alias: None
Product: kdevplatform
Classification: Developer tools
Component: sourceformatter (show other bugs)
Version: git master
Platform: Compiled Sources Linux
: NOR normal
Target Milestone: ---
Assignee: Igor Kushnir
URL:
Keywords:
Depends on:
Blocks:
 
Reported: 2021-09-24 17:07 UTC by Igor Kushnir
Modified: 2023-09-15 11:26 UTC (History)
0 users

See Also:
Latest Commit:
Version Fixed In: 5.13.231200
Sentry Crash Report:


Attachments

Note You need to log in before you can comment on or make changes to this bug.
Description Igor Kushnir 2021-09-24 17:07:21 UTC
SUMMARY
In a new session, when the user navigates to Source Formatter tab of KDevelop settings, doesn't change anything and presses the OK button, KDevelop stores the following in this session's sessionrc (sans the x-python[3] entries when kdev-python is not installed):

[SourceFormatter]
ModelinesEnabled=false
OverrideKateIndentation=false
text/x-c++hdr=kdevcustomscript||GNU_indent_GNU
text/x-c++src=kdevcustomscript||GNU_indent_GNU
text/x-chdr=kdevcustomscript||GNU_indent_GNU
text/x-csharp=kdevcustomscript||GNU_indent_GNU
text/x-csrc=kdevcustomscript||GNU_indent_GNU
text/x-java=kdevcustomscript||GNU_indent_GNU
text/x-objcsrc=kdevcustomscript||GNU_indent_GNU
text/x-python=kdevcustomscript||GNU_indent_GNU
text/x-python3=kdevcustomscript||GNU_indent_GNU

kdevcustomscript||GNU_indent_GNU is stored despite the fact that Artistic Style||1TBS is initially selected in the UI for each language. The bug is particularly glaring for Python[ 3] languages and x-python[3] MIME types: only kdev-python's autopep8 style supports these languages and MIME types. The Source Formatter configuration UI correctly shows autopep8 as the only style applicable to the Python languages.

STEPS TO REPRODUCE
1. Create a new session in KDevelop.
2. (optional, if kdev-python is installed) Open a Python source file (*.py) so that the Python languages are selectable in the Language combobox of the Source Formatter settings tab.
3. Open the Source Formatter tab of KDevelop settings.
4. Click OK.
5. Quit KDevelop.
6. Locate the new session's sessionrc. For example like this:
    cd ~/.local/share/kdevelop/sessions
    grep -rI "<the new session's name>" .
7. Find the [SourceFormatter] section in the sessionrc file.

OBSERVED RESULT
See the [SourceFormatter] section in the SUMMARY of this bug report.

EXPECTED RESULT
1. autopep8 rather than GNU_indent_GNU should be autoconfigured for the Python languages.

2. In this case the user has probably decided not to configure Source Formatter at the moment. The simplest option is to store nothing in sessionrc. A more user-friendly solution is to warn the user about the likely mistake and show the formatters+styles selected for each language in a dialog. The dialog could offer the following options/buttons:
  * go back to the configuration page;
  * don't change the configuration at all;
  * store the settings as "configured".
The dialog could also show a short help about the Source Formatter configuration page to prevent issues like Bug 358798 (I was also thoroughly confused by the UI at first). Plus the dialog could emphasize the importance of configuring the right source formatter for each relevant language: not only it is used by the Reformat Source/Line/Files actions, but also to format auto-generated or auto-refactored or auto-fixed code.

SOFTWARE/OS VERSIONS
Manjaro GNU/Linux, Xfce
KDE Frameworks Version: 5.85.0
Qt Version: 5.15.2+kde+r222
Comment 1 Igor Kushnir 2021-12-19 11:39:33 UTC
> kdevcustomscript||GNU_indent_GNU is stored despite the fact that Artistic Style||1TBS is initially selected in the UI for each language.
This statement is wrong. When Artistic Style||1TBS is initially selected in the UI for each non-Python language, the following is stored in the session's sessionrc:

[SourceFormatter]
ModelinesEnabled=false
OverrideKateIndentation=false
text/x-c++hdr=kdevastyle||1TBS
text/x-c++src=kdevastyle||1TBS
text/x-chdr=kdevastyle||1TBS
text/x-csharp=kdevastyle||1TBS
text/x-csrc=kdevastyle||1TBS
text/x-java=kdevastyle||1TBS
text/x-objcsrc=kdevastyle||1TBS
text/x-python=kdevcustomscript||GNU_indent_GNU
text/x-python3=kdevcustomscript||GNU_indent_GNU

I was mistaken because of another UI bug: the order of the formatter plugins in the Formatter combobox can change each time Configure KDevelop... dialog is opened. The first formatter in the combobox is preselected.
Comment 2 Igor Kushnir 2023-07-10 12:17:18 UTC
Git commit de70d41429d9ea54fbecc59a6e79076a2dab9871 by Igor Kushnir.
Committed on 06/06/2023 at 13:45.
Pushed by igorkushnir into branch 'fix-and-refactor-formatter-selection-edit'.

Overhaul SourceFormatterSelectionEdit

Don't select the first style automatically. The first style in the list
is not likely to match the user's preferred style. Let the user select a
style for each language [s]he cares about. Leave the other languages
with no selected style. Delete config entries of MIME types that belong
to languages with no selected style when settings are saved.

There were several issues with the old behavior replaced in this commit:
1. If the user didn't explicitly select a formatter or a style for a
   language, the first formatter in the list and its first style were
   preselected and stored in config for that language when settings were
   saved. Worse still, the GNU_indent_GNU style was stored for Python
   and Python 3 languages, even though this style does not support the
   Python languages and cannot be selected in the UI (Bug 442903).
2. It was possible to unselect a style in the UI via Ctrl+Click on the
   selected style. But SourceFormatterSelectionEdit completely ignored
   such deselection. The last deselected style was considered selected
   because QListWidget::currentItem() returned it. Now there is a
   well-defined no-style-selected state because:
   * QItemSelectionModel::selectedIndexes() is used instead of
     QListWidget::currentItem();
   * &QListWidget::itemSelectionChanged signal is connected to instead
     of &QListWidget::currentRowChanged.
3. The Apply button became enabled when language selection changed, even
   though merely selecting a different language does not modify
   formatting configuration. This behavior could reinforce possible user
   misunderstanding of the configuration: that selecting a language
   assigns it to an open project (Bug 358798).
4. The order of plugins in the Formatter combobox was not fixed, because
   LanguageSettings::formatters was an unordered QSet. So each time the
   Configure KDevelop... dialog was opened, the first (and preselected)
   item was unpredictably Artistic Style or Custom Script Formatter.

Invisible bugs fixed in this commit:
1. QMap::iterator::value() was called on the iterator already erased
   from QMap SourceFormatterSelectionEditPrivate::formatters in
   SourceFormatterSelectionEdit::removeSourceFormatter(). This was
   undefined behavior, which hasn't revealed itself yet.
2. SourceFormatterSelectionEdit::deleteStyle() erased the current style
   from StyleMap SourceFormatter::styles, but failed to destroy the
   SourceFormatterStyle object. This was a memory leak.

Error-prone manual memory management was eliminated:
1. SourceFormatter struct contained a QMap with mapped_type = owning raw
   SourceFormatterStyle pointer, which had to be deleted manually. Now
   FormatterData class contains a std::map with mapped_type =
   SourceFormatterStyle value.
2. Similarly, SourceFormatterSelectionEditPrivate::formatters was a
   QMap<QString, SourceFormatter*> with owning raw pointers. Now it is a
   std::vector<std::unique_ptr<FormatterData>>.

Optimizations:
1. LanguageSettings::mimetypes contained many duplicates because every
   MIME type of each supporting style was unconditionally appended to
   this list. SourceFormatterSelectionEdit::loadSettings() read and
   SourceFormatterSelectionEdit::saveSettings() wrote a config entry for
   each MIME type in this list. Now duplicate MIME types are not added
   to LanguageSettings::m_mimeTypes.
2. Most styles return the same built-in list from
   SourceFormatterStyle::mimeTypes(). Cache already encountered MIME
   lists and don't process duplicates. This optimization requires newly
   implemented operator== for SourceFormatterStyle::MimeHighlightPair.
3. Widgets' signals weren't blocked consistently. So the same code often
   run repeatedly and redundantly. For example, when
   SourceFormatterSelectionEdit::selectStyle() called
   QListWidget::setCurrentRow(), and the row actually changed,
   selectStyle() was called again. Infinite recursion was prevented by
   an early return when the current index did not change inside
   QItemSelectionModel::setCurrentIndex().
4. updatePreview(), enableStyleButtons() and changed() were often called
   repeatedly and redundantly when SourceFormatterSelectionEdit's public
   member functions invoked other public member functions.
5. Store addresses rather than names of formatters and styles as user
   data in QComboBox and QListWidget. The pointer stability of these
   FormatterData and SourceFormatterStyle objects is already necessary
   and guaranteed to prevent invalidating selected formatter and style
   pointers. Storing pointers replaces name lookup (search) with direct
   object access.
6. Use std::vector instead of QMap, QSet and QList to store small (< 10)
   numbers of elements. The impact of this container choice on code
   complexity is ambiguous (it has both pros and cons), but it is surely
   more efficient.

Other improvements:
1. Hide ui.descriptionLabel if the selected style's description is empty
   (which is the case for most styles). This leaves more vertical space
   to the style preview widget.
2. Don't disable Language and Formatter comboboxes and the Styles list
   widget when they are empty. The user cannot do anything with these
   empty widgets anyway. The code is shorter and less error-prone
   without the disabling and enabling.
3. Cache the currently selected language in
   SourceFormatterSelectionEditPrivate::currentLanguagePtr. The current
   language rarely changes, so the cache requires little supporting
   code. This data member cache allows to call currentLanguage()
   whenever needed instead of caching in a local variable as before:
   LanguageSettings& l = d->languages[d->ui.cbLanguages->currentText()];
4. Extract reading {MIME type => selected formatter and style} config
   entries into a new class SourceFormatter::ConfigForMimeType and use
   it in SourceFormatterController and SourceFormatterSelectionEdit
   instead of duplicating the parsing code. ConfigForMimeType's extra
   validation made the parsing stricter than it was in
   SourceFormatterSelectionEdit, which should be fine: better to ignore
   broken config entries than try to use them somehow.
5. If the style stored in the config for a language does not support
   this language, don't select it and print a warning.
6. qCDebug => qCWarning when a config entry is broken. Make these
   messages more informative.
7. Add assertions to help detect bugs sooner.
8. Add code documentation and comments.
FIXED-IN: 5.8.220400

M  +8    -0    kdevplatform/interfaces/isourceformatter.h
M  +1    -0    kdevplatform/shell/CMakeLists.txt
A  +106  -0    kdevplatform/shell/sourceformatterconfig.h     [License: LGPL(v2.0+)]
M  +37   -32   kdevplatform/shell/sourceformattercontroller.cpp
M  +9    -22   kdevplatform/shell/sourceformattercontroller.h
M  +775  -369  kdevplatform/shell/sourceformatterselectionedit.cpp
M  +2    -7    kdevplatform/shell/sourceformatterselectionedit.h
M  +1    -1    kdevplatform/shell/sourceformatterselectionedit.ui

https://invent.kde.org/kdevelop/kdevelop/-/commit/de70d41429d9ea54fbecc59a6e79076a2dab9871
Comment 3 Igor Kushnir 2023-09-15 11:26:27 UTC
Git commit 1b60b1dd18fb8f26c451c4ede77011e13aa7fcac by Igor Kushnir.
Committed on 15/09/2023 at 11:57.
Pushed by igorkushnir into branch 'master'.

Overhaul SourceFormatterSelectionEdit

Don't select the first style automatically. The first style in the list
is not likely to match the user's preferred style. Let the user select a
style for each language [s]he cares about. Leave the other languages
with no selected style. Delete config entries of MIME types that belong
to languages with no selected style when settings are saved.

There were several issues with the old behavior replaced in this commit:
1. If the user didn't explicitly select a formatter or a style for a
   language, the first formatter in the list and its first style were
   preselected and stored in config for that language when settings were
   saved. Worse still, the GNU_indent_GNU style was stored for Python
   and Python 3 languages, even though this style does not support the
   Python languages and cannot be selected in the UI (Bug 442903).
2. It was possible to unselect a style in the UI via Ctrl+Click on the
   selected style. But SourceFormatterSelectionEdit completely ignored
   such deselection. The last deselected style was considered selected
   because QListWidget::currentItem() returned it. Now there is a
   well-defined no-style-selected state because:
   * QItemSelectionModel::selectedIndexes() is used instead of
     QListWidget::currentItem();
   * &QListWidget::itemSelectionChanged signal is connected to instead
     of &QListWidget::currentRowChanged.
3. The Apply button became enabled when language selection changed, even
   though merely selecting a different language does not modify
   formatting configuration. This behavior could reinforce possible user
   misunderstanding of the configuration: that selecting a language
   assigns it to an open project (Bug 358798).
4. The order of plugins in the Formatter combobox was not fixed, because
   LanguageSettings::formatters was an unordered QSet. So each time the
   Configure KDevelop... dialog was opened, the first (and preselected)
   item was unpredictably Artistic Style or Custom Script Formatter.

Invisible bugs fixed in this commit:
1. QMap::iterator::value() was called on the iterator already erased
   from QMap SourceFormatterSelectionEditPrivate::formatters in
   SourceFormatterSelectionEdit::removeSourceFormatter(). This was
   undefined behavior, which hasn't revealed itself yet.
2. SourceFormatterSelectionEdit::deleteStyle() erased the current style
   from StyleMap SourceFormatter::styles, but failed to destroy the
   SourceFormatterStyle object. This was a memory leak.

Error-prone manual memory management was eliminated:
1. SourceFormatter struct contained a QMap with mapped_type = owning raw
   SourceFormatterStyle pointer, which had to be deleted manually. Now
   FormatterData class contains a std::map with mapped_type =
   SourceFormatterStyle value.
2. Similarly, SourceFormatterSelectionEditPrivate::formatters was a
   QMap<QString, SourceFormatter*> with owning raw pointers. Now it is a
   std::vector<std::unique_ptr<FormatterData>>.

Optimizations:
1. LanguageSettings::mimetypes contained many duplicates because every
   MIME type of each supporting style was unconditionally appended to
   this list. SourceFormatterSelectionEdit::loadSettings() read and
   SourceFormatterSelectionEdit::saveSettings() wrote a config entry for
   each MIME type in this list. Now duplicate MIME types are not added
   to LanguageSettings::m_mimeTypes.
2. Most styles return the same built-in list from
   SourceFormatterStyle::mimeTypes(). Cache already encountered MIME
   lists and don't process duplicates. This optimization requires newly
   implemented operator== for SourceFormatterStyle::MimeHighlightPair.
3. Widgets' signals weren't blocked consistently. So the same code often
   run repeatedly and redundantly. For example, when
   SourceFormatterSelectionEdit::selectStyle() called
   QListWidget::setCurrentRow(), and the row actually changed,
   selectStyle() was called again. Infinite recursion was prevented by
   an early return when the current index did not change inside
   QItemSelectionModel::setCurrentIndex().
4. updatePreview(), enableStyleButtons() and changed() were often called
   repeatedly and redundantly when SourceFormatterSelectionEdit's public
   member functions invoked other public member functions.
5. Store addresses rather than names of formatters and styles as user
   data in QComboBox and QListWidget. The pointer stability of these
   FormatterData and SourceFormatterStyle objects is already necessary
   and guaranteed to prevent invalidating selected formatter and style
   pointers. Storing pointers replaces name lookup (search) with direct
   object access.
6. Use std::vector instead of QMap, QSet and QList to store small (< 10)
   numbers of elements. The impact of this container choice on code
   complexity is ambiguous (it has both pros and cons), but it is surely
   more efficient.

Other improvements:
1. Hide ui.descriptionLabel if the selected style's description is empty
   (which is the case for most styles). This leaves more vertical space
   to the style preview widget.
2. Don't disable Language and Formatter comboboxes and the Styles list
   widget when they are empty. The user cannot do anything with these
   empty widgets anyway. The code is shorter and less error-prone
   without the disabling and enabling.
3. Cache the currently selected language in
   SourceFormatterSelectionEditPrivate::currentLanguagePtr. The current
   language rarely changes, so the cache requires little supporting
   code. This data member cache allows to call currentLanguage()
   whenever needed instead of caching in a local variable as before:
   LanguageSettings& l = d->languages[d->ui.cbLanguages->currentText()];
4. Extract reading {MIME type => selected formatter and style} config
   entries into a new class SourceFormatter::ConfigForMimeType and use
   it in SourceFormatterController and SourceFormatterSelectionEdit
   instead of duplicating the parsing code. ConfigForMimeType's extra
   validation made the parsing stricter than it was in
   SourceFormatterSelectionEdit, which should be fine: better to ignore
   broken config entries than try to use them somehow.
5. If the style stored in the config for a language does not support
   this language, don't select it and print a warning.
6. qCDebug => qCWarning when a config entry is broken. Make these
   messages more informative.
7. Add assertions to help detect bugs sooner.
8. Add code documentation and comments.
FIXED-IN: 5.13.231200

M  +8    -0    kdevplatform/interfaces/isourceformatter.h
M  +1    -0    kdevplatform/shell/CMakeLists.txt
A  +106  -0    kdevplatform/shell/sourceformatterconfig.h     [License: LGPL(v2.0+)]
M  +37   -32   kdevplatform/shell/sourceformattercontroller.cpp
M  +9    -22   kdevplatform/shell/sourceformattercontroller.h
M  +775  -369  kdevplatform/shell/sourceformatterselectionedit.cpp
M  +2    -7    kdevplatform/shell/sourceformatterselectionedit.h
M  +1    -1    kdevplatform/shell/sourceformatterselectionedit.ui

https://invent.kde.org/kdevelop/kdevelop/-/commit/1b60b1dd18fb8f26c451c4ede77011e13aa7fcac