diff options
| -rw-r--r-- | .pylintrc | 2 | ||||
| -rw-r--r-- | INSTALL.md | 1 | ||||
| -rw-r--r-- | bitbucket-pipelines.yml | 4 | ||||
| -rwxr-xr-x | scripts/qrc_lint.py | 97 | ||||
| -rw-r--r-- | src/application.cpp | 2 | ||||
| -rw-r--r-- | src/rekonq.kcfg | 3 | ||||
| -rw-r--r-- | src/settings/test/test_settings.cpp | 8 | ||||
| -rw-r--r-- | src/settings/test/test_settings.qrc | 10 | ||||
| -rw-r--r-- | third-party/resources.qrc | 20 | ||||
| -rw-r--r-- | third-party/tabler-icons.theme | 5 | 
10 files changed, 134 insertions, 18 deletions
| @@ -27,7 +27,7 @@ extension-pkg-allow-list=  # be loaded. Extensions are loading into the active Python interpreter and may  # run arbitrary code. (This is an alternative name to extension-pkg-allow-list  # for backward compatibility.) -extension-pkg-whitelist= +extension-pkg-whitelist=lxml  # Return non-zero exit code if any of these messages/categories are detected,  # even if score is above --fail-under value. Syntax same as enable. Messages @@ -16,6 +16,7 @@ To build, rekonq needs at least:  - gtest and gmock  - gcov, lcov  - clang-tidy +- python-lxml for some of the scripts  ## Building rekonq diff --git a/bitbucket-pipelines.yml b/bitbucket-pipelines.yml index 539eee04..1c2cf424 100644 --- a/bitbucket-pipelines.yml +++ b/bitbucket-pipelines.yml @@ -10,8 +10,9 @@ pipelines:                - python --version                - pylint scripts/check_license.py                - pylint scripts/rekonf.py +              - pylint scripts/qrc_lint.py          - step: -            name: license headers check +            name: Static checks              runs-on:                - 'self.hosted'                - 'linux.shell' @@ -19,6 +20,7 @@ pipelines:                - git submodule update --init                - cmake --preset debug -GNinja                - ninja -C cmake-build-debug rekonq_check_license +              - git ls-files | grep '\.qrc$' | xargs -n1 ./scripts/qrc_lint.py          - step:              name: Debug Build Test              runs-on: diff --git a/scripts/qrc_lint.py b/scripts/qrc_lint.py new file mode 100755 index 00000000..9520f22a --- /dev/null +++ b/scripts/qrc_lint.py @@ -0,0 +1,97 @@ +#!/usr/bin/env python3 +# ============================================================ +#     The rekonq project +# ============================================================ +# SPDX-License-Identifier: GPL-3.0-only +# Copyright (C) 2022 aqua <aqua@iserlohn-fortress.net> +# ============================================================ +""" Lint qrc files """ + +import argparse +import sys +from os.path import (dirname, exists, join) +from lxml import etree + + +def lint(dir_name: str, tree: etree._ElementTree) -> bool: +    """ Lint an etree and return if errors were found """ +    has_check_errors = False +    root = tree.getroot() + +    # check doctype +    if tree.docinfo.doctype != '<!DOCTYPE RCC>': +        print(f'unknown doctype { tree.docinfo.doctype }') +        has_check_errors = True + +    qresources = root.findall('qresource') +    # check if qresource's have prefix +    for qres in qresources: +        if qres.get('prefix') is None: +            print('qresource without prefix') +            has_check_errors = True + +    # check if qresource's are sorted by prefix +    sortedqres = sorted(qresources, key=lambda x: x.get('prefix')) +    if qresources != sortedqres: +        print('qresources are not sorted') +        has_check_errors = True + +    for qres in sortedqres: +        files = qres.findall('file') + +        # check if files are sorted +        if files != sorted(files, key=lambda x: x.get('alias')): +            print(f'qresources prefix={ qres.get("prefix") }:\tfiles are not sorted') +            has_check_errors = True + +        # check for missing files and missing alias +        for file in files: +            if file.get('alias') is None: +                print(f'file path={ file.text }:\tmissing file alias') +                has_check_errors = True +            path = join(dir_name, file.text) +            if not exists(path): +                print(f'file path={ path }:\tfile does not exist') +                has_check_errors = True + +    return has_check_errors + + +def regen(root: etree.Element, outfile: str): +    """ Regenerate an etree """ +    out_root = etree.Element('RCC', {'version': '1.0'}) +    for qresource in sorted(root.findall('qresource'), key=lambda x: x.get('prefix')): +        out_qres = etree.SubElement(out_root, 'qresource', {'prefix': qresource.get('prefix')}) + +        for file in sorted(qresource.findall('file'), key=lambda x: x.get('alias')): +            out_file = etree.SubElement(out_qres, 'file', {'alias': file.get('alias')}) +            out_file.text = file.text + +    with etree.xmlfile(outfile, encoding='utf-8', close=True) as xml_file: +        xml_file.write_declaration(standalone=True) +        xml_file.write_doctype('<!DOCTYPE RCC>') +        xml_file.write(out_root, pretty_print=True) + + +def main(): +    """ Main function """ +    parser = argparse.ArgumentParser(description='Lint qrc file') +    parser.add_argument('file', type=str, help='qrc file') +    parser.add_argument('--regen', action='store_true', help='Regenerate qrc file') +    args = parser.parse_args() + +    if not exists(args.file): +        sys.exit('input file "{args.file}" doesn\'t exist') +    else: +        print(args.file) + +    tree = etree.parse(args.file) + +    if args.regen: +        regen(tree.getroot(), args.file) +    elif lint(dirname(args.file), tree): +        sys.exit('Errors during lint') + + +if __name__ == '__main__': +    main() diff --git a/src/application.cpp b/src/application.cpp index 228b3d38..aba1fc69 100644 --- a/src/application.cpp +++ b/src/application.cpp @@ -51,7 +51,7 @@ Application::Application(int &argc, char *argv[]) : SingleApplication(argc, argv    }    // setWindowIcon(KIcon("rekonq")); -  if (const auto iconTheme = m_settings->value("IconTheme"); iconTheme.isValid()) { +  if (const auto iconTheme = m_settings->value("Appearance/IconTheme"); iconTheme.isValid()) {      spdlog::info("Set icon theme {}", qUtf8Printable(iconTheme.toString()));      QIcon::setThemeName(iconTheme.toString());    } diff --git a/src/rekonq.kcfg b/src/rekonq.kcfg index 0de9cd44..0b23782e 100644 --- a/src/rekonq.kcfg +++ b/src/rekonq.kcfg @@ -49,6 +49,9 @@  <!-- Fonts Settings -->  <group name="Appearance"> +    <entry name="Icon Theme" key="IconTheme" type="String"> +        <default>tabler</default> +    </entry>      <entry name="Standard Font Family" key="standardFontFamily" type="Font">          <default code="true">getFont(QFont::AnyStyle)</default>      </entry> diff --git a/src/settings/test/test_settings.cpp b/src/settings/test/test_settings.cpp index babbcdb6..24bbf349 100644 --- a/src/settings/test/test_settings.cpp +++ b/src/settings/test/test_settings.cpp @@ -76,14 +76,16 @@ TEST(settings, Settings)  TEST(settings, SettingsDialog)  { +  constexpr unsigned n_settings = 36; // there are 36 settings in total    MockSettings mockSettings;    // There are 4 groups in total, but General should not be calling beginGroup/endGroup    // beginGroup/endGroup are called twice: during the ctor and when accepted    EXPECT_CALL(mockSettings, beginGroup).Times(3 * 4);    EXPECT_CALL(mockSettings, endGroup).Times(3 * 4); -  // There are 35 settings in total, one of which is hidden and won't be set by the dialog -  EXPECT_CALL(mockSettings, value).Times(35 * 2).WillRepeatedly(ReturnArg<1>()); -  EXPECT_CALL(mockSettings, setValue(_, _)).Times(33 + 34); +  EXPECT_CALL(mockSettings, value).Times(n_settings * 2).WillRepeatedly(ReturnArg<1>()); +  // 1 setting is hidden and won't be set by the dialog +  // save and reset will both call setValue on all non-hidden settings +  EXPECT_CALL(mockSettings, setValue(_, _)).Times(n_settings * 2 - 3);    EXPECT_CALL(mockSettings, setValue(QStringEq("homepage"), QVariantEq("about:blank")));    // change setting diff --git a/src/settings/test/test_settings.qrc b/src/settings/test/test_settings.qrc index deaf088f..16e74fef 100644 --- a/src/settings/test/test_settings.qrc +++ b/src/settings/test/test_settings.qrc @@ -1,5 +1,7 @@ -<!DOCTYPE RCC><RCC version="1.0"> -    <qresource> -        <file>rekonqrc</file> -    </qresource> +<?xml version='1.0' encoding='utf-8' standalone='yes'?> +<!DOCTYPE RCC> +<RCC version="1.0"> +  <qresource prefix="/"> +    <file alias="rekonqrc">rekonqrc</file> +  </qresource>  </RCC> diff --git a/third-party/resources.qrc b/third-party/resources.qrc index f692daee..49496028 100644 --- a/third-party/resources.qrc +++ b/third-party/resources.qrc @@ -5,27 +5,31 @@      <file alias="index.theme">tabler-icons.theme</file>    </qresource>    <qresource prefix="/icons/tabler/actions/scalable"> -    <file alias="edit-clear.svg">tabler-icons/icons/backspace.svg</file> +    <file alias="arrow-right.svg">tabler-icons/icons/chevron-right.svg</file> +    <file alias="bookmark-new.svg">tabler-icons/icons/star.svg</file>      <file alias="document-new.svg">tabler-icons/icons/browser-plus.svg</file> -    <file alias="go-previous.svg">tabler-icons/icons/chevron-left.svg</file> -    <file alias="go-next.svg">tabler-icons/icons/chevron-right.svg</file> -    <file alias="go-jump.svg">tabler-icons/icons/chevrons-right.svg</file> +    <file alias="edit-clear.svg">tabler-icons/icons/backspace.svg</file> +    <file alias="edit-delete.svg">tabler-icons/icons/trash.svg</file>      <file alias="folder-new.svg">tabler-icons/icons/folder-plus.svg</file>      <file alias="go-home.svg">tabler-icons/icons/home.svg</file> +    <file alias="go-jump.svg">tabler-icons/icons/arrow-right.svg</file> +    <file alias="go-next.svg">tabler-icons/icons/arrow-right.svg</file> +    <file alias="go-previous.svg">tabler-icons/icons/arrow-left.svg</file>      <file alias="view-refresh.svg">tabler-icons/icons/refresh.svg</file> -    <file alias="bookmark-new.svg">tabler-icons/icons/star.svg</file> -    <file alias="edit-delete.svg">tabler-icons/icons/trash.svg</file>      <file alias="window-close.svg">tabler-icons/icons/x.svg</file>    </qresource> +  <qresource prefix="/icons/tabler/mimetypes/scalable"> +    <file alias="text-html.svg">tabler-icons/icons/file-code.svg</file> +  </qresource>    <qresource prefix="/icons/tabler/places/scalable">      <file alias="bookmark.svg">tabler-icons/icons/bookmark.svg</file>      <file alias="bookmarks.svg">tabler-icons/icons/bookmarks.svg</file>      <file alias="browser.svg">tabler-icons/icons/browser.svg</file>      <file alias="downloads.svg">tabler-icons/icons/download.svg</file>      <file alias="folder.svg">tabler-icons/icons/folder.svg</file> -    <file alias="shortcuts.svg">tabler-icons/icons/keyboard.svg</file> +    <file alias="plugin.svg">tabler-icons/icons/tool.svg</file>      <file alias="plugins.svg">tabler-icons/icons/puzzle.svg</file>      <file alias="preferences.svg">tabler-icons/icons/settings.svg</file> -    <file alias="plugin.svg">tabler-icons/icons/tool.svg</file> +    <file alias="shortcuts.svg">tabler-icons/icons/keyboard.svg</file>    </qresource>  </RCC> diff --git a/third-party/tabler-icons.theme b/third-party/tabler-icons.theme index 88941053..8d754e3d 100644 --- a/third-party/tabler-icons.theme +++ b/third-party/tabler-icons.theme @@ -7,6 +7,11 @@ Size=24  Context=Actions  Type=Scalable +[mimetypes/scalable] +Size=24 +Context=Mimetypes +Type=Scalable +  [places/scalable]  Size=24  Context=Places | 
