2e01a4db14
Moves to Python format strings for renaming, handles directory structures, moving of files to a destination directory, sanitizes file paths with pathvalidate and takes a different approach to smart filename cleanup using the Python string.Formatter class Moving to Python format strings means we can point to python documentation for syntax and all we have to do is document the properties and types that are attached to the GenericMetadata class. Switching to pathvalidate allows comictagger to more simply handle both directories and symbols in filenames. The only changes to the string.Formatter class is: 1. format_field returns an empty string if the value is none or an empty string regardless of the format specifier. 2. _vformat drops the previous literal text if the field value is an empty string and lstrips the following literal text of closing special characters.
367 lines
15 KiB
Python
367 lines
15 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.
|
|
|
|
import platform
|
|
import os
|
|
import sys
|
|
|
|
from PyQt5 import QtCore, QtGui, QtWidgets, uic
|
|
|
|
from .settings import ComicTaggerSettings
|
|
from .comicvinecacher import ComicVineCacher
|
|
from .comicvinetalker import ComicVineTalker
|
|
from .imagefetcher import ImageFetcher
|
|
from .filerenamer import FileRenamer
|
|
from .genericmetadata import GenericMetadata
|
|
from . import utils
|
|
|
|
|
|
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,
|
|
which is free to use.)</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>
|
|
"""
|
|
|
|
class SettingsWindow(QtWidgets.QDialog):
|
|
|
|
def __init__(self, parent, settings):
|
|
super(SettingsWindow, self).__init__(parent)
|
|
|
|
uic.loadUi(ComicTaggerSettings.getUIFile('settingswindow.ui'), self)
|
|
|
|
self.setWindowFlags(self.windowFlags() &
|
|
~QtCore.Qt.WindowContextHelpButtonHint)
|
|
|
|
self.settings = settings
|
|
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.lower())
|
|
self.btnResetSettings.setText("Default " + self.name)
|
|
|
|
nldtTip = (
|
|
"""<html>The <b>Default Name Length Match Tolerance</b> is for eliminating automatic
|
|
search matches that are too long compared to your series name search. The higher
|
|
it is, the more likely to have a good match, but each search will take longer and
|
|
use more bandwidth. Too low, and only the very closest lexical matches will be
|
|
explored.</html>""")
|
|
|
|
self.leNameLengthDeltaThresh.setToolTip(nldtTip)
|
|
|
|
pblTip = (
|
|
"""<html>
|
|
The <b>Publisher Blacklist</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.tePublisherBlacklist.setToolTip(pblTip)
|
|
|
|
validator = QtGui.QIntValidator(1, 4, self)
|
|
self.leIssueNumPadding.setValidator(validator)
|
|
|
|
validator = QtGui.QIntValidator(0, 99, self)
|
|
self.leNameLengthDeltaThresh.setValidator(validator)
|
|
|
|
self.settingsToForm()
|
|
|
|
self.btnBrowseRar.clicked.connect(self.selectRar)
|
|
self.btnClearCache.clicked.connect(self.clearCache)
|
|
self.btnResetSettings.clicked.connect(self.resetSettings)
|
|
self.btnTestKey.clicked.connect(self.testAPIKey)
|
|
self.btnTemplateHelp.clicked.connect(self.showTemplateHelp)
|
|
|
|
def configRenamer(self):
|
|
md = GenericMetadata()
|
|
md.isEmpty = False
|
|
md.tagOrigin = "testing"
|
|
|
|
md.series = "series name"
|
|
md.issue = "1"
|
|
md.title = "issue title"
|
|
md.publisher = "publisher"
|
|
md.year = 1998
|
|
md.month = 4
|
|
md.day = 4
|
|
md.issueCount = 1
|
|
md.volume = 256
|
|
md.genre = "test"
|
|
md.language = "en" # 2 letter iso code
|
|
md.comments = "This is definitly a comic." # use same way as Summary in CIX
|
|
|
|
md.volumeCount = 4096
|
|
md.criticalRating = "Worst Comic Ever"
|
|
md.country = "US"
|
|
|
|
md.alternateSeries = "None"
|
|
md.alternateNumber = 4.4
|
|
md.alternateCount = 4444
|
|
md.imprint = 'imprint'
|
|
md.notes = "This doesn't actually exist"
|
|
md.webLink = "https://example.com/series name/1"
|
|
md.format = "Box Set"
|
|
md.manga = "Yes"
|
|
md.blackAndWhite = False
|
|
md.pageCount = 4
|
|
md.maturityRating = "Everyone"
|
|
|
|
md.storyArc = "story"
|
|
md.seriesGroup = "seriesGroup"
|
|
md.scanInfo = "(lordwelch)"
|
|
|
|
md.characters = "character 1, character 2"
|
|
md.teams = "None"
|
|
md.locations = "Earth, 444 B.C."
|
|
|
|
md.credits = [dict({'role': 'Everything', 'person': 'author', 'primary': True})]
|
|
md.tags = ["testing", "not real"]
|
|
md.pages = [dict({'Image': '0', 'Type': 'Front Cover'}), dict({'Image': '1', 'Type': 'Story'})]
|
|
|
|
# Some CoMet-only items
|
|
md.price = 0.00
|
|
md.isVersionOf = "SERIES #1"
|
|
md.rights = "None"
|
|
md.identifier = "LW4444-Comic"
|
|
md.lastMark = "0"
|
|
md.coverImage = "https://example.com/series name/1/cover"
|
|
|
|
self.renamer = FileRenamer(md)
|
|
self.renamer.setTemplate(str(self.leRenameTemplate.text()))
|
|
self.renamer.setIssueZeroPadding(self.settings.rename_issue_number_padding)
|
|
self.renamer.setSmartCleanup(self.settings.rename_use_smart_string_cleanup)
|
|
|
|
def settingsToForm(self):
|
|
|
|
# Copy values from settings to form
|
|
self.leRarExePath.setText(self.settings.rar_exe_path)
|
|
self.leNameLengthDeltaThresh.setText(
|
|
str(self.settings.id_length_delta_thresh))
|
|
self.tePublisherBlacklist.setPlainText(
|
|
self.settings.id_publisher_blacklist)
|
|
|
|
if self.settings.check_for_new_version:
|
|
self.cbxCheckForNewVersion.setCheckState(QtCore.Qt.Checked)
|
|
|
|
if self.settings.parse_scan_info:
|
|
self.cbxParseScanInfo.setCheckState(QtCore.Qt.Checked)
|
|
|
|
if self.settings.use_series_start_as_volume:
|
|
self.cbxUseSeriesStartAsVolume.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.clear_form_before_populating_from_cv:
|
|
self.cbxClearFormBeforePopulating.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.remove_html_tables:
|
|
self.cbxRemoveHtmlTables.setCheckState(QtCore.Qt.Checked)
|
|
self.leKey.setText(str(self.settings.cv_api_key))
|
|
|
|
if self.settings.assume_lone_credit_is_primary:
|
|
self.cbxAssumeLoneCreditIsPrimary.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_characters_to_tags:
|
|
self.cbxCopyCharactersToTags.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_teams_to_tags:
|
|
self.cbxCopyTeamsToTags.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_locations_to_tags:
|
|
self.cbxCopyLocationsToTags.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_storyarcs_to_tags:
|
|
self.cbxCopyStoryArcsToTags.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_notes_to_comments:
|
|
self.cbxCopyNotesToComments.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.copy_weblink_to_comments:
|
|
self.cbxCopyWebLinkToComments.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.apply_cbl_transform_on_cv_import:
|
|
self.cbxApplyCBLTransformOnCVIMport.setCheckState(
|
|
QtCore.Qt.Checked)
|
|
if self.settings.apply_cbl_transform_on_bulk_operation:
|
|
self.cbxApplyCBLTransformOnBatchOperation.setCheckState(
|
|
QtCore.Qt.Checked)
|
|
|
|
self.leRenameTemplate.setText(self.settings.rename_template)
|
|
self.leIssueNumPadding.setText(
|
|
str(self.settings.rename_issue_number_padding))
|
|
if self.settings.rename_use_smart_string_cleanup:
|
|
self.cbxSmartCleanup.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.rename_extension_based_on_archive:
|
|
self.cbxChangeExtension.setCheckState(QtCore.Qt.Checked)
|
|
if self.settings.rename_move_dir:
|
|
self.cbxMoveFiles.setCheckState(QtCore.Qt.Checked)
|
|
self.leDirectory.setText(self.settings.rename_dir)
|
|
|
|
def accept(self):
|
|
|
|
self.configRenamer()
|
|
|
|
|
|
try:
|
|
new_name = self.renamer.determineName('test.cbz')
|
|
except Exception as e:
|
|
QtWidgets.QMessageBox.critical(self, 'Invalid format string!',
|
|
'Your rename template is invalid!'
|
|
'<br/><br/>{}<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>'.format(e))
|
|
return
|
|
|
|
# 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.addtopath(os.path.dirname(self.settings.rar_exe_path))
|
|
|
|
if not str(self.leNameLengthDeltaThresh.text()).isdigit():
|
|
self.leNameLengthDeltaThresh.setText("0")
|
|
|
|
if not str(self.leIssueNumPadding.text()).isdigit():
|
|
self.leIssueNumPadding.setText("0")
|
|
|
|
self.settings.check_for_new_version = self.cbxCheckForNewVersion.isChecked()
|
|
|
|
self.settings.id_length_delta_thresh = int(
|
|
self.leNameLengthDeltaThresh.text())
|
|
self.settings.id_publisher_blacklist = str(
|
|
self.tePublisherBlacklist.toPlainText())
|
|
|
|
self.settings.parse_scan_info = self.cbxParseScanInfo.isChecked()
|
|
|
|
self.settings.use_series_start_as_volume = self.cbxUseSeriesStartAsVolume.isChecked()
|
|
self.settings.clear_form_before_populating_from_cv = self.cbxClearFormBeforePopulating.isChecked()
|
|
self.settings.remove_html_tables = self.cbxRemoveHtmlTables.isChecked()
|
|
self.settings.cv_api_key = str(self.leKey.text())
|
|
ComicVineTalker.api_key = self.settings.cv_api_key.strip()
|
|
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_cv_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.save()
|
|
QtWidgets.QDialog.accept(self)
|
|
|
|
def selectRar(self):
|
|
self.selectFile(self.leRarExePath, "RAR")
|
|
|
|
def clearCache(self):
|
|
ImageFetcher().clearCache()
|
|
ComicVineCacher().clearCache()
|
|
QtWidgets.QMessageBox.information(
|
|
self, self.name, "Cache has been cleared.")
|
|
|
|
def testAPIKey(self):
|
|
if ComicVineTalker().testKey(str(self.leKey.text()).strip()):
|
|
QtWidgets.QMessageBox.information(
|
|
self, "API Key Test", "Key is valid!")
|
|
else:
|
|
QtWidgets.QMessageBox.warning(
|
|
self, "API Key Test", "Key is NOT valid.")
|
|
|
|
def resetSettings(self):
|
|
self.settings.reset()
|
|
self.settingsToForm()
|
|
QtWidgets.QMessageBox.information(
|
|
self,
|
|
self.name,
|
|
self.name +
|
|
" have been returned to default values.")
|
|
|
|
def selectFile(self, control, name):
|
|
|
|
dialog = QtWidgets.QFileDialog(self)
|
|
dialog.setFileMode(QtWidgets.QFileDialog.ExistingFile)
|
|
|
|
if platform.system() == "Windows":
|
|
if name == "RAR":
|
|
filter = self.tr("Rar Program (Rar.exe)")
|
|
else:
|
|
filter = self.tr("Libraries (*.dll)")
|
|
dialog.setNameFilter(filter)
|
|
else:
|
|
# QtCore.QDir.Executable | QtCore.QDir.Files)
|
|
dialog.setFilter(QtCore.QDir.Files)
|
|
pass
|
|
|
|
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_()):
|
|
fileList = dialog.selectedFiles()
|
|
control.setText(str(fileList[0]))
|
|
|
|
def showRenameTab(self):
|
|
self.tabWidget.setCurrentIndex(5)
|
|
|
|
def showTemplateHelp(self):
|
|
TemplateHelpWin = TemplateHelpWindow(self)
|
|
TemplateHelpWin.setModal(False)
|
|
TemplateHelpWin.show()
|
|
|
|
class TemplateHelpWindow(QtWidgets.QDialog):
|
|
|
|
def __init__(self, parent):
|
|
super(TemplateHelpWindow, self).__init__(parent)
|
|
|
|
uic.loadUi(ComicTaggerSettings.getUIFile('TemplateHelp.ui'), self)
|
|
|
|
|