595 lines
28 KiB
Python
595 lines
28 KiB
Python
"""A PyQT4 dialog to enter app settings"""
|
|
#
|
|
# Copyright 2012-2014 Anthony Beville
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
from __future__ import annotations
|
|
|
|
import html
|
|
import logging
|
|
import os
|
|
import pathlib
|
|
import platform
|
|
import webbrowser
|
|
from urllib.parse import urlparse
|
|
|
|
from PyQt5 import QtCore, QtGui, QtWidgets, uic
|
|
|
|
import comictalker.comictalkerapi as ct_api
|
|
from comicapi import utils
|
|
from comicapi.genericmetadata import md_test
|
|
from comictaggerlib.filerenamer import FileRenamer
|
|
from comictaggerlib.imagefetcher import ImageFetcher
|
|
from comictaggerlib.settings import ComicTaggerSettings
|
|
from comictaggerlib.ui import ui_path
|
|
from comictalker.comiccacher import ComicCacher
|
|
from comictalker.talkerbase import ComicTalker
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
windowsRarHelp = """
|
|
<html><head/><body><p>To write to CBR/RAR archives,
|
|
you will need to have the tools from
|
|
<span style=" text-decoration: underline; color:#0000ff;">
|
|
<a href="http://www.win-rar.com/download.html">WINRar</a></span>
|
|
installed. (ComicTagger only uses the command-line rar tool.)
|
|
</p></body></html>
|
|
"""
|
|
|
|
linuxRarHelp = """
|
|
<html><head/><body><p>To write to CBR/RAR archives,
|
|
you will need to have the shareware rar tool from RARLab installed.
|
|
Your package manager should have rar (e.g. "apt-get install rar"). If not, download it
|
|
<span style=" text-decoration: underline; color:#0000ff;">
|
|
<a href="https://www.rarlab.com/download.htm">here</a></span>,
|
|
and install in your path. </p></body></html>
|
|
"""
|
|
|
|
macRarHelp = """
|
|
<html><head/><body><p>To write to CBR/RAR archives,
|
|
you will need the rar tool. The easiest way to get this is
|
|
to install <span style=" text-decoration: underline; color:#0000ff;">
|
|
<a href="https://brew.sh/">homebrew</a></span>.
|
|
</p>Once homebrew is installed, run: <b>brew install caskroom/cask/rar</b></body></html>
|
|
"""
|
|
|
|
|
|
template_tooltip = """
|
|
The template for the new filename. Uses python format strings https://docs.python.org/3/library/string.html#format-string-syntax
|
|
Accepts the following variables:
|
|
{is_empty} (boolean)
|
|
{tag_origin} (string)
|
|
{series} (string)
|
|
{issue} (string)
|
|
{title} (string)
|
|
{publisher} (string)
|
|
{month} (integer)
|
|
{year} (integer)
|
|
{day} (integer)
|
|
{issue_count} (integer)
|
|
{volume} (integer)
|
|
{genre} (string)
|
|
{language} (string)
|
|
{comments} (string)
|
|
{volume_count} (integer)
|
|
{critical_rating} (float)
|
|
{country} (string)
|
|
{alternate_series} (string)
|
|
{alternate_number} (string)
|
|
{alternate_count} (integer)
|
|
{imprint} (string)
|
|
{notes} (string)
|
|
{web_link} (string)
|
|
{format} (string)
|
|
{manga} (string)
|
|
{black_and_white} (boolean)
|
|
{page_count} (integer)
|
|
{maturity_rating} (string)
|
|
{story_arc} (string)
|
|
{series_group} (string)
|
|
{scan_info} (string)
|
|
{characters} (string)
|
|
{teams} (string)
|
|
{locations} (string)
|
|
{credits} (list of dict({'role': string, 'person': string, 'primary': boolean}))
|
|
{writer} (string)
|
|
{penciller} (string)
|
|
{inker} (string)
|
|
{colorist} (string)
|
|
{letterer} (string)
|
|
{cover artist} (string)
|
|
{editor} (string)
|
|
{tags} (list of str)
|
|
{pages} (list of dict({'Image': string(int), 'Type': string, 'Bookmark': string, 'DoublePage': boolean}))
|
|
|
|
CoMet-only items:
|
|
{price} (float)
|
|
{is_version_of} (string)
|
|
{rights} (string)
|
|
{identifier} (string)
|
|
{last_mark} (string)
|
|
{cover_image} (string)
|
|
|
|
Examples:
|
|
|
|
{series} {issue} ({year})
|
|
Spider-Geddon 1 (2018)
|
|
|
|
{series} #{issue} - {title}
|
|
Spider-Geddon #1 - New Players; Check In
|
|
"""
|
|
|
|
|
|
class SettingsWindow(QtWidgets.QDialog):
|
|
def __init__(self, parent: QtWidgets.QWidget, settings: ComicTaggerSettings, talker_api: ComicTalker) -> None:
|
|
super().__init__(parent)
|
|
|
|
uic.loadUi(ui_path / "settingswindow.ui", self)
|
|
|
|
self.setWindowFlags(
|
|
QtCore.Qt.WindowType(self.windowFlags() & ~QtCore.Qt.WindowType.WindowContextHelpButtonHint)
|
|
)
|
|
|
|
self.settings = settings
|
|
# TODO Quick hack to allow menus to work
|
|
self.available_talkers = ct_api.get_talkers()
|
|
self.talker_api = talker_api
|
|
self.name = "Settings"
|
|
|
|
if platform.system() == "Windows":
|
|
self.lblRarHelp.setText(windowsRarHelp)
|
|
|
|
elif platform.system() == "Linux":
|
|
self.lblRarHelp.setText(linuxRarHelp)
|
|
|
|
elif platform.system() == "Darwin":
|
|
self.leRarExePath.setReadOnly(False)
|
|
|
|
self.lblRarHelp.setText(macRarHelp)
|
|
self.name = "Preferences"
|
|
|
|
self.setWindowTitle("ComicTagger " + self.name)
|
|
self.lblDefaultSettings.setText("Revert to default " + self.name.casefold())
|
|
self.btnResetSettings.setText("Default " + self.name)
|
|
|
|
nmit_tip = """<html>The <b>Name Match Ratio Threshold: Auto-Identify</b> is for eliminating automatic
|
|
search matches that are too long compared to your series name search. The lower
|
|
it is, the more likely to have a good match, but each search will take longer and
|
|
use more bandwidth. Too high, and only the very closest matches will be explored.</html>"""
|
|
nmst_tip = """<html>The <b>Name Match Ratio Threshold: Search</b> is for reducing the total
|
|
number of results that are returned from a search. The lower it is, the more pages will
|
|
be returned (max 5 pages or 500 results)</html>"""
|
|
|
|
self.sbNameMatchIdentifyThresh.setToolTip(nmit_tip)
|
|
self.sbNameMatchSearchThresh.setToolTip(nmst_tip)
|
|
|
|
pbl_tip = """<html>
|
|
The <b>Publisher Filter</b> is for eliminating automatic matches to certain publishers
|
|
that you know are incorrect. Useful for avoiding international re-prints with same
|
|
covers or series names. Enter publisher names separated by commas.
|
|
</html>"""
|
|
self.tePublisherFilter.setToolTip(pbl_tip)
|
|
|
|
validator = QtGui.QIntValidator(1, 4, self)
|
|
self.leIssueNumPadding.setValidator(validator)
|
|
|
|
self.leRenameTemplate.setToolTip(f"<pre>{html.escape(template_tooltip)}</pre>")
|
|
self.settings_to_form()
|
|
self.rename_error: Exception | None = None
|
|
self.rename_test()
|
|
self.dir_test()
|
|
|
|
self.btnBrowseRar.clicked.connect(self.select_rar)
|
|
self.btnClearCache.clicked.connect(self.clear_cache)
|
|
self.btnResetSettings.clicked.connect(self.reset_settings)
|
|
self.btnTemplateHelp.clicked.connect(self.show_template_help)
|
|
self.leRenameTemplate.textEdited.connect(self._rename_test)
|
|
self.cbxMoveFiles.clicked.connect(self.rename_test)
|
|
self.cbxMoveFiles.clicked.connect(self.dir_test)
|
|
self.cbxRenameStrict.clicked.connect(self.rename_test)
|
|
self.leDirectory.textEdited.connect(self.dir_test)
|
|
self.cbxComplicatedParser.clicked.connect(self.switch_parser)
|
|
|
|
self.sources: dict = {}
|
|
self.generate_source_option_tabs()
|
|
|
|
def generate_source_option_tabs(self) -> None:
|
|
# TODO Move to utils?
|
|
def open_web_link(link) -> None:
|
|
if link is not None:
|
|
web_link = link.strip()
|
|
try:
|
|
result = urlparse(web_link)
|
|
all([result.scheme in ["http", "https"], result.netloc])
|
|
webbrowser.open_new_tab(web_link)
|
|
except ValueError:
|
|
QtWidgets.QMessageBox.warning(self, "Web Link", "Web Link is invalid.")
|
|
|
|
# Add source sub tabs to Comic Sources tab
|
|
for source_cls in self.available_talkers.values():
|
|
# TODO Remove hack
|
|
source = source_cls()
|
|
# Add source to general tab dropdown list
|
|
self.cobxInfoSource.addItem(source.source_details.name, source.source_details.id)
|
|
# Use a dict to make a var name from var
|
|
source_info = {}
|
|
tab_name = source.source_details.id
|
|
source_info[tab_name] = {"tab": QtWidgets.QWidget(), "widgets": {}}
|
|
layout_grid = QtWidgets.QGridLayout()
|
|
row = 0
|
|
|
|
# Add logo and web link if available
|
|
if source.static_options.website:
|
|
source_label_logo = QtGui.QPixmap(source.source_details.logo)
|
|
|
|
source_website_icon_link = QtWidgets.QPushButton()
|
|
source_website_icon_link.clicked.connect(lambda _: open_web_link(source.static_options.website))
|
|
source_website_icon_link.setToolTip("Click to visit website")
|
|
source_website_icon_link.setMaximumSize(250, 100)
|
|
|
|
if source_label_logo.isNull():
|
|
source_website_icon_link.setFont(QtGui.QFont("Arial", 14, 3))
|
|
source_website_icon_link.setText(source.source_details.name)
|
|
else:
|
|
if source_label_logo.height() > 100 or source_label_logo.width() > 250:
|
|
source_label_logo = source_label_logo.scaled(
|
|
250, 100, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation
|
|
)
|
|
|
|
source_website_icon_link.setIconSize(QtCore.QSize(source_label_logo.rect().size()))
|
|
source_website_icon_link.setFixedSize(QtCore.QSize(source_label_logo.rect().size()))
|
|
source_website_icon_link.setIcon(QtGui.QIcon(source_label_logo))
|
|
source_website_icon_link.setFlat(True)
|
|
|
|
layout_grid.addWidget(source_website_icon_link, row, 0, 1, -1)
|
|
row += 1
|
|
|
|
for option in source.settings_options.values():
|
|
if not option["hidden"]:
|
|
current_widget = None
|
|
if option["type"] is bool:
|
|
# bool equals a checkbox (QCheckBox)
|
|
current_widget = QtWidgets.QCheckBox(option["text"])
|
|
# Set widget status
|
|
# This works because when a talker class is initialised it loads its settings from disk
|
|
if option["value"]:
|
|
current_widget.setChecked(option["value"])
|
|
# Add widget and span all columns
|
|
layout_grid.addWidget(current_widget, row, 0, 1, -1)
|
|
if option["type"] is int:
|
|
# int equals a spinbox (QSpinBox)
|
|
lbl = QtWidgets.QLabel(option["text"])
|
|
# Create a label
|
|
layout_grid.addWidget(lbl, row, 0)
|
|
current_widget = QtWidgets.QSpinBox()
|
|
current_widget.setRange(0, 9999)
|
|
if option["value"]:
|
|
current_widget.setValue(option["value"])
|
|
layout_grid.addWidget(current_widget, row, 1, alignment=QtCore.Qt.AlignLeft)
|
|
if option["type"] is float:
|
|
# float equals a spinbox (QDoubleSpinBox)
|
|
lbl = QtWidgets.QLabel(option["text"])
|
|
# Create a label
|
|
layout_grid.addWidget(lbl, row, 0)
|
|
current_widget = QtWidgets.QDoubleSpinBox()
|
|
current_widget.setRange(0, 9999.99)
|
|
if option["value"]:
|
|
current_widget.setValue(option["value"])
|
|
layout_grid.addWidget(current_widget, row, 1, alignment=QtCore.Qt.AlignLeft)
|
|
if option["type"] is str:
|
|
# str equals a text field (QLineEdit)
|
|
lbl = QtWidgets.QLabel(option["text"])
|
|
# Create a label
|
|
layout_grid.addWidget(lbl, row, 0)
|
|
current_widget = QtWidgets.QLineEdit()
|
|
# Set widget status
|
|
if option["value"]:
|
|
current_widget.setText(option["value"])
|
|
layout_grid.addWidget(current_widget, row, 1)
|
|
# Special case for api_key, make a test button
|
|
if option["name"] == "api_key":
|
|
btn = QtWidgets.QPushButton("Test Key")
|
|
layout_grid.addWidget(btn, row, 2)
|
|
btn.clicked.connect(lambda: self.test_api_key(source.source_details.id))
|
|
row += 1
|
|
|
|
if current_widget:
|
|
# Add tooltip text
|
|
current_widget.setToolTip(option["help_text"])
|
|
|
|
source_info[tab_name]["widgets"][option["name"]] = current_widget
|
|
else:
|
|
# An empty current_widget implies an unsupported type
|
|
logger.info(f"Unsupported talker option found. Name: {option['name']} Type: {option['type']}")
|
|
|
|
# Add vertical spacer
|
|
vspacer = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
|
layout_grid.addItem(vspacer, row, 0)
|
|
# Display the new widgets
|
|
source_info[tab_name]["tab"].setLayout(layout_grid)
|
|
|
|
# Add new sub tab to Comic Source tab
|
|
self.tComicSourcesOptions.addTab(source_info[tab_name]["tab"], source.source_details.name)
|
|
self.sources.update(source_info)
|
|
|
|
# Select active source in dropdown
|
|
self.cobxInfoSource.setCurrentIndex(self.cobxInfoSource.findData(self.settings.comic_info_source))
|
|
|
|
# Set General as start tab
|
|
self.tabWidget.setCurrentIndex(0)
|
|
|
|
def rename_test(self) -> None:
|
|
self._rename_test(self.leRenameTemplate.text())
|
|
|
|
def dir_test(self) -> None:
|
|
self.lblDir.setText(
|
|
str(pathlib.Path(self.leDirectory.text().strip()).absolute()) if self.cbxMoveFiles.isChecked() else ""
|
|
)
|
|
|
|
def _rename_test(self, template: str) -> None:
|
|
fr = FileRenamer(md_test, platform="universal" if self.cbxRenameStrict.isChecked() else "auto")
|
|
fr.move = self.cbxMoveFiles.isChecked()
|
|
fr.set_template(template)
|
|
fr.set_issue_zero_padding(int(self.leIssueNumPadding.text()))
|
|
fr.set_smart_cleanup(self.cbxSmartCleanup.isChecked())
|
|
try:
|
|
self.lblRenameTest.setText(fr.determine_name(".cbz"))
|
|
self.rename_error = None
|
|
except Exception as e:
|
|
self.rename_error = e
|
|
self.lblRenameTest.setText(str(e))
|
|
|
|
def switch_parser(self) -> None:
|
|
complicated = self.cbxComplicatedParser.isChecked()
|
|
|
|
self.cbxRemoveC2C.setEnabled(complicated)
|
|
self.cbxRemoveFCBD.setEnabled(complicated)
|
|
self.cbxRemovePublisher.setEnabled(complicated)
|
|
|
|
def settings_to_form(self) -> None:
|
|
# Copy values from settings to form
|
|
self.leRarExePath.setText(self.settings.rar_exe_path)
|
|
self.sbNameMatchIdentifyThresh.setValue(self.settings.id_series_match_identify_thresh)
|
|
self.sbNameMatchSearchThresh.setValue(self.settings.id_series_match_search_thresh)
|
|
self.tePublisherFilter.setPlainText(self.settings.id_publisher_filter)
|
|
|
|
self.cbxCheckForNewVersion.setChecked(self.settings.check_for_new_version)
|
|
|
|
self.cbxComplicatedParser.setChecked(self.settings.complicated_parser)
|
|
self.cbxRemoveC2C.setChecked(self.settings.remove_c2c)
|
|
self.cbxRemoveFCBD.setChecked(self.settings.remove_fcbd)
|
|
self.cbxRemovePublisher.setChecked(self.settings.remove_publisher)
|
|
self.switch_parser()
|
|
|
|
self.cbxClearFormBeforePopulating.setChecked(self.settings.clear_form_before_populating)
|
|
|
|
self.cbxUseFilter.setChecked(self.settings.always_use_publisher_filter)
|
|
self.cbxSortByYear.setChecked(self.settings.sort_series_by_year)
|
|
self.cbxExactMatches.setChecked(self.settings.exact_series_matches_first)
|
|
|
|
self.cbxAssumeLoneCreditIsPrimary.setChecked(self.settings.assume_lone_credit_is_primary)
|
|
self.cbxCopyCharactersToTags.setChecked(self.settings.copy_characters_to_tags)
|
|
self.cbxCopyTeamsToTags.setChecked(self.settings.copy_teams_to_tags)
|
|
self.cbxCopyLocationsToTags.setChecked(self.settings.copy_locations_to_tags)
|
|
self.cbxCopyStoryArcsToTags.setChecked(self.settings.copy_storyarcs_to_tags)
|
|
self.cbxCopyNotesToComments.setChecked(self.settings.copy_notes_to_comments)
|
|
self.cbxCopyWebLinkToComments.setChecked(self.settings.copy_weblink_to_comments)
|
|
self.cbxApplyCBLTransformOnCVIMport.setChecked(self.settings.apply_cbl_transform_on_ct_import)
|
|
self.cbxApplyCBLTransformOnBatchOperation.setChecked(self.settings.apply_cbl_transform_on_bulk_operation)
|
|
|
|
self.leRenameTemplate.setText(self.settings.rename_template)
|
|
self.leIssueNumPadding.setText(str(self.settings.rename_issue_number_padding))
|
|
self.cbxSmartCleanup.setChecked(self.settings.rename_use_smart_string_cleanup)
|
|
self.cbxChangeExtension.setChecked(self.settings.rename_extension_based_on_archive)
|
|
self.cbxMoveFiles.setChecked(self.settings.rename_move_dir)
|
|
self.leDirectory.setText(self.settings.rename_dir)
|
|
self.cbxRenameStrict.setChecked(self.settings.rename_strict)
|
|
|
|
def accept(self) -> None:
|
|
self.rename_test()
|
|
if self.rename_error is not None:
|
|
if isinstance(self.rename_error, ValueError):
|
|
logger.exception("Invalid format string: %s", self.settings.rename_template)
|
|
QtWidgets.QMessageBox.critical(
|
|
self,
|
|
"Invalid format string!",
|
|
"Your rename template is invalid!"
|
|
f"<br/><br/>{self.rename_error}<br/><br/>"
|
|
"Please consult the template help in the "
|
|
"settings and the documentation on the format at "
|
|
"<a href='https://docs.python.org/3/library/string.html#format-string-syntax'>"
|
|
"https://docs.python.org/3/library/string.html#format-string-syntax</a>",
|
|
)
|
|
return
|
|
else:
|
|
logger.exception(
|
|
"Formatter failure: %s metadata: %s", self.settings.rename_template, self.renamer.metadata
|
|
)
|
|
QtWidgets.QMessageBox.critical(
|
|
self,
|
|
"The formatter had an issue!",
|
|
"The formatter has experienced an unexpected error!"
|
|
f"<br/><br/>{type(self.rename_error).__name__}: {self.rename_error}<br/><br/>"
|
|
"Please open an issue at "
|
|
"<a href='https://github.com/comictagger/comictagger'>"
|
|
"https://github.com/comictagger/comictagger</a>",
|
|
)
|
|
|
|
# Copy values from form to settings and save
|
|
self.settings.rar_exe_path = str(self.leRarExePath.text())
|
|
|
|
# make sure rar program is now in the path for the rar class
|
|
if self.settings.rar_exe_path:
|
|
utils.add_to_path(os.path.dirname(self.settings.rar_exe_path))
|
|
|
|
if not str(self.leIssueNumPadding.text()).isdigit():
|
|
self.leIssueNumPadding.setText("0")
|
|
|
|
self.settings.check_for_new_version = self.cbxCheckForNewVersion.isChecked()
|
|
|
|
self.settings.id_series_match_identify_thresh = self.sbNameMatchIdentifyThresh.value()
|
|
self.settings.id_series_match_search_thresh = self.sbNameMatchSearchThresh.value()
|
|
self.settings.id_publisher_filter = str(self.tePublisherFilter.toPlainText())
|
|
self.settings.comic_info_source = str(self.cobxInfoSource.itemData(self.cobxInfoSource.currentIndex()))
|
|
# Also change current talker_api object
|
|
if self.settings.comic_info_source != self.talker_api.source_details.id:
|
|
self.talker_api = ct_api.get_comic_talker(self.settings.comic_info_source)()
|
|
|
|
self.settings.complicated_parser = self.cbxComplicatedParser.isChecked()
|
|
self.settings.remove_c2c = self.cbxRemoveC2C.isChecked()
|
|
self.settings.remove_fcbd = self.cbxRemoveFCBD.isChecked()
|
|
self.settings.remove_publisher = self.cbxRemovePublisher.isChecked()
|
|
|
|
self.settings.clear_form_before_populating = self.cbxClearFormBeforePopulating.isChecked()
|
|
self.settings.always_use_publisher_filter = self.cbxUseFilter.isChecked()
|
|
self.settings.sort_series_by_year = self.cbxSortByYear.isChecked()
|
|
self.settings.exact_series_matches_first = self.cbxExactMatches.isChecked()
|
|
|
|
self.settings.assume_lone_credit_is_primary = self.cbxAssumeLoneCreditIsPrimary.isChecked()
|
|
self.settings.copy_characters_to_tags = self.cbxCopyCharactersToTags.isChecked()
|
|
self.settings.copy_teams_to_tags = self.cbxCopyTeamsToTags.isChecked()
|
|
self.settings.copy_locations_to_tags = self.cbxCopyLocationsToTags.isChecked()
|
|
self.settings.copy_storyarcs_to_tags = self.cbxCopyStoryArcsToTags.isChecked()
|
|
self.settings.copy_notes_to_comments = self.cbxCopyNotesToComments.isChecked()
|
|
self.settings.copy_weblink_to_comments = self.cbxCopyWebLinkToComments.isChecked()
|
|
self.settings.apply_cbl_transform_on_ct_import = self.cbxApplyCBLTransformOnCVIMport.isChecked()
|
|
self.settings.apply_cbl_transform_on_bulk_operation = self.cbxApplyCBLTransformOnBatchOperation.isChecked()
|
|
|
|
self.settings.rename_template = str(self.leRenameTemplate.text())
|
|
self.settings.rename_issue_number_padding = int(self.leIssueNumPadding.text())
|
|
self.settings.rename_use_smart_string_cleanup = self.cbxSmartCleanup.isChecked()
|
|
self.settings.rename_extension_based_on_archive = self.cbxChangeExtension.isChecked()
|
|
self.settings.rename_move_dir = self.cbxMoveFiles.isChecked()
|
|
self.settings.rename_dir = self.leDirectory.text()
|
|
|
|
self.settings.rename_strict = self.cbxRenameStrict.isChecked()
|
|
|
|
# Read settings from sources tabs and generate self.settings.config data
|
|
for source_cls in self.available_talkers.values():
|
|
# TODO Remove hack
|
|
source = source_cls()
|
|
source_info = self.sources[source.source_details.id]
|
|
if not self.settings.config.has_section(source.source_details.id):
|
|
self.settings.config.add_section(source.source_details.id)
|
|
# Iterate over sources options and get the tab setting
|
|
for option in source.settings_options.values():
|
|
# Only save visible here
|
|
if option["name"] in source_info["widgets"]:
|
|
# Set the tab setting for the talker class var
|
|
if option["type"] is bool:
|
|
current_widget: QtWidgets.QCheckBox = source_info["widgets"][option["name"]]
|
|
option["value"] = current_widget.isChecked()
|
|
if option["type"] is int:
|
|
current_widget: QtWidgets.QSpinBox = source_info["widgets"][option["name"]]
|
|
option["value"] = current_widget.value()
|
|
if option["type"] is float:
|
|
current_widget: QtWidgets.QDoubleSpinBox = source_info["widgets"][option["name"]]
|
|
option["value"] = current_widget.value()
|
|
if option["type"] is str:
|
|
current_widget: QtWidgets.QLineEdit = source_info["widgets"][option["name"]]
|
|
option["value"] = current_widget.text().strip()
|
|
|
|
else:
|
|
# Handle hidden, assume changed programmatically
|
|
if option["name"] == "enabled":
|
|
# Set to disabled if is not the selected talker
|
|
if source.source_details.id != self.settings.comic_info_source:
|
|
source.settings_options["enabled"]["value"] = False
|
|
else:
|
|
source.settings_options["enabled"]["value"] = True
|
|
else:
|
|
# Ensure correct type
|
|
if option["type"] is bool:
|
|
option["value"] = bool(option["value"])
|
|
if option["type"] is int:
|
|
option["value"] = int(option["value"])
|
|
if option["type"] is float:
|
|
option["value"] = float(option["value"])
|
|
if option["type"] is str:
|
|
option["value"] = str(option["value"]).strip()
|
|
|
|
# Save out option
|
|
self.settings.config.set(source.source_details.id, option["name"], option["value"])
|
|
|
|
# Update talker options in object
|
|
if source.source_details.id == self.talker_api.source_details.id:
|
|
self.talker_api.settings_options[option["name"]]["value"] = option["value"]
|
|
|
|
self.settings.save()
|
|
QtWidgets.QDialog.accept(self)
|
|
|
|
def select_rar(self) -> None:
|
|
self.select_file(self.leRarExePath, "RAR")
|
|
|
|
def clear_cache(self) -> None:
|
|
ImageFetcher().clear_cache()
|
|
ComicCacher().clear_cache()
|
|
QtWidgets.QMessageBox.information(self, self.name, "Cache has been cleared.")
|
|
|
|
def test_api_key(self, source_id) -> None:
|
|
# TODO Only allow testing of active talker?
|
|
if source_id == self.settings.comic_info_source:
|
|
key = self.sources[source_id]["widgets"]["api_key"].text().strip()
|
|
url = self.sources[source_id]["widgets"]["url_root"].text().strip()
|
|
|
|
if self.talker_api.check_api_key(key, url):
|
|
QtWidgets.QMessageBox.information(self, "API Key Test", "Key is valid!")
|
|
else:
|
|
QtWidgets.QMessageBox.warning(self, "API Key Test", "Key is NOT valid!")
|
|
else:
|
|
QtWidgets.QMessageBox.warning(self, "API Key Test", "Can only test active comic source.")
|
|
|
|
def reset_settings(self) -> None:
|
|
self.settings.reset()
|
|
self.settings_to_form()
|
|
QtWidgets.QMessageBox.information(self, self.name, self.name + " have been returned to default values.")
|
|
|
|
def select_file(self, control: QtWidgets.QLineEdit, name: str) -> None:
|
|
|
|
dialog = QtWidgets.QFileDialog(self)
|
|
dialog.setFileMode(QtWidgets.QFileDialog.FileMode.ExistingFile)
|
|
|
|
if platform.system() == "Windows":
|
|
if name == "RAR":
|
|
flt = "Rar Program (Rar.exe)"
|
|
else:
|
|
flt = "Libraries (*.dll)"
|
|
dialog.setNameFilter(flt)
|
|
else:
|
|
dialog.setFilter(QtCore.QDir.Filter.Files)
|
|
|
|
dialog.setDirectory(os.path.dirname(str(control.text())))
|
|
if name == "RAR":
|
|
dialog.setWindowTitle("Find " + name + " program")
|
|
else:
|
|
dialog.setWindowTitle("Find " + name + " library")
|
|
|
|
if dialog.exec():
|
|
file_list = dialog.selectedFiles()
|
|
control.setText(str(file_list[0]))
|
|
|
|
def show_rename_tab(self) -> None:
|
|
self.tabWidget.setCurrentIndex(5)
|
|
|
|
def show_template_help(self) -> None:
|
|
template_help_win = TemplateHelpWindow(self)
|
|
template_help_win.setModal(False)
|
|
template_help_win.show()
|
|
|
|
|
|
class TemplateHelpWindow(QtWidgets.QDialog):
|
|
def __init__(self, parent: QtWidgets.QWidget) -> None:
|
|
super().__init__(parent)
|
|
|
|
uic.loadUi(ui_path / "TemplateHelp.ui", self)
|