Merge pull request #2519 from twodoorcoupe/format_converting

PICARD-2936: Add option to convert cover art images to a different format
This commit is contained in:
Laurent Monin
2024-06-26 18:57:49 +02:00
committed by GitHub
8 changed files with 267 additions and 32 deletions

View File

@@ -153,3 +153,4 @@ DEFAULT_TIME_FORMAT = '%Y-%m-%d %H:%M:%S'
DEFAULT_COVER_MIN_SIZE = 250 DEFAULT_COVER_MIN_SIZE = 250
DEFAULT_COVER_MAX_SIZE = 1000 DEFAULT_COVER_MAX_SIZE = 1000
DEFAULT_COVER_CONVERTING_FORMAT = "jpeg"

View File

@@ -70,8 +70,8 @@ class ResizeImage(ImageProcessor):
target_width = config.setting['cover_tags_resize_target_width'] target_width = config.setting['cover_tags_resize_target_width']
use_height = config.setting['cover_tags_resize_use_height'] use_height = config.setting['cover_tags_resize_use_height']
target_height = config.setting['cover_tags_resize_target_height'] target_height = config.setting['cover_tags_resize_target_height']
stretch = config.setting["cover_tags_stretch"] stretch = config.setting['cover_tags_stretch']
crop = config.setting["cover_tags_crop"] crop = config.setting['cover_tags_crop']
else: else:
scale_up = config.setting['cover_file_scale_up'] scale_up = config.setting['cover_file_scale_up']
scale_down = config.setting['cover_file_scale_down'] scale_down = config.setting['cover_file_scale_down']
@@ -79,8 +79,8 @@ class ResizeImage(ImageProcessor):
target_width = config.setting['cover_file_resize_target_width'] target_width = config.setting['cover_file_resize_target_width']
use_height = config.setting['cover_file_resize_use_height'] use_height = config.setting['cover_file_resize_use_height']
target_height = config.setting['cover_file_resize_target_height'] target_height = config.setting['cover_file_resize_target_height']
stretch = config.setting["cover_file_stretch"] stretch = config.setting['cover_file_stretch']
crop = config.setting["cover_file_crop"] crop = config.setting['cover_file_crop']
width_resize = target_width if use_width else image.info.width width_resize = target_width if use_width else image.info.width
height_resize = target_height if use_height else image.info.height height_resize = target_height if use_height else image.info.height
@@ -130,4 +130,41 @@ class ResizeImage(ImageProcessor):
image.set_result(scaled_image) image.set_result(scaled_image)
class ConvertImage(ImageProcessor):
_format_aliases = {
"jpeg": {"jpg", "jpeg"},
"png": {"png"},
"webp": {"webp"},
"tiff": {"tif", "tiff"},
}
def save_to_tags(self):
config = get_config()
return config.setting['cover_tags_convert_images']
def save_to_file(self):
config = get_config()
return config.setting['cover_file_convert_images']
def same_processing(self):
config = get_config()
same_format = config.setting['cover_tags_convert_to_format'] == config.setting['cover_file_convert_to_format']
return self.save_to_file() and self.save_to_tags() and same_format
def run(self, image, target):
config = get_config()
if target == ProcessingTarget.TAGS:
new_format = config.setting['cover_tags_convert_to_format'].lower()
else:
new_format = config.setting['cover_file_convert_to_format'].lower()
previous_format = image.info.format
if previous_format in self._format_aliases[new_format]:
return
image.info.extension = f".{new_format}"
image.info.mime = f"image/{new_format}"
log.debug("Changed cover art format from %s to %s", previous_format, new_format)
register_cover_art_processor(ResizeImage) register_cover_art_processor(ResizeImage)
register_cover_art_processor(ConvertImage)

View File

@@ -72,7 +72,8 @@ class ProcessingImage:
buffer = QBuffer() buffer = QBuffer()
if not self._qimage.save(buffer, image_format, quality=quality): if not self._qimage.save(buffer, image_format, quality=quality):
raise CoverArtEncodingError(f"Failed to encode into {image_format}") raise CoverArtEncodingError(f"Failed to encode into {image_format}")
return buffer.data() qbytearray = buffer.data()
return qbytearray.data()
class ImageProcessor: class ImageProcessor:

View File

@@ -39,6 +39,7 @@ from picard.const.defaults import (
DEFAULT_CAA_IMAGE_TYPE_EXCLUDE, DEFAULT_CAA_IMAGE_TYPE_EXCLUDE,
DEFAULT_CAA_IMAGE_TYPE_INCLUDE, DEFAULT_CAA_IMAGE_TYPE_INCLUDE,
DEFAULT_CACHE_SIZE_IN_BYTES, DEFAULT_CACHE_SIZE_IN_BYTES,
DEFAULT_COVER_CONVERTING_FORMAT,
DEFAULT_COVER_IMAGE_FILENAME, DEFAULT_COVER_IMAGE_FILENAME,
DEFAULT_COVER_MAX_SIZE, DEFAULT_COVER_MAX_SIZE,
DEFAULT_COVER_MIN_SIZE, DEFAULT_COVER_MIN_SIZE,
@@ -183,6 +184,8 @@ BoolOption('setting', 'cover_tags_resize_use_height', True)
IntOption('setting', 'cover_tags_resize_target_height', DEFAULT_COVER_MAX_SIZE) IntOption('setting', 'cover_tags_resize_target_height', DEFAULT_COVER_MAX_SIZE)
BoolOption('setting', 'cover_tags_stretch', False) BoolOption('setting', 'cover_tags_stretch', False)
BoolOption('setting', 'cover_tags_crop', False) BoolOption('setting', 'cover_tags_crop', False)
BoolOption('setting', 'cover_tags_convert_images', False)
TextOption('setting', 'cover_tags_convert_to_format', DEFAULT_COVER_CONVERTING_FORMAT)
BoolOption('setting', 'cover_file_scale_up', False) BoolOption('setting', 'cover_file_scale_up', False)
BoolOption('setting', 'cover_file_scale_down', False) BoolOption('setting', 'cover_file_scale_down', False)
BoolOption('setting', 'cover_file_resize_use_width', True) BoolOption('setting', 'cover_file_resize_use_width', True)
@@ -191,6 +194,8 @@ BoolOption('setting', 'cover_file_resize_use_height', True)
IntOption('setting', 'cover_file_resize_target_height', DEFAULT_COVER_MAX_SIZE) IntOption('setting', 'cover_file_resize_target_height', DEFAULT_COVER_MAX_SIZE)
BoolOption('setting', 'cover_file_stretch', False) BoolOption('setting', 'cover_file_stretch', False)
BoolOption('setting', 'cover_file_crop', False) BoolOption('setting', 'cover_file_crop', False)
BoolOption('setting', 'cover_file_convert_images', False)
TextOption('setting', 'cover_file_convert_to_format', DEFAULT_COVER_CONVERTING_FORMAT)
# picard/ui/options/dialog.py # picard/ui/options/dialog.py
# Attached Profiles # Attached Profiles

View File

@@ -17,7 +17,7 @@ from picard.i18n import gettext as _
class Ui_CoverProcessingOptionsPage(object): class Ui_CoverProcessingOptionsPage(object):
def setupUi(self, CoverProcessingOptionsPage): def setupUi(self, CoverProcessingOptionsPage):
CoverProcessingOptionsPage.setObjectName("CoverProcessingOptionsPage") CoverProcessingOptionsPage.setObjectName("CoverProcessingOptionsPage")
CoverProcessingOptionsPage.resize(478, 423) CoverProcessingOptionsPage.resize(529, 467)
self.verticalLayout = QtWidgets.QVBoxLayout(CoverProcessingOptionsPage) self.verticalLayout = QtWidgets.QVBoxLayout(CoverProcessingOptionsPage)
self.verticalLayout.setObjectName("verticalLayout") self.verticalLayout.setObjectName("verticalLayout")
self.filtering = QtWidgets.QGroupBox(parent=CoverProcessingOptionsPage) self.filtering = QtWidgets.QGroupBox(parent=CoverProcessingOptionsPage)
@@ -272,6 +272,45 @@ class Ui_CoverProcessingOptionsPage(object):
self.verticalLayout_4.addWidget(self.file_resize_mode) self.verticalLayout_4.addWidget(self.file_resize_mode)
self.horizontalLayout_7.addWidget(self.save_to_file) self.horizontalLayout_7.addWidget(self.save_to_file)
self.verticalLayout.addWidget(self.resizing) self.verticalLayout.addWidget(self.resizing)
self.converting = QtWidgets.QGroupBox(parent=CoverProcessingOptionsPage)
self.converting.setCheckable(False)
self.converting.setChecked(False)
self.converting.setObjectName("converting")
self.horizontalLayout_12 = QtWidgets.QHBoxLayout(self.converting)
self.horizontalLayout_12.setObjectName("horizontalLayout_12")
self.convert_tags = QtWidgets.QGroupBox(parent=self.converting)
self.convert_tags.setCheckable(True)
self.convert_tags.setObjectName("convert_tags")
self.horizontalLayout_13 = QtWidgets.QHBoxLayout(self.convert_tags)
self.horizontalLayout_13.setObjectName("horizontalLayout_13")
self.convert_tags_label = QtWidgets.QLabel(parent=self.convert_tags)
self.convert_tags_label.setObjectName("convert_tags_label")
self.horizontalLayout_13.addWidget(self.convert_tags_label)
self.convert_tags_format = QtWidgets.QComboBox(parent=self.convert_tags)
self.convert_tags_format.setObjectName("convert_tags_format")
self.convert_tags_format.addItem("")
self.convert_tags_format.addItem("")
self.convert_tags_format.addItem("")
self.convert_tags_format.addItem("")
self.horizontalLayout_13.addWidget(self.convert_tags_format)
self.horizontalLayout_12.addWidget(self.convert_tags)
self.convert_file = QtWidgets.QGroupBox(parent=self.converting)
self.convert_file.setCheckable(True)
self.convert_file.setObjectName("convert_file")
self.horizontalLayout_14 = QtWidgets.QHBoxLayout(self.convert_file)
self.horizontalLayout_14.setObjectName("horizontalLayout_14")
self.convert_file_label = QtWidgets.QLabel(parent=self.convert_file)
self.convert_file_label.setObjectName("convert_file_label")
self.horizontalLayout_14.addWidget(self.convert_file_label)
self.convert_file_format = QtWidgets.QComboBox(parent=self.convert_file)
self.convert_file_format.setObjectName("convert_file_format")
self.convert_file_format.addItem("")
self.convert_file_format.addItem("")
self.convert_file_format.addItem("")
self.convert_file_format.addItem("")
self.horizontalLayout_14.addWidget(self.convert_file_format)
self.horizontalLayout_12.addWidget(self.convert_file)
self.verticalLayout.addWidget(self.converting)
spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Policy.Minimum, QtWidgets.QSizePolicy.Policy.Expanding) spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Policy.Minimum, QtWidgets.QSizePolicy.Policy.Expanding)
self.verticalLayout.addItem(spacerItem) self.verticalLayout.addItem(spacerItem)
@@ -306,3 +345,16 @@ class Ui_CoverProcessingOptionsPage(object):
self.file_keep.setText(_("Fit")) self.file_keep.setText(_("Fit"))
self.file_crop.setText(_("Fill")) self.file_crop.setText(_("Fill"))
self.file_stretch.setText(_("Stretch")) self.file_stretch.setText(_("Stretch"))
self.converting.setTitle(_("Convert images to the given format"))
self.convert_tags.setTitle(_("Convert images saved to tags"))
self.convert_tags_label.setText(_("New format:"))
self.convert_tags_format.setItemText(0, _("JPEG"))
self.convert_tags_format.setItemText(1, _("PNG"))
self.convert_tags_format.setItemText(2, _("WebP"))
self.convert_tags_format.setItemText(3, _("TIFF"))
self.convert_file.setTitle(_("Convert images saved to files"))
self.convert_file_label.setText(_("New format:"))
self.convert_file_format.setItemText(0, _("JPEG"))
self.convert_file_format.setItemText(1, _("PNG"))
self.convert_file_format.setItemText(2, _("WebP"))
self.convert_file_format.setItemText(3, _("TIFF"))

View File

@@ -55,6 +55,8 @@ class CoverProcessingOptionsPage(OptionsPage):
self.register_setting('cover_tags_resize_target_height') self.register_setting('cover_tags_resize_target_height')
self.register_setting('cover_tags_stretch') self.register_setting('cover_tags_stretch')
self.register_setting('cover_tags_crop') self.register_setting('cover_tags_crop')
self.register_setting('cover_tags_convert_images')
self.register_setting('cover_tags_convert_to_format')
self.register_setting('cover_file_scale_up') self.register_setting('cover_file_scale_up')
self.register_setting('cover_file_scale_down') self.register_setting('cover_file_scale_down')
self.register_setting('cover_file_resize_use_width') self.register_setting('cover_file_resize_use_width')
@@ -63,6 +65,8 @@ class CoverProcessingOptionsPage(OptionsPage):
self.register_setting('cover_file_resize_target_height') self.register_setting('cover_file_resize_target_height')
self.register_setting('cover_file_stretch') self.register_setting('cover_file_stretch')
self.register_setting('cover_file_crop') self.register_setting('cover_file_crop')
self.register_setting('cover_file_convert_images')
self.register_setting('cover_file_convert_to_format')
tooltip_keep = N_( tooltip_keep = N_(
"<p>" "<p>"
@@ -146,6 +150,8 @@ class CoverProcessingOptionsPage(OptionsPage):
self.ui.tags_resize_height_value.setValue(config.setting['cover_tags_resize_target_height']) self.ui.tags_resize_height_value.setValue(config.setting['cover_tags_resize_target_height'])
self.ui.tags_stretch.setChecked(config.setting['cover_tags_stretch']) self.ui.tags_stretch.setChecked(config.setting['cover_tags_stretch'])
self.ui.tags_crop.setChecked(config.setting['cover_tags_crop']) self.ui.tags_crop.setChecked(config.setting['cover_tags_crop'])
self.ui.convert_tags.setChecked(config.setting['cover_tags_convert_images'])
self.ui.convert_tags_format.setCurrentText(config.setting['cover_tags_convert_to_format'])
self.ui.file_scale_up.setChecked(config.setting['cover_file_scale_up']) self.ui.file_scale_up.setChecked(config.setting['cover_file_scale_up'])
self.ui.file_scale_down.setChecked(config.setting['cover_file_scale_down']) self.ui.file_scale_down.setChecked(config.setting['cover_file_scale_down'])
self.ui.file_resize_width_label.setChecked(config.setting['cover_file_resize_use_width']) self.ui.file_resize_width_label.setChecked(config.setting['cover_file_resize_use_width'])
@@ -154,6 +160,8 @@ class CoverProcessingOptionsPage(OptionsPage):
self.ui.file_resize_height_value.setValue(config.setting['cover_file_resize_target_height']) self.ui.file_resize_height_value.setValue(config.setting['cover_file_resize_target_height'])
self.ui.file_stretch.setChecked(config.setting['cover_file_stretch']) self.ui.file_stretch.setChecked(config.setting['cover_file_stretch'])
self.ui.file_crop.setChecked(config.setting['cover_file_crop']) self.ui.file_crop.setChecked(config.setting['cover_file_crop'])
self.ui.convert_file.setChecked(config.setting['cover_file_convert_images'])
self.ui.convert_file_format.setCurrentText(config.setting['cover_file_convert_to_format'])
def save(self): def save(self):
config = get_config() config = get_config()
@@ -168,6 +176,8 @@ class CoverProcessingOptionsPage(OptionsPage):
config.setting['cover_tags_resize_target_height'] = self.ui.tags_resize_height_value.value() config.setting['cover_tags_resize_target_height'] = self.ui.tags_resize_height_value.value()
config.setting['cover_tags_stretch'] = self.ui.tags_stretch.isChecked() config.setting['cover_tags_stretch'] = self.ui.tags_stretch.isChecked()
config.setting['cover_tags_crop'] = self.ui.tags_crop.isChecked() config.setting['cover_tags_crop'] = self.ui.tags_crop.isChecked()
config.setting['cover_tags_convert_images'] = self.ui.convert_tags.isChecked()
config.setting['cover_tags_convert_to_format'] = self.ui.convert_tags_format.currentText()
config.setting['cover_file_scale_up'] = self.ui.file_scale_up.isChecked() config.setting['cover_file_scale_up'] = self.ui.file_scale_up.isChecked()
config.setting['cover_file_scale_down'] = self.ui.file_scale_down.isChecked() config.setting['cover_file_scale_down'] = self.ui.file_scale_down.isChecked()
config.setting['cover_file_resize_use_width'] = self.ui.file_resize_width_label.isChecked() config.setting['cover_file_resize_use_width'] = self.ui.file_resize_width_label.isChecked()
@@ -176,6 +186,8 @@ class CoverProcessingOptionsPage(OptionsPage):
config.setting['cover_file_resize_target_height'] = self.ui.file_resize_height_value.value() config.setting['cover_file_resize_target_height'] = self.ui.file_resize_height_value.value()
config.setting['cover_file_stretch'] = self.ui.file_stretch.isChecked() config.setting['cover_file_stretch'] = self.ui.file_stretch.isChecked()
config.setting['cover_file_crop'] = self.ui.file_crop.isChecked() config.setting['cover_file_crop'] = self.ui.file_crop.isChecked()
config.setting['cover_file_convert_images'] = self.ui.convert_file.isChecked()
config.setting['cover_file_convert_to_format'] = self.ui.convert_file_format.currentText()
register_options_page(CoverProcessingOptionsPage) register_options_page(CoverProcessingOptionsPage)

View File

@@ -32,12 +32,16 @@ from picard.coverart.processing.filters import (
size_filter, size_filter,
size_metadata_filter, size_metadata_filter,
) )
from picard.coverart.processing.processors import ResizeImage from picard.coverart.processing.processors import (
ConvertImage,
ResizeImage,
)
from picard.extension_points.cover_art_processors import ( from picard.extension_points.cover_art_processors import (
CoverArtProcessingError, CoverArtProcessingError,
ProcessingImage, ProcessingImage,
ProcessingTarget, ProcessingTarget,
) )
from picard.util import imageinfo
def create_fake_image(width, height, image_format): def create_fake_image(width, height, image_format):
@@ -59,9 +63,9 @@ class ImageFiltersTest(PicardTestCase):
self.set_config_values(settings) self.set_config_values(settings)
def test_filter_by_size(self): def test_filter_by_size(self):
image1 = create_fake_image(400, 600, "png") image1 = create_fake_image(400, 600, 'png')
image2 = create_fake_image(500, 500, "jpeg") image2 = create_fake_image(500, 500, 'jpeg')
image3 = create_fake_image(600, 600, "bmp") image3 = create_fake_image(600, 600, 'bmp')
self.assertFalse(size_filter(image1)) self.assertFalse(size_filter(image1))
self.assertTrue(size_filter(image2)) self.assertTrue(size_filter(image2))
self.assertTrue(size_filter(image3)) self.assertTrue(size_filter(image3))
@@ -88,6 +92,8 @@ class ImageProcessorsTest(PicardTestCase):
'cover_tags_resize_target_height': 500, 'cover_tags_resize_target_height': 500,
'cover_tags_stretch': False, 'cover_tags_stretch': False,
'cover_tags_crop': False, 'cover_tags_crop': False,
'cover_tags_convert_images': False,
'cover_tags_convert_to_format': 'jpeg',
'cover_file_scale_up': True, 'cover_file_scale_up': True,
'cover_file_scale_down': True, 'cover_file_scale_down': True,
'cover_file_resize_use_width': True, 'cover_file_resize_use_width': True,
@@ -98,11 +104,13 @@ class ImageProcessorsTest(PicardTestCase):
'cover_file_crop': False, 'cover_file_crop': False,
'save_images_to_tags': True, 'save_images_to_tags': True,
'save_images_to_files': True, 'save_images_to_files': True,
'cover_file_convert_images': False,
'cover_file_convert_to_format': 'jpeg',
} }
def _check_image_processors(self, size, expected_tags_size, expected_file_size=None): def _check_image_processors(self, size, expected_tags_size, expected_file_size=None):
coverartimage = CoverArtImage() coverartimage = CoverArtImage()
image = create_fake_image(size[0], size[1], "jpg") image = create_fake_image(size[0], size[1], 'jpg')
run_image_processors(image, coverartimage) run_image_processors(image, coverartimage)
tags_size = (coverartimage.width, coverartimage.height) tags_size = (coverartimage.width, coverartimage.height)
if config.setting['save_images_to_tags']: if config.setting['save_images_to_tags']:
@@ -116,7 +124,7 @@ class ImageProcessorsTest(PicardTestCase):
else: else:
self.assertIsNone(coverartimage.external_file_coverart) self.assertIsNone(coverartimage.external_file_coverart)
extension = coverartimage.extension[1:] extension = coverartimage.extension[1:]
self.assertEqual(extension, "jpg") self.assertEqual(extension, 'jpg')
def test_image_processors_save_to_both(self): def test_image_processors_save_to_both(self):
self.set_config_values(self.settings) self.set_config_values(self.settings)
@@ -151,7 +159,7 @@ class ImageProcessorsTest(PicardTestCase):
self.set_config_values(self.settings) self.set_config_values(self.settings)
def _check_resize_image(self, size, expected_size): def _check_resize_image(self, size, expected_size):
image = ProcessingImage(create_fake_image(size[0], size[1], "jpg")) image = ProcessingImage(create_fake_image(size[0], size[1], 'jpg'))
processor = ResizeImage() processor = ResizeImage()
processor.run(image, ProcessingTarget.TAGS) processor.run(image, ProcessingTarget.TAGS)
new_size = (image.get_result().width(), image.get_result().height()) new_size = (image.get_result().width(), image.get_result().height())
@@ -167,7 +175,7 @@ class ImageProcessorsTest(PicardTestCase):
def test_scale_down_only_width(self): def test_scale_down_only_width(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_resize_use_height"] = False settings['cover_tags_resize_use_height'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 1000), (500, 500)) self._check_resize_image((1000, 1000), (500, 500))
self._check_resize_image((1000, 500), (500, 250)) self._check_resize_image((1000, 500), (500, 250))
@@ -176,7 +184,7 @@ class ImageProcessorsTest(PicardTestCase):
def test_scale_down_only_height(self): def test_scale_down_only_height(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_resize_use_width"] = False settings['cover_tags_resize_use_width'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 1000), (500, 500)) self._check_resize_image((1000, 1000), (500, 500))
self._check_resize_image((1000, 500), (1000, 500)) self._check_resize_image((1000, 500), (1000, 500))
@@ -191,7 +199,7 @@ class ImageProcessorsTest(PicardTestCase):
def test_scale_up_only_width(self): def test_scale_up_only_width(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_resize_use_height"] = False settings['cover_tags_resize_use_height'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((250, 250), (500, 500)) self._check_resize_image((250, 250), (500, 500))
self._check_resize_image((400, 500), (500, 625)) self._check_resize_image((400, 500), (500, 625))
@@ -200,7 +208,7 @@ class ImageProcessorsTest(PicardTestCase):
def test_scale_up_only_height(self): def test_scale_up_only_height(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_resize_use_width"] = False settings['cover_tags_resize_use_width'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((250, 250), (500, 500)) self._check_resize_image((250, 250), (500, 500))
self._check_resize_image((400, 500), (400, 500)) self._check_resize_image((400, 500), (400, 500))
@@ -209,15 +217,15 @@ class ImageProcessorsTest(PicardTestCase):
def test_scale_priority(self): def test_scale_priority(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_resize_target_width"] = 500 settings['cover_tags_resize_target_width'] = 500
settings["cover_tags_resize_target_height"] = 1000 settings['cover_tags_resize_target_height'] = 1000
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((750, 750), (500, 500)) self._check_resize_image((750, 750), (500, 500))
self.set_config_values(self.settings) self.set_config_values(self.settings)
def test_stretch_both_dimensions(self): def test_stretch_both_dimensions(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_stretch"] = True settings['cover_tags_stretch'] = True
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (500, 500)) self._check_resize_image((1000, 100), (500, 500))
self._check_resize_image((200, 500), (500, 500)) self._check_resize_image((200, 500), (500, 500))
@@ -226,8 +234,8 @@ class ImageProcessorsTest(PicardTestCase):
def test_stretch_only_width(self): def test_stretch_only_width(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_stretch"] = True settings['cover_tags_stretch'] = True
settings["cover_tags_resize_use_height"] = False settings['cover_tags_resize_use_height'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (500, 100)) self._check_resize_image((1000, 100), (500, 100))
self._check_resize_image((200, 500), (500, 500)) self._check_resize_image((200, 500), (500, 500))
@@ -236,8 +244,8 @@ class ImageProcessorsTest(PicardTestCase):
def test_stretch_only_height(self): def test_stretch_only_height(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_stretch"] = True settings['cover_tags_stretch'] = True
settings["cover_tags_resize_use_width"] = False settings['cover_tags_resize_use_width'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (1000, 500)) self._check_resize_image((1000, 100), (1000, 500))
self._check_resize_image((200, 500), (200, 500)) self._check_resize_image((200, 500), (200, 500))
@@ -246,7 +254,7 @@ class ImageProcessorsTest(PicardTestCase):
def test_crop_both_dimensions(self): def test_crop_both_dimensions(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_crop"] = True settings['cover_tags_crop'] = True
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (500, 500)) self._check_resize_image((1000, 100), (500, 500))
self._check_resize_image((750, 1000), (500, 500)) self._check_resize_image((750, 1000), (500, 500))
@@ -255,8 +263,8 @@ class ImageProcessorsTest(PicardTestCase):
def test_crop_only_width(self): def test_crop_only_width(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_crop"] = True settings['cover_tags_crop'] = True
settings["cover_tags_resize_use_height"] = False settings['cover_tags_resize_use_height'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (500, 100)) self._check_resize_image((1000, 100), (500, 100))
self._check_resize_image((750, 1000), (500, 1000)) self._check_resize_image((750, 1000), (500, 1000))
@@ -265,16 +273,34 @@ class ImageProcessorsTest(PicardTestCase):
def test_crop_only_height(self): def test_crop_only_height(self):
settings = copy(self.settings) settings = copy(self.settings)
settings["cover_tags_crop"] = True settings['cover_tags_crop'] = True
settings["cover_tags_resize_use_width"] = False settings['cover_tags_resize_use_width'] = False
self.set_config_values(settings) self.set_config_values(settings)
self._check_resize_image((1000, 100), (1000, 500)) self._check_resize_image((1000, 100), (1000, 500))
self._check_resize_image((750, 1000), (750, 500)) self._check_resize_image((750, 1000), (750, 500))
self._check_resize_image((250, 1000), (250, 500)) self._check_resize_image((250, 1000), (250, 500))
self.set_config_values(self.settings) self.set_config_values(self.settings)
def _check_convert_image(self, format, expected_format):
image = ProcessingImage(create_fake_image(100, 100, format))
processor = ConvertImage()
processor.run(image, ProcessingTarget.TAGS)
new_image = image.get_result(default_format=True)
new_info = imageinfo.identify(new_image)
self.assertIn(new_info.format, ConvertImage._format_aliases[expected_format])
def test_format_conversion(self):
settings = copy(self.settings)
settings['cover_tags_convert_images'] = True
formats = ['jpeg', 'png', 'webp', 'tiff']
for format in formats:
settings['cover_tags_convert_to_format'] = format
self.set_config_values(settings)
self._check_convert_image('jpeg', format)
self.set_config_values(self.settings)
def test_identification_error(self): def test_identification_error(self):
image = create_fake_image(0, 0, "jpg") image = create_fake_image(0, 0, 'jpg')
coverartimage = CoverArtImage() coverartimage = CoverArtImage()
with self.assertRaises(CoverArtProcessingError): with self.assertRaises(CoverArtProcessingError):
run_image_processors(image, coverartimage) run_image_processors(image, coverartimage)

View File

@@ -6,8 +6,8 @@
<rect> <rect>
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>478</width> <width>529</width>
<height>423</height> <height>467</height>
</rect> </rect>
</property> </property>
<property name="windowTitle"> <property name="windowTitle">
@@ -564,6 +564,107 @@
</layout> </layout>
</widget> </widget>
</item> </item>
<item>
<widget class="QGroupBox" name="converting">
<property name="title">
<string>Convert images to the given format</string>
</property>
<property name="checkable">
<bool>false</bool>
</property>
<property name="checked">
<bool>false</bool>
</property>
<layout class="QHBoxLayout" name="horizontalLayout_12">
<item>
<widget class="QGroupBox" name="convert_tags">
<property name="title">
<string>Convert images saved to tags</string>
</property>
<property name="checkable">
<bool>true</bool>
</property>
<layout class="QHBoxLayout" name="horizontalLayout_13">
<item>
<widget class="QLabel" name="convert_tags_label">
<property name="text">
<string>New format:</string>
</property>
</widget>
</item>
<item>
<widget class="QComboBox" name="convert_tags_format">
<item>
<property name="text">
<string>JPEG</string>
</property>
</item>
<item>
<property name="text">
<string>PNG</string>
</property>
</item>
<item>
<property name="text">
<string>WebP</string>
</property>
</item>
<item>
<property name="text">
<string>TIFF</string>
</property>
</item>
</widget>
</item>
</layout>
</widget>
</item>
<item>
<widget class="QGroupBox" name="convert_file">
<property name="title">
<string>Convert images saved to files</string>
</property>
<property name="checkable">
<bool>true</bool>
</property>
<layout class="QHBoxLayout" name="horizontalLayout_14">
<item>
<widget class="QLabel" name="convert_file_label">
<property name="text">
<string>New format:</string>
</property>
</widget>
</item>
<item>
<widget class="QComboBox" name="convert_file_format">
<item>
<property name="text">
<string>JPEG</string>
</property>
</item>
<item>
<property name="text">
<string>PNG</string>
</property>
</item>
<item>
<property name="text">
<string>WebP</string>
</property>
</item>
<item>
<property name="text">
<string>TIFF</string>
</property>
</item>
</widget>
</item>
</layout>
</widget>
</item>
</layout>
</widget>
</item>
<item> <item>
<spacer name="verticalSpacer"> <spacer name="verticalSpacer">
<property name="orientation"> <property name="orientation">