-
-
Notifications
You must be signed in to change notification settings - Fork 392
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
PICARD-1001 Rework local cover art handling #1934
base: master
Are you sure you want to change the base?
Conversation
Local cover art as a provider doesn't make sense - local cover art is a property of a file, same as tag cover art. Move the handling of local cover art to the File class. TODO: Need to resolve how to configure load ordering of tag cover art vs local cover art. Also, there needs to be a config ui written. In the current state, the local_cover_regex setting is re-used, and a new load_local_cover_art boolean is used to enable the feature. This will probably fix many of the causes of PICARD-1001.
(Pushed a minor updates, this adds the settings back that had disappeared when I deleted the provider ui code, and fixed initialization of the local file cover art object from url) |
Ah, looking into it, this doesn't actually solve the problem. The issue I am having now is that when a release has a local file loaded, and also gets a cover image from CAA, it shows up as "changed" even if both images are the same image. I'm not sure what the solution to this is. (in this example, top cover is from CAA provider, bottom one is loaded from local file) How is this resolved for the case of images saved in tags? As far as I can tell, i'm using the same codepaths here - does that also fail to match? |
Looking at the contents of
So I think what's needed here is a better equality or comparison method on the picard.util.ImageList or maybe picard.coverart.image.CoverArtImage implementation which can figure out that the two images are the same despite being different classes, so a save is not required. |
Interesting… took a look at the Either we need to have some special handling for the case of the frontiest image that ignores multiple types, or I can set I'll push the |
This fixes the comparison with CAA art when the CAA art has multiple types. (For example, comparing cover.jpg with a CAA "cover, booklet" image)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
First off, your observations are correct. For the reasons you stated "local cover art" never worked as it should and moving this away from being a cover art provider is definitely right.
But we must be aware that there are differences in behavior with this change. One is that Picard will overwrite the cover art with cover loaded from providers. Before that you could put local cover art provider to be the first one, and it would mean local cover art ist used and other providers not even called. I think we should do the following:
- Have an option to "Do not overwrite existing images", which will keep the images loaded for a file even if different cover art has been loaded.
- Keep the "Local Files" provider, but rename it to "Local Files (deprecated)". That way people relying on this can continue using this for now, and we get some time to see what users need. Alternative to this would be to put it in a plugin instead. But my preference is to keep it in the code for the next release.
- Maybe we should have a settings kigration that nebales
load_local_cover_art
if the provider is active? But not sure about this.
Regarding the UI I can help out and do the necessary changes this evening. I can also do an actual code review then.
This approach doesn't let the user set the order of preference for cover art sources. Before any review, PR needs to pass tests. Also I disagree with statement That said, it's clear cover art code needs improvements and any patch is welcome. |
So, the way I think about local cover art files is that they should be treated equivalently to cover art embedded into tags in a file. Aligning the picard code to match that was the goal of this PR. I agree that an option to preserve existing cover art would be desirable. Such an option would be applicable both to local cover art files and embedded cover art in tags. The particular goals of this PR were to:
|
Agreed. The local cover art provider definitely did work for specific use cases. I think the main problem with it is that by design it can only run if files are attached to the release already when it gets loaded. That means it does not work for quite a few cases. But one use case worked quite well: I have the local cover art provider on top of my provider list. That way when I retag files the existing images are used. That means it remembers my choice of cover art, even if when tagging I chose a different cover art than what my primary provider would have given.
I fully agree with that. I often had the opposite desire, though: Have embedded images work the same as the local cover art provider. I think we must provide some use cases which users might have with local cover art. Some that come to my mind:
I would claim that Picard currently does not support any of the above use cases well, and both this implementation and the current local cover art provider solve different use cases differently well. Local cover art provider does not solve 1. It can solve 2, 3 and 4, but only under certain circumstances (files are available when a release gets loaded). It cannot solve 2, 3 and 4 for embedded cover art. This PR does solve 1. It does not solve 2 or 3. It can be used for 4 given that there is no cover art loaded from providers. If we add an option to prefer local or embedded files and not replace it with provider files (that means "Use original cover art" is the default behavior), and maybe also an option to toggle loading embedded images on and off then it would also solve 2 and 4. Not sure how it could solve 3 (which is based on provider order). What did I forget? |
@@ -250,6 +260,48 @@ def _loading_finished(self, callback, result=None, error=None): | |||
self.update() | |||
callback(self) | |||
|
|||
_types_split_re = re.compile('[^a-z0-9]', re.IGNORECASE) | |||
_known_types = set([t['name'] for t in CAA_TYPES]) | |||
_default_types = ['front'] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This could be written directly as {t['name'] for t in CAA_TYPES}
_default_types = ['front'] | ||
|
||
def get_types(self, string): | ||
found = set([x.lower() for x in self._types_split_re.split(string) if x]) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can also be done with a set comprehension {x.lower() for x in self._types_split_re.split(string) if x}
@@ -480,3 +480,6 @@ def __init__(self, filepath, types=None, comment='', | |||
super().__init__(url=url, types=types, comment=comment) | |||
self.support_types = support_types | |||
self.support_multi_types = support_multi_types | |||
path = self.url.toLocalFile() | |||
with open(path, 'rb') as file: | |||
self.set_data(file.read()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think loading the data here makes sense, but this needs fixing a couple of tests that currently fail due to this. The LocalFileCoverArtImageTest
currently works without files, should be updated to work on temporary files.
You summed it up pretty well, I agree on all 4 cases. First, we need to define what are local cover art files, they may have a way to determine types or not. Also they may not be at same directory level, I can imagine someone having following structure:
Also source image files can be named in infinite number of ways.
We have a regex to match those, but I think it's insufficient. MB doesn't have track cover art support yet. So, I think we need a way to "map" local cover art, to "remote" cover art, and decide how to handle multiple images of the same type. |
I think I pretty much agree with this - an option to preserve existing cover art would be a necessity before any change like this gets merged. The use case for 3 seems to be fairly niche? And yeah, that would be hard to do. The particular concern I had was that there's no ability to set the relative priority of cover art from local file vs. embedded cover art right now. Adding a whole second priority list to the UI for this doesn't seem like a good idea, so I'm kind of stumped on that front. … another possible option might be to have "cover art providers" actually have multiple possible hook points? So when a cover art provider registers itself, it could register itself to run at either the Any opinions on that? If this is done, it might make sense to have tag cover art be in the providers list as well to allow setting its relative priority (even tho it's a special case, since I think the implementation for that might need to stay in the specific file format implementations) |
Summary
Problem
Local cover art as a provider doesn't make sense - local cover art is a property of a file, same as tag cover art. Problems caused by local cover art being treated as a provider include incorrectly indicating that files have been modified, or if an album was loaded without any files associated (for example via a tagger link on musicbrainz.org), no cover art at all.
Solution
Move the handling of local cover art to the File class and delete the Local File cover art provider
Action
I don't know how to write Qt UI, so someone familiar with that will have to update the UI for this change.
In the current state, the
local_cover_regex
setting is re-used, and a newload_local_cover_art
boolean is used to enable the feature. The new setting should default to the same as the enabled status of the Local FIle cover art provider used to be.Some decision has to be made on how to result ordering of loading cover art from tags vs. local files. Currently local file art loading is skipped if cover art was loaded from tags.