import collections import itertools import random import typing from hydrus.core import HydrusConstants as HC from hydrus.core import HydrusText from hydrus.core import HydrusData from hydrus.core import HydrusExceptions from hydrus.core import HydrusGlobals as HG from hydrus.core import HydrusImageHandling from hydrus.core import HydrusSerialisable from hydrus.client import ClientConstants as CC from hydrus.client import ClientData from hydrus.client import ClientLocation from hydrus.client import ClientSearch from hydrus.client.media import ClientMediaManagers from hydrus.client.media import ClientMediaResult from hydrus.client.metadata import ClientTags hashes_to_jpeg_quality = {} hashes_to_pixel_hashes = {} def FilterServiceKeysToContentUpdates( full_service_keys_to_content_updates, hashes ): if not isinstance( hashes, set ): hashes = set( hashes ) filtered_service_keys_to_content_updates = collections.defaultdict( list ) for ( service_key, full_content_updates ) in full_service_keys_to_content_updates.items(): filtered_content_updates = [] for content_update in full_content_updates: if not hashes.isdisjoint( content_update.GetHashes() ): filtered_content_updates.append( content_update ) if len( filtered_content_updates ) > 0: filtered_service_keys_to_content_updates[ service_key ] = filtered_content_updates return filtered_service_keys_to_content_updates def FlattenMedia( media_list ): flat_media = [] for media in media_list: if media.IsCollection(): flat_media.extend( media.GetFlatMedia() ) else: flat_media.append( media ) return flat_media def GetDuplicateComparisonScore( shown_media, comparison_media ): statements_and_scores = GetDuplicateComparisonStatements( shown_media, comparison_media ) total_score = sum( ( score for ( statement, score ) in statements_and_scores.values() ) ) return total_score def GetDuplicateComparisonStatements( shown_media, comparison_media ): new_options = HG.client_controller.new_options duplicate_comparison_score_higher_jpeg_quality = new_options.GetInteger( 'duplicate_comparison_score_higher_jpeg_quality' ) duplicate_comparison_score_much_higher_jpeg_quality = new_options.GetInteger( 'duplicate_comparison_score_much_higher_jpeg_quality' ) duplicate_comparison_score_higher_filesize = new_options.GetInteger( 'duplicate_comparison_score_higher_filesize' ) duplicate_comparison_score_much_higher_filesize = new_options.GetInteger( 'duplicate_comparison_score_much_higher_filesize' ) duplicate_comparison_score_higher_resolution = new_options.GetInteger( 'duplicate_comparison_score_higher_resolution' ) duplicate_comparison_score_much_higher_resolution = new_options.GetInteger( 'duplicate_comparison_score_much_higher_resolution' ) duplicate_comparison_score_more_tags = new_options.GetInteger( 'duplicate_comparison_score_more_tags' ) duplicate_comparison_score_older = new_options.GetInteger( 'duplicate_comparison_score_older' ) duplicate_comparison_score_nicer_ratio = new_options.GetInteger( 'duplicate_comparison_score_nicer_ratio' ) # statements_and_scores = {} s_hash = shown_media.GetHash() c_hash = comparison_media.GetHash() s_mime = shown_media.GetMime() c_mime = comparison_media.GetMime() # size s_size = shown_media.GetSize() c_size = comparison_media.GetSize() is_a_pixel_dupe = False if shown_media.IsStaticImage() and comparison_media.IsStaticImage() and shown_media.GetResolution() == comparison_media.GetResolution(): global hashes_to_pixel_hashes if s_hash not in hashes_to_pixel_hashes: path = HG.client_controller.client_files_manager.GetFilePath( s_hash, s_mime ) hashes_to_pixel_hashes[ s_hash ] = HydrusImageHandling.GetImagePixelHash( path, s_mime ) if c_hash not in hashes_to_pixel_hashes: path = HG.client_controller.client_files_manager.GetFilePath( c_hash, c_mime ) hashes_to_pixel_hashes[ c_hash ] = HydrusImageHandling.GetImagePixelHash( path, c_mime ) s_pixel_hash = hashes_to_pixel_hashes[ s_hash ] c_pixel_hash = hashes_to_pixel_hashes[ c_hash ] if s_pixel_hash == c_pixel_hash: is_a_pixel_dupe = True if s_mime == HC.IMAGE_PNG and c_mime != HC.IMAGE_PNG: statement = 'this is a pixel-for-pixel duplicate png!' score = -100 elif s_mime != HC.IMAGE_PNG and c_mime == HC.IMAGE_PNG: statement = 'other file is a pixel-for-pixel duplicate png!' score = 100 else: statement = 'images are pixel-for-pixel duplicates!' score = 0 statements_and_scores[ 'pixel_duplicates' ] = ( statement, score ) if s_size != c_size: absolute_size_ratio = max( s_size, c_size ) / min( s_size, c_size ) if absolute_size_ratio > 2.0: if s_size > c_size: operator = '>>' score = duplicate_comparison_score_much_higher_filesize else: operator = '<<' score = -duplicate_comparison_score_much_higher_filesize elif absolute_size_ratio > 1.05: if s_size > c_size: operator = '>' score = duplicate_comparison_score_higher_filesize else: operator = '<' score = -duplicate_comparison_score_higher_filesize else: operator = CC.UNICODE_ALMOST_EQUAL_TO score = 0 if is_a_pixel_dupe: score = 0 statement = '{} {} {}'.format( HydrusData.ToHumanBytes( s_size ), operator, HydrusData.ToHumanBytes( c_size ) ) statements_and_scores[ 'filesize' ] = ( statement, score ) # higher/same res s_resolution = shown_media.GetResolution() c_resolution = comparison_media.GetResolution() if s_resolution is not None and c_resolution is not None and s_resolution != c_resolution: s_res = shown_media.GetResolution() c_res = comparison_media.GetResolution() ( s_w, s_h ) = s_res ( c_w, c_h ) = c_res resolution_ratio = ( s_w * s_h ) / ( c_w * c_h ) if resolution_ratio == 1.0: operator = '!=' score = 0 elif resolution_ratio > 2.0: operator = '>>' score = duplicate_comparison_score_much_higher_resolution elif resolution_ratio > 1.00: operator = '>' score = duplicate_comparison_score_higher_resolution elif resolution_ratio < 0.5: operator = '<<' score = -duplicate_comparison_score_much_higher_resolution else: operator = '<' score = -duplicate_comparison_score_higher_resolution if s_res in HC.NICE_RESOLUTIONS: s_string = HC.NICE_RESOLUTIONS[ s_res ] else: s_string = HydrusData.ConvertResolutionToPrettyString( s_resolution ) if s_w % 2 == 1 or s_h % 2 == 1: s_string += ' (unusual)' if c_res in HC.NICE_RESOLUTIONS: c_string = HC.NICE_RESOLUTIONS[ c_res ] else: c_string = HydrusData.ConvertResolutionToPrettyString( c_resolution ) if c_w % 2 == 1 or c_h % 2 == 1: c_string += ' (unusual)' statement = '{} {} {}'.format( s_string, operator, c_string ) statements_and_scores[ 'resolution' ] = ( statement, score ) # s_ratio = s_w / s_h c_ratio = c_w / c_h s_nice = s_ratio in HC.NICE_RATIOS c_nice = c_ratio in HC.NICE_RATIOS if s_nice or c_nice: if s_nice: s_string = HC.NICE_RATIOS[ s_ratio ] else: s_string = 'unusual' if c_nice: c_string = HC.NICE_RATIOS[ c_ratio ] else: c_string = 'unusual' if s_nice and c_nice: operator = '-' score = 0 elif s_nice: operator = '>' score = duplicate_comparison_score_nicer_ratio elif c_nice: operator = '<' score = -duplicate_comparison_score_nicer_ratio if s_string == c_string: statement = 'both {}'.format( s_string ) else: statement = '{} {} {}'.format( s_string, operator, c_string ) statements_and_scores[ 'ratio' ] = ( statement, score ) # same/diff mime if s_mime != c_mime: statement = '{} vs {}'.format( HC.mime_string_lookup[ s_mime ], HC.mime_string_lookup[ c_mime ] ) score = 0 statements_and_scores[ 'mime' ] = ( statement, score ) # more tags s_num_tags = len( shown_media.GetTagsManager().GetCurrentAndPending( CC.COMBINED_TAG_SERVICE_KEY, ClientTags.TAG_DISPLAY_ACTUAL ) ) c_num_tags = len( comparison_media.GetTagsManager().GetCurrentAndPending( CC.COMBINED_TAG_SERVICE_KEY, ClientTags.TAG_DISPLAY_ACTUAL ) ) if s_num_tags != c_num_tags: if s_num_tags > 0 and c_num_tags > 0: if s_num_tags > c_num_tags: operator = '>' score = duplicate_comparison_score_more_tags else: operator = '<' score = -duplicate_comparison_score_more_tags elif s_num_tags > 0: operator = '>>' score = duplicate_comparison_score_more_tags elif c_num_tags > 0: operator = '<<' score = -duplicate_comparison_score_more_tags statement = '{} tags {} {} tags'.format( HydrusData.ToHumanInt( s_num_tags ), operator, HydrusData.ToHumanInt( c_num_tags ) ) statements_and_scores[ 'num_tags' ] = ( statement, score ) # older s_ts = shown_media.GetLocationsManager().GetCurrentTimestamp( CC.COMBINED_LOCAL_FILE_SERVICE_KEY ) c_ts = comparison_media.GetLocationsManager().GetCurrentTimestamp( CC.COMBINED_LOCAL_FILE_SERVICE_KEY ) one_month = 86400 * 30 if s_ts is not None and c_ts is not None and abs( s_ts - c_ts ) > one_month: if s_ts < c_ts: operator = 'older than' score = duplicate_comparison_score_older else: operator = 'newer than' score = -duplicate_comparison_score_older if is_a_pixel_dupe: score = 0 statement = '{}, {} {}'.format( ClientData.TimestampToPrettyTimeDelta( s_ts, history_suffix = ' old' ), operator, ClientData.TimestampToPrettyTimeDelta( c_ts, history_suffix = ' old' ) ) statements_and_scores[ 'time_imported' ] = ( statement, score ) if s_mime == HC.IMAGE_JPEG and c_mime == HC.IMAGE_JPEG: global hashes_to_jpeg_quality if s_hash not in hashes_to_jpeg_quality: path = HG.client_controller.client_files_manager.GetFilePath( s_hash, s_mime ) hashes_to_jpeg_quality[ s_hash ] = HydrusImageHandling.GetJPEGQuantizationQualityEstimate( path ) if c_hash not in hashes_to_jpeg_quality: path = HG.client_controller.client_files_manager.GetFilePath( c_hash, c_mime ) hashes_to_jpeg_quality[ c_hash ] = HydrusImageHandling.GetJPEGQuantizationQualityEstimate( path ) ( s_label, s_jpeg_quality ) = hashes_to_jpeg_quality[ s_hash ] ( c_label, c_jpeg_quality ) = hashes_to_jpeg_quality[ c_hash ] score = 0 if s_label != c_label: if c_jpeg_quality is None or s_jpeg_quality is None: score = 0 else: # other way around, low score is good here quality_ratio = c_jpeg_quality / s_jpeg_quality if quality_ratio > 2.0: score = duplicate_comparison_score_much_higher_jpeg_quality elif quality_ratio > 1.0: score = duplicate_comparison_score_higher_jpeg_quality elif quality_ratio < 0.5: score = -duplicate_comparison_score_much_higher_jpeg_quality else: score = -duplicate_comparison_score_higher_jpeg_quality statement = '{} vs {} jpeg quality'.format( s_label, c_label ) statements_and_scores[ 'jpeg_quality' ] = ( statement, score ) return statements_and_scores def GetMediasTags( pool, tag_service_key, tag_display_type, content_statuses ): tags_managers = [] for media in pool: if media.IsCollection(): tags_managers.extend( media.GetSingletonsTagsManagers() ) else: tags_managers.append( media.GetTagsManager() ) tags = set() for tags_manager in tags_managers: statuses_to_tags = tags_manager.GetStatusesToTags( tag_service_key, tag_display_type ) for content_status in content_statuses: tags.update( statuses_to_tags[ content_status ] ) return tags def GetMediaResultsTagCount( media_results, tag_service_key, tag_display_type ): tags_managers = [ media_result.GetTagsManager() for media_result in media_results ] return GetTagsManagersTagCount( tags_managers, tag_service_key, tag_display_type ) def GetMediasTagCount( pool, tag_service_key, tag_display_type ): tags_managers = [] for media in pool: if media.IsCollection(): tags_managers.extend( media.GetSingletonsTagsManagers() ) else: tags_managers.append( media.GetTagsManager() ) return GetTagsManagersTagCount( tags_managers, tag_service_key, tag_display_type ) def GetTagsManagersTagCount( tags_managers, tag_service_key, tag_display_type ): current_tags_to_count = collections.Counter() deleted_tags_to_count = collections.Counter() pending_tags_to_count = collections.Counter() petitioned_tags_to_count = collections.Counter() for tags_manager in tags_managers: statuses_to_tags = tags_manager.GetStatusesToTags( tag_service_key, tag_display_type ) current_tags_to_count.update( statuses_to_tags[ HC.CONTENT_STATUS_CURRENT ] ) deleted_tags_to_count.update( statuses_to_tags[ HC.CONTENT_STATUS_DELETED ] ) pending_tags_to_count.update( statuses_to_tags[ HC.CONTENT_STATUS_PENDING ] ) petitioned_tags_to_count.update( statuses_to_tags[ HC.CONTENT_STATUS_PETITIONED ] ) return ( current_tags_to_count, deleted_tags_to_count, pending_tags_to_count, petitioned_tags_to_count ) class Media( object ): def __init__( self ): self._id = HydrusData.GenerateKey() self._id_hash = self._id.__hash__() def __eq__( self, other ): if isinstance( other, Media ): return self.__hash__() == other.__hash__() return NotImplemented def __hash__( self ): return self._id_hash def __ne__( self, other ): return self.__hash__() != other.__hash__() def GetDisplayMedia( self ) -> 'Media': raise NotImplementedError() def GetDuration( self ) -> typing.Optional[ int ]: raise NotImplementedError() def GetFileViewingStatsManager( self ) -> ClientMediaManagers.FileViewingStatsManager: raise NotImplementedError() def GetHash( self ) -> bytes: raise NotImplementedError() def GetHashes( self, has_location = None, discriminant = None, not_uploaded_to = None, ordered = False ): raise NotImplementedError() def GetLocationsManager( self ) -> ClientMediaManagers.LocationsManager: raise NotImplementedError() def GetMime( self ) -> int: raise NotImplementedError() def GetNumFiles( self ) -> int: raise NotImplementedError() def GetNumFrames( self ) -> typing.Optional[ int ]: raise NotImplementedError() def GetNumInbox( self ) -> int: raise NotImplementedError() def GetNumWords( self ) -> typing.Optional[ int ]: raise NotImplementedError() def GetCurrentTimestamp( self, service_key: bytes ) -> typing.Optional[ int ]: raise NotImplementedError() def GetDeletedTimestamps( self, service_key: bytes ) -> typing.Tuple[ typing.Optional[ int ], typing.Optional[ int ] ]: raise NotImplementedError() def GetPrettyInfoLines( self ) -> typing.List[ str ]: raise NotImplementedError() def GetRatingsManager( self ) -> ClientMediaManagers.RatingsManager: raise NotImplementedError() def GetResolution( self ) -> typing.Tuple[ int, int ]: raise NotImplementedError() def GetSize( self ) -> int: raise NotImplementedError() def GetTagsManager( self ) -> ClientMediaManagers.TagsManager: raise NotImplementedError() def HasAnyOfTheseHashes( self, hashes ) -> bool: raise NotImplementedError() def HasArchive( self ) -> bool: raise NotImplementedError() def HasAudio( self ) -> bool: raise NotImplementedError() def HasDuration( self ) -> bool: raise NotImplementedError() def HasImages( self ) -> bool: raise NotImplementedError() def HasInbox( self ) -> bool: raise NotImplementedError() def HasNotes( self ) -> bool: raise NotImplementedError() def IsCollection( self ) -> bool: raise NotImplementedError() def IsImage( self ) -> bool: raise NotImplementedError() def IsSizeDefinite( self ) -> bool: raise NotImplementedError() def UpdateFileInfo( self, hashes_to_media_results ): raise NotImplementedError() class MediaCollect( HydrusSerialisable.SerialisableBase ): SERIALISABLE_TYPE = HydrusSerialisable.SERIALISABLE_TYPE_MEDIA_COLLECT SERIALISABLE_NAME = 'Media Collect' SERIALISABLE_VERSION = 1 def __init__( self, namespaces = None, rating_service_keys = None, collect_unmatched = None ): if namespaces is None: namespaces = [] if rating_service_keys is None: rating_service_keys = [] if collect_unmatched is None: collect_unmatched = True self.namespaces = namespaces self.rating_service_keys = rating_service_keys self.collect_unmatched = collect_unmatched def _GetSerialisableInfo( self ): serialisable_rating_service_keys = [ key.hex() for key in self.rating_service_keys ] return ( self.namespaces, serialisable_rating_service_keys, self.collect_unmatched ) def _InitialiseFromSerialisableInfo( self, serialisable_info ): ( self.namespaces, serialisable_rating_service_keys, self.collect_unmatched ) = serialisable_info self.rating_service_keys = [ bytes.fromhex( serialisable_key ) for serialisable_key in serialisable_rating_service_keys ] def DoesACollect( self ): return len( self.namespaces ) > 0 or len( self.rating_service_keys ) > 0 def ToString( self ): s_list = list( self.namespaces ) s_list.extend( [ HG.client_controller.services_manager.GetName( service_key ) for service_key in self.rating_service_keys if HG.client_controller.services_manager.ServiceExists( service_key ) ] ) if len( s_list ) == 0: return 'no collections' else: return ', '.join( s_list ) HydrusSerialisable.SERIALISABLE_TYPES_TO_OBJECT_TYPES[ HydrusSerialisable.SERIALISABLE_TYPE_MEDIA_COLLECT ] = MediaCollect class MediaList( object ): def __init__( self, location_context: ClientLocation.LocationContext, media_results ): hashes_seen = set() media_results_dedupe = [] for media_result in media_results: hash = media_result.GetHash() if hash in hashes_seen: continue media_results_dedupe.append( media_result ) hashes_seen.add( hash ) media_results = media_results_dedupe self._location_context = location_context self._hashes = set() self._hashes_ordered = [] self._hashes_to_singleton_media = {} self._hashes_to_collected_media = {} self._media_sort = MediaSort( ( 'system', CC.SORT_FILES_BY_FILESIZE ), CC.SORT_ASC ) self._media_collect = MediaCollect() self._sorted_media = SortedList( [ self._GenerateMediaSingleton( media_result ) for media_result in media_results ] ) self._selected_media = set() self._singleton_media = set( self._sorted_media ) self._collected_media = set() self._RecalcHashes() def __len__( self ): return len( self._singleton_media ) + sum( map( len, self._collected_media ) ) def _CalculateCollectionKeysToMedias( self, media_collect, medias ): keys_to_medias = collections.defaultdict( list ) namespaces_to_collect_by = list( media_collect.namespaces ) ratings_to_collect_by = list( media_collect.rating_service_keys ) for media in medias: if len( namespaces_to_collect_by ) > 0: namespace_key = media.GetTagsManager().GetNamespaceSlice( namespaces_to_collect_by, ClientTags.TAG_DISPLAY_ACTUAL ) else: namespace_key = frozenset() if len( ratings_to_collect_by ) > 0: rating_key = media.GetRatingsManager().GetRatingSlice( ratings_to_collect_by ) else: rating_key = frozenset() keys_to_medias[ ( namespace_key, rating_key ) ].append( media ) return keys_to_medias def _GenerateMediaCollection( self, media_results ): return MediaCollection( self._location_context, media_results ) def _GenerateMediaSingleton( self, media_result ): return MediaSingleton( media_result ) def _GetFirst( self ): if len( self._sorted_media ) > 0: return self._sorted_media[ 0 ] else: return None def _GetLast( self ): if len( self._sorted_media ) > 0: return self._sorted_media[ -1 ] else: return None def _GetMedia( self, hashes, discriminator = None ): if hashes.isdisjoint( self._hashes ): return [] medias = [] if discriminator is None or discriminator == 'singletons': medias.extend( ( self._hashes_to_singleton_media[ hash ] for hash in hashes if hash in self._hashes_to_singleton_media ) ) if discriminator is None or discriminator == 'collections': medias.extend( { self._hashes_to_collected_media[ hash ] for hash in hashes if hash in self._hashes_to_collected_media } ) return medias def _GetNext( self, media ): if media is None: return None next_index = self._sorted_media.index( media ) + 1 if next_index == len( self._sorted_media ): return self._GetFirst() else: return self._sorted_media[ next_index ] def _GetPrevious( self, media ): if media is None: return None previous_index = self._sorted_media.index( media ) - 1 if previous_index == -1: return self._GetLast() else: return self._sorted_media[ previous_index ] def _HasHashes( self, hashes ): for hash in hashes: if hash in self._hashes: return True return False def _RecalcAfterContentUpdates( self, service_keys_to_content_updates ): pass def _RecalcAfterMediaRemove( self ): self._RecalcHashes() def _RecalcHashes( self ): self._hashes = set() self._hashes_ordered = [] self._hashes_to_singleton_media = {} self._hashes_to_collected_media = {} for m in self._sorted_media: if isinstance( m, MediaCollection ): hashes = m.GetHashes( ordered = True ) self._hashes.update( hashes ) self._hashes_ordered.extend( hashes ) for hash in hashes: self._hashes_to_collected_media[ hash ] = m else: hash = m.GetHash() self._hashes.add( hash ) self._hashes_ordered.append( hash ) self._hashes_to_singleton_media[ hash ] = m def _RemoveMediaByHashes( self, hashes ): if not isinstance( hashes, set ): hashes = set( hashes ) affected_singleton_media = self._GetMedia( hashes, discriminator = 'singletons' ) for media in self._collected_media: media._RemoveMediaByHashes( hashes ) affected_collected_media = [ media for media in self._collected_media if media.HasNoMedia() ] self._RemoveMediaDirectly( affected_singleton_media, affected_collected_media ) def _RemoveMediaDirectly( self, singleton_media, collected_media ): if not isinstance( singleton_media, set ): singleton_media = set( singleton_media ) if not isinstance( collected_media, set ): collected_media = set( collected_media ) self._singleton_media.difference_update( singleton_media ) self._collected_media.difference_update( collected_media ) self._sorted_media.remove_items( singleton_media.union( collected_media ) ) self._RecalcAfterMediaRemove() def AddMedia( self, new_media ): new_media = FlattenMedia( new_media ) addable_media = [] for media in new_media: hash = media.GetHash() if hash in self._hashes: continue addable_media.append( media ) self._hashes.add( hash ) self._hashes_ordered.append( hash ) self._hashes_to_singleton_media[ hash ] = media self._singleton_media.update( addable_media ) self._sorted_media.append_items( addable_media ) return new_media def Collect( self, media_collect = None ): if media_collect == None: media_collect = self._media_collect self._media_collect = media_collect flat_media = list( self._singleton_media ) for media in self._collected_media: flat_media.extend( [ self._GenerateMediaSingleton( media_result ) for media_result in media.GenerateMediaResults() ] ) if self._media_collect.DoesACollect(): keys_to_medias = self._CalculateCollectionKeysToMedias( media_collect, flat_media ) # add an option here I think, to media_collect to say if collections with one item should be singletons or not self._singleton_media = set()#{ medias[0] for ( key, medias ) in keys_to_medias.items() if len( medias ) == 1 } if not self._media_collect.collect_unmatched: unmatched_key = ( frozenset(), frozenset() ) if unmatched_key in keys_to_medias: unmatched_medias = keys_to_medias[ unmatched_key ] self._singleton_media.update( unmatched_medias ) del keys_to_medias[ unmatched_key ] self._collected_media = { self._GenerateMediaCollection( [ media.GetMediaResult() for media in medias ] ) for ( key, medias ) in keys_to_medias.items() }# if len( medias ) > 1 } else: self._singleton_media = set( flat_media ) self._collected_media = set() self._sorted_media = SortedList( list( self._singleton_media ) + list( self._collected_media ) ) self._RecalcHashes() def DeletePending( self, service_key ): for media in self._collected_media: media.DeletePending( service_key ) def GetFilteredFileCount( self, file_filter ): if file_filter.filter_type == FILE_FILTER_ALL: return self.GetNumFiles() elif file_filter.filter_type == FILE_FILTER_SELECTED: return sum( ( m.GetNumFiles() for m in self._selected_media ) ) elif file_filter.filter_type == FILE_FILTER_NOT_SELECTED: return self.GetNumFiles() - sum( ( m.GetNumFiles() for m in self._selected_media ) ) elif file_filter.filter_type == FILE_FILTER_NONE: return 0 elif file_filter.filter_type == FILE_FILTER_INBOX: return sum( ( m.GetNumInbox() for m in self._selected_media ) ) elif file_filter.filter_type == FILE_FILTER_ARCHIVE: return self.GetNumFiles() - sum( ( m.GetNumInbox() for m in self._selected_media ) ) else: flat_media = self.GetFlatMedia() if file_filter.filter_type == FILE_FILTER_FILE_SERVICE: file_service_key = file_filter.filter_data return sum( ( 1 for m in flat_media if file_service_key in m.GetLocationsManager().GetCurrent() ) ) elif file_filter.filter_type == FILE_FILTER_LOCAL: return sum( ( 1 for m in flat_media if m.GetLocationsManager().IsLocal() ) ) elif file_filter.filter_type == FILE_FILTER_REMOTE: return sum( ( 1 for m in flat_media if m.GetLocationsManager().IsRemote() ) ) elif file_filter.filter_type == FILE_FILTER_TAGS: ( tag_service_key, and_or_or, select_tags ) = file_filter.filter_data if and_or_or == 'AND': select_tags = set( select_tags ) return sum( ( 1 for m in flat_media if select_tags.issubset( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ) ) ) ) elif and_or_or == 'OR': return sum( ( 1 for m in flat_media if HydrusData.SetsIntersect( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ), select_tags ) ) ) return 0 def GetFilteredHashes( self, file_filter ): if file_filter.filter_type == FILE_FILTER_ALL: return self._hashes elif file_filter.filter_type == FILE_FILTER_SELECTED: hashes = set() for m in self._selected_media: hashes.update( m.GetHashes() ) return hashes elif file_filter.filter_type == FILE_FILTER_NOT_SELECTED: hashes = set() for m in self._sorted_media: if m not in self._selected_media: hashes.update( m.GetHashes() ) return hashes elif file_filter.filter_type == FILE_FILTER_NONE: return set() else: flat_media = self.GetFlatMedia() if file_filter.filter_type == FILE_FILTER_INBOX: filtered_media = [ m for m in flat_media if m.HasInbox() ] elif file_filter.filter_type == FILE_FILTER_ARCHIVE: filtered_media = [ m for m in flat_media if not m.HasInbox() ] elif file_filter.filter_type == FILE_FILTER_FILE_SERVICE: file_service_key = file_filter.filter_data filtered_media = [ m for m in flat_media if file_service_key in m.GetLocationsManager().GetCurrent() ] elif file_filter.filter_type == FILE_FILTER_LOCAL: filtered_media = [ m for m in flat_media if m.GetLocationsManager().IsLocal() ] elif file_filter.filter_type == FILE_FILTER_REMOTE: filtered_media = [ m for m in flat_media if m.GetLocationsManager().IsRemote() ] elif file_filter.filter_type == FILE_FILTER_TAGS: ( tag_service_key, and_or_or, select_tags ) = file_filter.filter_data if and_or_or == 'AND': select_tags = set( select_tags ) filtered_media = [ m for m in flat_media if select_tags.issubset( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ) ) ] elif and_or_or == 'OR': filtered_media = [ m for m in flat_media if HydrusData.SetsIntersect( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ), select_tags ) ] hashes = { m.GetHash() for m in filtered_media } return hashes return set() def GetFilteredMedia( self, file_filter ): if file_filter.filter_type == FILE_FILTER_ALL: return set( self._sorted_media ) elif file_filter.filter_type == FILE_FILTER_SELECTED: return self._selected_media elif file_filter.filter_type == FILE_FILTER_NOT_SELECTED: return { m for m in self._sorted_media if m not in self._selected_media } elif file_filter.filter_type == FILE_FILTER_NONE: return set() else: if file_filter.filter_type == FILE_FILTER_INBOX: filtered_media = { m for m in self._sorted_media if m.HasInbox() } elif file_filter.filter_type == FILE_FILTER_ARCHIVE: filtered_media = { m for m in self._sorted_media if not m.HasInbox() } elif file_filter.filter_type == FILE_FILTER_FILE_SERVICE: file_service_key = file_filter.filter_data filtered_media = { m for m in self._sorted_media if file_service_key in m.GetLocationsManager().GetCurrent() } elif file_filter.filter_type == FILE_FILTER_LOCAL: filtered_media = { m for m in self._sorted_media if m.GetLocationsManager().IsLocal() } elif file_filter.filter_type == FILE_FILTER_REMOTE: filtered_media = { m for m in self._sorted_media if m.GetLocationsManager().IsRemote() } elif file_filter.filter_type == FILE_FILTER_TAGS: ( tag_service_key, and_or_or, select_tags ) = file_filter.filter_data if and_or_or == 'AND': select_tags = set( select_tags ) filtered_media = { m for m in self._sorted_media if select_tags.issubset( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ) ) } elif and_or_or == 'OR': filtered_media = { m for m in self._sorted_media if HydrusData.SetsIntersect( m.GetTagsManager().GetCurrentAndPending( tag_service_key, ClientTags.TAG_DISPLAY_ACTUAL ), select_tags ) } return filtered_media return set() def GenerateMediaResults( self, has_location = None, discriminant = None, selected_media = None, unrated = None, for_media_viewer = False ): media_results = [] for media in self._sorted_media: if has_location is not None: locations_manager = media.GetLocationsManager() if has_location not in locations_manager.GetCurrent(): continue if selected_media is not None and media not in selected_media: continue if media.IsCollection(): # don't include selected_media here as it is not valid at the deeper collection level media_results.extend( media.GenerateMediaResults( has_location = has_location, discriminant = discriminant, unrated = unrated, for_media_viewer = True ) ) else: if discriminant is not None: locations_manager = media.GetLocationsManager() if discriminant == CC.DISCRIMINANT_INBOX: p = media.HasInbox() elif discriminant == CC.DISCRIMINANT_ARCHIVE: p = not media.HasInbox() elif discriminant == CC.DISCRIMINANT_LOCAL: p = locations_manager.IsLocal() elif discriminant == CC.DISCRIMINANT_LOCAL_BUT_NOT_IN_TRASH: p = locations_manager.IsLocal() and not locations_manager.IsTrashed() elif discriminant == CC.DISCRIMINANT_NOT_LOCAL: p = not locations_manager.IsLocal() elif discriminant == CC.DISCRIMINANT_DOWNLOADING: p = locations_manager.IsDownloading() if not p: continue if unrated is not None: ratings_manager = media.GetRatingsManager() if ratings_manager.GetRating( unrated ) is not None: continue if for_media_viewer: new_options = HG.client_controller.new_options ( media_show_action, media_start_paused, media_start_with_embed ) = new_options.GetMediaShowAction( media.GetMime() ) if media_show_action in ( CC.MEDIA_VIEWER_ACTION_DO_NOT_SHOW_ON_ACTIVATION_OPEN_EXTERNALLY, CC.MEDIA_VIEWER_ACTION_DO_NOT_SHOW ): continue media_results.append( media.GetMediaResult() ) return media_results def GetAPIInfoDict( self, simple ): d = {} d[ 'num_files' ] = self.GetNumFiles() flat_media = self.GetFlatMedia() d[ 'hash_ids' ] = [ m.GetMediaResult().GetHashId() for m in flat_media ] if not simple: hashes = self.GetHashes( ordered = True ) d[ 'hashes' ] = [ hash.hex() for hash in hashes ] return d def GetFirst( self ): return self._GetFirst() def GetFlatMedia( self ): flat_media = [] for media in self._sorted_media: if media.IsCollection(): flat_media.extend( media.GetFlatMedia() ) else: flat_media.append( media ) return flat_media def GetHashes( self, has_location = None, discriminant = None, not_uploaded_to = None, ordered = False ): if has_location is None and discriminant is None and not_uploaded_to is None: if ordered: return self._hashes_ordered else: return self._hashes else: if ordered: result = [] for media in self._sorted_media: result.extend( media.GetHashes( has_location, discriminant, not_uploaded_to, ordered ) ) else: result = set() for media in self._sorted_media: result.update( media.GetHashes( has_location, discriminant, not_uploaded_to, ordered ) ) return result def GetLast( self ): return self._GetLast() def GetMediaIndex( self, media ): return self._sorted_media.index( media ) def GetNext( self, media ): return self._GetNext( media ) def GetNumArchive( self ): num_archive = sum( ( 1 for m in self._singleton_media if not m.HasInbox() ) ) + sum( ( m.GetNumArchive() for m in self._collected_media ) ) return num_archive def GetNumFiles( self ): return len( self._hashes ) def GetNumInbox( self ): num_inbox = sum( ( 1 for m in self._singleton_media if m.HasInbox() ) ) + sum( ( m.GetNumInbox() for m in self._collected_media ) ) return num_inbox def GetPrevious( self, media ): return self._GetPrevious( media ) def GetSortedMedia( self ): return self._sorted_media def HasAnyOfTheseHashes( self, hashes: set ): return not hashes.isdisjoint( self._hashes ) def HasMedia( self, media ): if media is None: return False if media in self._singleton_media: return True elif media in self._collected_media: return True else: for media_collection in self._collected_media: if media_collection.HasMedia( media ): return True return False def HasNoMedia( self ): return len( self._sorted_media ) == 0 def ProcessContentUpdates( self, full_service_keys_to_content_updates ): service_keys_to_content_updates = FilterServiceKeysToContentUpdates( full_service_keys_to_content_updates, self._hashes ) if len( service_keys_to_content_updates ) == 0: return for m in self._collected_media: m.ProcessContentUpdates( service_keys_to_content_updates ) for ( service_key, content_updates ) in service_keys_to_content_updates.items(): for content_update in content_updates: ( data_type, action, row ) = content_update.ToTuple() hashes = content_update.GetHashes() if data_type == HC.CONTENT_TYPE_FILES: if action == HC.CONTENT_UPDATE_DELETE: local_file_domains = HG.client_controller.services_manager.GetServiceKeys( ( HC.LOCAL_FILE_DOMAIN, ) ) all_local_file_services = set( list( local_file_domains ) + [ CC.COMBINED_LOCAL_FILE_SERVICE_KEY, CC.TRASH_SERVICE_KEY ] ) # physically_deleted = service_key == CC.COMBINED_LOCAL_FILE_SERVICE_KEY trashed = service_key in local_file_domains deleted_from_our_domain = self._location_context.IsOneDomain() and service_key in self._location_context.current_service_keys our_view_is_all_local = self._location_context.IncludesCurrent() and not self._location_context.IncludesDeleted() and self._location_context.current_service_keys.issubset( all_local_file_services ) physically_deleted_and_local_view = physically_deleted and our_view_is_all_local user_says_remove_and_trashed_from_non_trash_local_view = HC.options[ 'remove_trashed_files' ] and trashed and our_view_is_all_local and CC.TRASH_SERVICE_KEY not in self._location_context.current_service_keys deleted_from_repo_and_repo_view = service_key not in all_local_file_services and deleted_from_our_domain if physically_deleted_and_local_view or user_says_remove_and_trashed_from_non_trash_local_view or deleted_from_repo_and_repo_view: self._RemoveMediaByHashes( hashes ) self._RecalcAfterContentUpdates( service_keys_to_content_updates ) def ProcessServiceUpdates( self, service_keys_to_service_updates ): for ( service_key, service_updates ) in service_keys_to_service_updates.items(): for service_update in service_updates: ( action, row ) = service_update.ToTuple() if action == HC.SERVICE_UPDATE_DELETE_PENDING: self.DeletePending( service_key ) elif action == HC.SERVICE_UPDATE_RESET: self.ResetService( service_key ) def ResetService( self, service_key ): if self._location_context.IsOneDomain() and service_key in self._location_context.current_service_keys: self._RemoveMediaDirectly( self._singleton_media, self._collected_media ) else: for media in self._collected_media: media.ResetService( service_key ) def Sort( self, media_sort = None ): for media in self._collected_media: media.Sort( media_sort ) if media_sort is None: media_sort = self._media_sort self._media_sort = media_sort media_sort_fallback = HG.client_controller.new_options.GetFallbackSort() media_sort_fallback.Sort( self._location_context, self._sorted_media ) # this is a stable sort, so the fallback order above will remain for equal items self._media_sort.Sort( self._location_context, self._sorted_media ) self._RecalcHashes() FILE_FILTER_ALL = 0 FILE_FILTER_NOT_SELECTED = 1 FILE_FILTER_NONE = 2 FILE_FILTER_INBOX = 3 FILE_FILTER_ARCHIVE = 4 FILE_FILTER_FILE_SERVICE = 5 FILE_FILTER_LOCAL = 6 FILE_FILTER_REMOTE = 7 FILE_FILTER_TAGS = 8 FILE_FILTER_SELECTED = 9 FILE_FILTER_MIME = 10 file_filter_str_lookup = {} file_filter_str_lookup[ FILE_FILTER_ALL ] = 'all' file_filter_str_lookup[ FILE_FILTER_NOT_SELECTED ] = 'not selected' file_filter_str_lookup[ FILE_FILTER_SELECTED ] = 'selected' file_filter_str_lookup[ FILE_FILTER_NONE ] = 'none' file_filter_str_lookup[ FILE_FILTER_INBOX ] = 'inbox' file_filter_str_lookup[ FILE_FILTER_ARCHIVE ] = 'archive' file_filter_str_lookup[ FILE_FILTER_FILE_SERVICE ] = 'file service' file_filter_str_lookup[ FILE_FILTER_LOCAL ] = 'local' file_filter_str_lookup[ FILE_FILTER_REMOTE ] = 'not local' file_filter_str_lookup[ FILE_FILTER_TAGS ] = 'tags' file_filter_str_lookup[ FILE_FILTER_MIME ] = 'filetype' class FileFilter( object ): def __init__( self, filter_type, filter_data = None ): self.filter_type = filter_type self.filter_data = filter_data def __eq__( self, other ): if isinstance( other, FileFilter ): return self.__hash__() == other.__hash__() return NotImplemented def __hash__( self ): if self.filter_data is None: return self.filter_type.__hash__() else: return ( self.filter_type, self.filter_data ).__hash__() def PopulateFilterCounts( self, media_list: MediaList, filter_counts: dict ): if self not in filter_counts: if self.filter_type == FILE_FILTER_NONE: filter_counts[ self ] = 0 return quick_inverse_lookups= {} quick_inverse_lookups[ FileFilter( FILE_FILTER_INBOX ) ] = FileFilter( FILE_FILTER_ARCHIVE ) quick_inverse_lookups[ FileFilter( FILE_FILTER_ARCHIVE ) ] = FileFilter( FILE_FILTER_INBOX ) quick_inverse_lookups[ FileFilter( FILE_FILTER_SELECTED ) ] = FileFilter( FILE_FILTER_NOT_SELECTED ) quick_inverse_lookups[ FileFilter( FILE_FILTER_NOT_SELECTED ) ] = FileFilter( FILE_FILTER_SELECTED ) quick_inverse_lookups[ FileFilter( FILE_FILTER_LOCAL ) ] = FileFilter( FILE_FILTER_REMOTE ) quick_inverse_lookups[ FileFilter( FILE_FILTER_REMOTE ) ] = FileFilter( FILE_FILTER_LOCAL ) if self in quick_inverse_lookups: inverse = quick_inverse_lookups[ self ] all_filter = FileFilter( FILE_FILTER_ALL ) if all_filter in filter_counts and inverse in filter_counts: filter_counts[ self ] = filter_counts[ all_filter ] - filter_counts[ inverse ] return count = media_list.GetFilteredFileCount( self ) filter_counts[ self ] = count def GetCount( self, media_list: MediaList, filter_counts: dict ): self.PopulateFilterCounts( media_list, filter_counts ) return filter_counts[ self ] def ToString( self, media_list: MediaList, filter_counts: dict ): if self.filter_type == FILE_FILTER_FILE_SERVICE: file_service_key = self.filter_data s = HG.client_controller.services_manager.GetName( file_service_key ) elif self.filter_type == FILE_FILTER_TAGS: ( tag_service_key, and_or_or, select_tags ) = self.filter_data s = and_or_or.join( select_tags ) if tag_service_key != CC.COMBINED_TAG_SERVICE_KEY: s = '{} on {}'.format( s, HG.client_controller.services_manager.GetName( tag_service_key ) ) s = HydrusText.ElideText( s, 64 ) elif self.filter_type == FILE_FILTER_MIME: mime = self.filter_data s = HC.mime_string_lookup[ mime ] else: s = file_filter_str_lookup[ self.filter_type ] self.PopulateFilterCounts( media_list, filter_counts ) my_count = filter_counts[ self ] s += ' ({})'.format( HydrusData.ToHumanInt( my_count ) ) if self.filter_type == FILE_FILTER_ALL: inbox_filter = FileFilter( FILE_FILTER_INBOX ) archive_filter = FileFilter( FILE_FILTER_ARCHIVE ) inbox_filter.PopulateFilterCounts( media_list, filter_counts ) archive_filter.PopulateFilterCounts( media_list, filter_counts ) inbox_count = filter_counts[ inbox_filter ] if inbox_count > 0 and inbox_count == my_count: s += ' (all in inbox)' else: archive_count = filter_counts[ archive_filter ] if archive_count > 0 and archive_count == my_count: s += ' (all in archive)' return s class ListeningMediaList( MediaList ): def __init__( self, location_context: ClientLocation.LocationContext, media_results ): MediaList.__init__( self, location_context, media_results ) HG.client_controller.sub( self, 'ProcessContentUpdates', 'content_updates_gui' ) HG.client_controller.sub( self, 'ProcessServiceUpdates', 'service_updates_gui' ) def AddMediaResults( self, media_results ): new_media = [] for media_result in media_results: hash = media_result.GetHash() if hash in self._hashes: continue new_media.append( self._GenerateMediaSingleton( media_result ) ) self.AddMedia( new_media ) return new_media class MediaCollection( MediaList, Media ): def __init__( self, location_context: ClientLocation.LocationContext, media_results ): # note for later: ideal here is to stop this multiple inheritance mess and instead have this be a media that *has* a list, not *is* a list Media.__init__( self ) MediaList.__init__( self, location_context, media_results ) self._archive = True self._inbox = False self._size = 0 self._size_definite = True self._width = None self._height = None self._duration = None self._num_frames = None self._num_words = None self._has_audio = None self._tags_manager = None self._locations_manager = None self._file_viewing_stats_manager = None self._internals_dirty = False self._RecalcInternals() def _RecalcAfterContentUpdates( self, service_keys_to_content_updates ): archive_or_inbox = False data_types = set() for ( service_key, content_updates ) in service_keys_to_content_updates.items(): for content_update in content_updates: data_type = content_update.GetDataType() if data_type in ( HC.CONTENT_TYPE_URLS, HC.CONTENT_TYPE_NOTES ): continue elif data_type == HC.CONTENT_TYPE_FILES: action = content_update.GetAction() if action in ( HC.CONTENT_UPDATE_ARCHIVE, HC.CONTENT_UPDATE_INBOX ): archive_or_inbox = True continue data_types.add( data_type ) if archive_or_inbox and data_types.issubset( { HC.CONTENT_TYPE_RATINGS, HC.CONTENT_TYPE_FILE_VIEWING_STATS, HC.CONTENT_TYPE_MAPPINGS }): if archive_or_inbox: self._RecalcArchiveInbox() for data_type in data_types: if data_type == HC.CONTENT_TYPE_RATINGS: self._RecalcRatings() elif data_type == HC.CONTENT_TYPE_FILE_VIEWING_STATS: self._RecalcFileViewingStats() elif data_type == HC.CONTENT_TYPE_MAPPINGS: self._RecalcTags() elif len( data_types ) > 0: self._RecalcInternals() def _RecalcAfterMediaRemove( self ): MediaList._RecalcAfterMediaRemove( self ) self._RecalcArchiveInbox() def _RecalcArchiveInbox( self ): self._archive = True in ( media.HasArchive() for media in self._sorted_media ) self._inbox = True in ( media.HasInbox() for media in self._sorted_media ) def _RecalcFileViewingStats( self ): preview_views = 0 preview_viewtime = 0.0 media_views = 0 media_viewtime = 0.0 for m in self._sorted_media: fvsm = m.GetFileViewingStatsManager() preview_views += fvsm.preview_views preview_viewtime += fvsm.preview_viewtime media_views += fvsm.media_views media_viewtime += fvsm.media_viewtime self._file_viewing_stats_manager = ClientMediaManagers.FileViewingStatsManager( preview_views, preview_viewtime, media_views, media_viewtime ) def _RecalcHashes( self ): MediaList._RecalcHashes( self ) all_locations_managers = [ media.GetLocationsManager() for media in self._sorted_media ] current_to_timestamps = {} deleted_to_timestamps = {} for service_key in HG.client_controller.services_manager.GetServiceKeys( HC.FILE_SERVICES ): current_timestamps = [ timestamp for timestamp in ( locations_manager.GetCurrentTimestamp( service_key ) for locations_manager in all_locations_managers ) if timestamp is not None ] if len( current_timestamps ) > 0: current_to_timestamps[ service_key ] = max( current_timestamps ) deleted_timestamps = [ timestamps for timestamps in ( locations_manager.GetDeletedTimestamps( service_key ) for locations_manager in all_locations_managers ) if timestamps is not None and timestamps[0] is not None ] if len( deleted_timestamps ) > 0: deleted_to_timestamps[ service_key ] = max( deleted_timestamps, key = lambda ts: ts[0] ) pending = HydrusData.MassUnion( [ locations_manager.GetPending() for locations_manager in all_locations_managers ] ) petitioned = HydrusData.MassUnion( [ locations_manager.GetPetitioned() for locations_manager in all_locations_managers ] ) self._locations_manager = ClientMediaManagers.LocationsManager( current_to_timestamps, deleted_to_timestamps, pending, petitioned ) def _RecalcInternals( self ): self._RecalcHashes() self._RecalcTags() self._RecalcArchiveInbox() self._size = sum( [ media.GetSize() for media in self._sorted_media ] ) self._size_definite = not False in ( media.IsSizeDefinite() for media in self._sorted_media ) duration_sum = sum( [ media.GetDuration() for media in self._sorted_media if media.HasDuration() ] ) if duration_sum > 0: self._duration = duration_sum else: self._duration = None self._has_audio = True in ( media.HasAudio() for media in self._sorted_media ) self._has_notes = True in ( media.HasNotes() for media in self._sorted_media ) self._RecalcRatings() self._RecalcFileViewingStats() def _RecalcRatings( self ): # horrible compromise if len( self._sorted_media ) > 0: self._ratings_manager = self._sorted_media[0].GetRatingsManager() else: self._ratings_manager = ClientMediaManagers.RatingsManager( {} ) def _RecalcTags( self ): tags_managers = [ m.GetTagsManager() for m in self._sorted_media ] self._tags_manager = ClientMediaManagers.TagsManager.MergeTagsManagers( tags_managers ) def AddMedia( self, new_media ): MediaList.AddMedia( self, new_media ) self._RecalcInternals() def DeletePending( self, service_key ): MediaList.DeletePending( self, service_key ) self._RecalcInternals() def GetCurrentTimestamp( self, service_key: bytes ) -> typing.Optional[ int ]: return self._locations_manager.GetCurrentTimestamp( service_key ) def GetDeletedTimestamps( self, service_key: bytes ) -> typing.Tuple[ typing.Optional[ int ], typing.Optional[ int ] ]: return self._locations_manager.GetDeletedTimestamps( service_key ) def GetDisplayMedia( self ): first = self._GetFirst() if first is None: return None else: return first.GetDisplayMedia() def GetDuration( self ): return self._duration def GetFileViewingStatsManager( self ): return self._file_viewing_stats_manager def GetHash( self ): display_media = self.GetDisplayMedia() if display_media is None: return None else: return display_media.GetHash() def GetLocationsManager( self ): return self._locations_manager def GetMime( self ): return HC.APPLICATION_HYDRUS_CLIENT_COLLECTION def GetNumInbox( self ): return sum( ( media.GetNumInbox() for media in self._sorted_media ) ) def GetNumFrames( self ): num_frames = ( media.GetNumFrames() for media in self._sorted_media ) return sum( ( nf for nf in num_frames if nf is not None ) ) def GetNumWords( self ): num_words = ( media.GetNumWords() for media in self._sorted_media ) return sum( ( nw for nw in num_words if nw is not None ) ) def GetPrettyInfoLines( self ): size = HydrusData.ToHumanBytes( self._size ) mime = HC.mime_string_lookup[ HC.APPLICATION_HYDRUS_CLIENT_COLLECTION ] info_string = size + ' ' + mime info_string += ' (' + HydrusData.ToHumanInt( self.GetNumFiles() ) + ' files)' return [ info_string ] def GetRatingsManager( self ): return self._ratings_manager def GetResolution( self ): if self._width is None: return ( 0, 0 ) else: return ( self._width, self._height ) def GetSingletonsTagsManagers( self ): tags_managers = [ m.GetTagsManager() for m in self._singleton_media ] for m in self._collected_media: tags_managers.extend( m.GetSingletonsTagsManagers() ) return tags_managers def GetSize( self ): return self._size def GetTagsManager( self ): return self._tags_manager def HasArchive( self ): return self._archive def HasAudio( self ): return self._has_audio def HasDuration( self ): return self._duration is not None def HasImages( self ): return True in ( media.HasImages() for media in self._sorted_media ) def HasInbox( self ): return self._inbox def HasNotes( self ): return self._has_notes def IsCollection( self ): return True def IsImage( self ): return False def IsSizeDefinite( self ): return self._size_definite def RecalcInternals( self ): self._RecalcInternals() def ResetService( self, service_key ): MediaList.ResetService( self, service_key ) self._RecalcInternals() def UpdateFileInfo( self, hashes_to_media_results ): for media in self._sorted_media: media.UpdateFileInfo( hashes_to_media_results ) self._RecalcInternals() class MediaSingleton( Media ): def __init__( self, media_result: ClientMediaResult.MediaResult ): Media.__init__( self ) self._media_result = media_result def Duplicate( self ): return MediaSingleton( self._media_result.Duplicate() ) def GetDisplayMedia( self ) -> 'MediaSingleton': return self def GetDuration( self ): return self._media_result.GetDuration() def GetFileViewingStatsManager( self ): return self._media_result.GetFileViewingStatsManager() def GetHash( self ): return self._media_result.GetHash() def GetHashId( self ): return self._media_result.GetHashId() def GetHashes( self, has_location = None, discriminant = None, not_uploaded_to = None, ordered = False ): if self.MatchesDiscriminant( has_location = has_location, discriminant = discriminant, not_uploaded_to = not_uploaded_to ): if ordered: return [ self._media_result.GetHash() ] else: return { self._media_result.GetHash() } else: if ordered: return [] else: return set() def GetLocationsManager( self ): return self._media_result.GetLocationsManager() def GetMediaResult( self ): return self._media_result def GetMime( self ): return self._media_result.GetMime() def GetNotesManager( self ): return self._media_result.GetNotesManager() def GetNumFiles( self ): return 1 def GetNumFrames( self ): return self._media_result.GetNumFrames() def GetNumInbox( self ): if self.HasInbox(): return 1 else: return 0 def GetNumWords( self ): return self._media_result.GetNumWords() def GetCurrentTimestamp( self, service_key ) -> typing.Optional[ int ]: return self._media_result.GetLocationsManager().GetCurrentTimestamp( service_key ) def GetDeletedTimestamps( self, service_key: bytes ) -> typing.Tuple[ typing.Optional[ int ], typing.Optional[ int ] ]: return self._media_result.GetLocationsManager().GetDeletedTimestamps( service_key ) def GetPrettyInfoLines( self ): file_info_manager = self._media_result.GetFileInfoManager() locations_manager = self._media_result.GetLocationsManager() ( hash_id, hash, size, mime, width, height, duration, num_frames, has_audio, num_words ) = file_info_manager.ToTuple() info_string = HydrusData.ToHumanBytes( size ) + ' ' + HC.mime_string_lookup[ mime ] if width is not None and height is not None: info_string += ' ({})'.format( HydrusData.ConvertResolutionToPrettyString( ( width, height ) ) ) if duration is not None: info_string += ', ' + HydrusData.ConvertMillisecondsToPrettyTime( duration ) if num_frames is not None: if duration is None or duration == 0 or num_frames == 0: framerate_insert = '' else: framerate_insert = ', {}fps'.format( round( num_frames / ( duration / 1000 ) ) ) info_string += ' ({} frames{})'.format( HydrusData.ToHumanInt( num_frames ), framerate_insert ) if has_audio: info_string += ', {}'.format( HG.client_controller.new_options.GetString( 'has_audio_label' ) ) if num_words is not None: info_string += ' (' + HydrusData.ToHumanInt( num_words ) + ' words)' lines = [ info_string ] locations_manager = self._media_result.GetLocationsManager() current_service_keys = locations_manager.GetCurrent() deleted_service_keys = locations_manager.GetDeleted() local_file_services = HG.client_controller.services_manager.GetLocalMediaFileServices() current_local_file_services = [ service for service in local_file_services if service.GetServiceKey() in current_service_keys ] if len( current_local_file_services ) > 0: for local_file_service in current_local_file_services: timestamp = locations_manager.GetCurrentTimestamp( local_file_service.GetServiceKey() ) lines.append( 'added to {} {}'.format( local_file_service.GetName(), ClientData.TimestampToPrettyTimeDelta( timestamp ) ) ) elif CC.COMBINED_LOCAL_FILE_SERVICE_KEY in current_service_keys: timestamp = locations_manager.GetCurrentTimestamp( CC.COMBINED_LOCAL_FILE_SERVICE_KEY ) lines.append( 'imported {}'.format( ClientData.TimestampToPrettyTimeDelta( timestamp ) ) ) deleted_local_file_services = [ service for service in local_file_services if service.GetServiceKey() in deleted_service_keys ] if CC.COMBINED_LOCAL_FILE_SERVICE_KEY in deleted_service_keys: ( timestamp, original_timestamp ) = locations_manager.GetDeletedTimestamps( CC.COMBINED_LOCAL_FILE_SERVICE_KEY ) lines.append( 'deleted from this client {}'.format( ClientData.TimestampToPrettyTimeDelta( timestamp ) ) ) elif len( deleted_local_file_services ) > 0: for local_file_service in deleted_local_file_services: ( timestamp, original_timestamp ) = locations_manager.GetDeletedTimestamps( local_file_service.GetServiceKey() ) lines.append( 'removed from {} {}'.format( local_file_service.GetName(), ClientData.TimestampToPrettyTimeDelta( timestamp ) ) ) if locations_manager.IsTrashed(): lines.append( 'in the trash' ) file_modified_timestamp = locations_manager.GetFileModifiedTimestamp() if file_modified_timestamp is not None: lines.append( 'file modified: {}'.format( ClientData.TimestampToPrettyTimeDelta( file_modified_timestamp ) ) ) for service_key in current_service_keys.intersection( HG.client_controller.services_manager.GetServiceKeys( HC.REMOTE_FILE_SERVICES ) ): timestamp = locations_manager.GetCurrentTimestamp( service_key ) try: service = HG.client_controller.services_manager.GetService( service_key ) except HydrusExceptions.DataMissing: continue service_type = service.GetServiceType() if service_type == HC.IPFS: status = 'pinned ' else: status = 'uploaded ' lines.append( status + 'to ' + service.GetName() + ' ' + ClientData.TimestampToPrettyTimeDelta( timestamp ) ) return lines def GetRatingsManager( self ): return self._media_result.GetRatingsManager() def GetResolution( self ): ( width, height ) = self._media_result.GetResolution() if width is None: return ( 0, 0 ) else: return ( width, height ) def GetSize( self ): size = self._media_result.GetSize() if size is None: return 0 else: return size def GetTagsManager( self ): return self._media_result.GetTagsManager() def GetTitleString( self ): new_options = HG.client_controller.new_options tag_summary_generator = new_options.GetTagSummaryGenerator( 'media_viewer_top' ) tags = self.GetTagsManager().GetCurrentAndPending( CC.COMBINED_TAG_SERVICE_KEY, ClientTags.TAG_DISPLAY_SINGLE_MEDIA ) if len( tags ) == 0: return '' summary = tag_summary_generator.GenerateSummary( tags ) return summary def HasAnyOfTheseHashes( self, hashes ): return self._media_result.GetHash() in hashes def HasArchive( self ): return not self._media_result.GetInbox() def HasAudio( self ): return self._media_result.HasAudio() def HasDuration( self ): duration = self._media_result.GetDuration() return duration is not None and duration > 0 def HasImages( self ): return self.IsImage() def HasInbox( self ): return self._media_result.GetInbox() def HasNotes( self ): return self._media_result.HasNotes() def IsCollection( self ): return False def IsImage( self ): return self._media_result.GetMime() in HC.IMAGES def IsSizeDefinite( self ): return self._media_result.GetSize() is not None def IsStaticImage( self ): return self._media_result.IsStaticImage() def MatchesDiscriminant( self, has_location = None, discriminant = None, not_uploaded_to = None ): if discriminant is not None: inbox = self._media_result.GetInbox() locations_manager = self._media_result.GetLocationsManager() if discriminant == CC.DISCRIMINANT_INBOX: p = inbox elif discriminant == CC.DISCRIMINANT_ARCHIVE: p = not inbox elif discriminant == CC.DISCRIMINANT_LOCAL: p = locations_manager.IsLocal() elif discriminant == CC.DISCRIMINANT_LOCAL_BUT_NOT_IN_TRASH: p = locations_manager.IsLocal() and not locations_manager.IsTrashed() elif discriminant == CC.DISCRIMINANT_NOT_LOCAL: p = not locations_manager.IsLocal() elif discriminant == CC.DISCRIMINANT_DOWNLOADING: p = locations_manager.IsDownloading() if not p: return False if has_location is not None: locations_manager = self._media_result.GetLocationsManager() if has_location not in locations_manager.GetCurrent(): return False if not_uploaded_to is not None: locations_manager = self._media_result.GetLocationsManager() if not_uploaded_to in locations_manager.GetCurrent(): return False return True def UpdateFileInfo( self, hashes_to_media_results ): hash = self.GetHash() if hash in hashes_to_media_results: media_result = hashes_to_media_results[ hash ] self._media_result = media_result class MediaSort( HydrusSerialisable.SerialisableBase ): SERIALISABLE_TYPE = HydrusSerialisable.SERIALISABLE_TYPE_MEDIA_SORT SERIALISABLE_NAME = 'Media Sort' SERIALISABLE_VERSION = 2 def __init__( self, sort_type = None, sort_order = None ): if sort_type is None: sort_type = ( 'system', CC.SORT_FILES_BY_FILESIZE ) if sort_order is None: sort_order = CC.SORT_ASC ( sort_metatype, sort_data ) = sort_type if sort_metatype == 'namespaces': ( namespaces, tag_display_type ) = sort_data sort_data = ( tuple( namespaces ), tag_display_type ) sort_type = ( sort_metatype, sort_data ) self.sort_type = sort_type self.sort_order = sort_order def _GetSerialisableInfo( self ): ( sort_metatype, sort_data ) = self.sort_type if sort_metatype == 'system': serialisable_sort_data = sort_data elif sort_metatype == 'namespaces': serialisable_sort_data = sort_data elif sort_metatype == 'rating': service_key = sort_data serialisable_sort_data = service_key.hex() return ( sort_metatype, serialisable_sort_data, self.sort_order ) def _InitialiseFromSerialisableInfo( self, serialisable_info ): ( sort_metatype, serialisable_sort_data, self.sort_order ) = serialisable_info if sort_metatype == 'system': sort_data = serialisable_sort_data elif sort_metatype == 'namespaces': ( namespaces, tag_display_type ) = serialisable_sort_data sort_data = ( tuple( namespaces ), tag_display_type ) elif sort_metatype == 'rating': sort_data = bytes.fromhex( serialisable_sort_data ) self.sort_type = ( sort_metatype, sort_data ) def _UpdateSerialisableInfo( self, version, old_serialisable_info ): if version == 1: ( sort_metatype, serialisable_sort_data, sort_order ) = old_serialisable_info if sort_metatype == 'namespaces': namespaces = serialisable_sort_data serialisable_sort_data = ( namespaces, ClientTags.TAG_DISPLAY_ACTUAL ) new_serialisable_info = ( sort_metatype, serialisable_sort_data, sort_order ) return ( 2, new_serialisable_info ) def CanAsc( self ): ( sort_metatype, sort_data ) = self.sort_type if sort_metatype == 'system': if sort_data in ( CC.SORT_FILES_BY_MIME, CC.SORT_FILES_BY_RANDOM ): return False return True def GetNamespaces( self ): ( sort_metadata, sort_data ) = self.sort_type if sort_metadata == 'namespaces': ( namespaces, tag_display_type ) = sort_data return list( namespaces ) else: return [] def GetSortKeyAndReverse( self, location_context: ClientLocation.LocationContext ): ( sort_metadata, sort_data ) = self.sort_type def deal_with_none( x ): if x is None: return -1 else: return x if sort_metadata == 'system': if sort_data == CC.SORT_FILES_BY_RANDOM: def sort_key( x ): return random.random() elif sort_data == CC.SORT_FILES_BY_APPROX_BITRATE: def sort_key( x ): # videos > images > pdfs # heavy vids first, heavy images first duration = x.GetDuration() num_frames = x.GetNumFrames() size = x.GetSize() resolution = x.GetResolution() if duration is None or duration == 0: if size is None or size == 0: duration_bitrate = -1 frame_bitrate = -1 else: duration_bitrate = 0 if resolution is None: frame_bitrate = 0 else: ( width, height ) = x.GetResolution() num_pixels = width * height if size is None or size == 0 or num_pixels == 0: frame_bitrate = -1 else: frame_bitrate = size / num_pixels else: if size is None or size == 0: duration_bitrate = -1 frame_bitrate = -1 else: duration_bitrate = size / duration if num_frames is None or num_frames == 0: frame_bitrate = 0 else: frame_bitrate = duration_bitrate / num_frames return ( duration_bitrate, frame_bitrate ) elif sort_data == CC.SORT_FILES_BY_FILESIZE: def sort_key( x ): return deal_with_none( x.GetSize() ) elif sort_data == CC.SORT_FILES_BY_DURATION: def sort_key( x ): return deal_with_none( x.GetDuration() ) elif sort_data == CC.SORT_FILES_BY_FRAMERATE: def sort_key( x ): num_frames = x.GetNumFrames() if num_frames is None or num_frames == 0: return -1 duration = x.GetDuration() if duration is None or duration == 0: return -1 return num_frames / duration elif sort_data == CC.SORT_FILES_BY_NUM_COLLECTION_FILES: def sort_key( x ): return ( x.GetNumFiles(), isinstance( x, MediaCollection ) ) elif sort_data == CC.SORT_FILES_BY_NUM_FRAMES: def sort_key( x ): return deal_with_none( x.GetNumFrames() ) elif sort_data == CC.SORT_FILES_BY_HAS_AUDIO: def sort_key( x ): return - deal_with_none( x.HasAudio() ) elif sort_data == CC.SORT_FILES_BY_IMPORT_TIME: def sort_key( x ): return deal_with_none( x.GetLocationsManager().GetBestCurrentTimestamp( location_context ) ) elif sort_data == CC.SORT_FILES_BY_FILE_MODIFIED_TIMESTAMP: def sort_key( x ): return deal_with_none( x.GetLocationsManager().GetFileModifiedTimestamp() ) elif sort_data == CC.SORT_FILES_BY_HEIGHT: def sort_key( x ): return deal_with_none( x.GetResolution()[1] ) elif sort_data == CC.SORT_FILES_BY_WIDTH: def sort_key( x ): return deal_with_none( x.GetResolution()[0] ) elif sort_data == CC.SORT_FILES_BY_RATIO: def sort_key( x ): ( width, height ) = x.GetResolution() if width is None or height is None or width == 0 or height == 0: return -1 else: return width / height elif sort_data == CC.SORT_FILES_BY_NUM_PIXELS: def sort_key( x ): ( width, height ) = x.GetResolution() if width is None or height is None: return -1 else: return width * height elif sort_data == CC.SORT_FILES_BY_NUM_TAGS: def sort_key( x ): tags_manager = x.GetTagsManager() return len( tags_manager.GetCurrentAndPending( CC.COMBINED_TAG_SERVICE_KEY, ClientTags.TAG_DISPLAY_ACTUAL ) ) elif sort_data == CC.SORT_FILES_BY_MIME: def sort_key( x ): return x.GetMime() elif sort_data == CC.SORT_FILES_BY_MEDIA_VIEWS: def sort_key( x ): fvsm = x.GetFileViewingStatsManager() # do not do viewtime as a secondary sort here, to allow for user secondary sort to help out return fvsm.media_views elif sort_data == CC.SORT_FILES_BY_MEDIA_VIEWTIME: def sort_key( x ): fvsm = x.GetFileViewingStatsManager() # do not do views as a secondary sort here, to allow for user secondary sort to help out return fvsm.media_viewtime elif sort_metadata == 'namespaces': ( namespaces, tag_display_type ) = sort_data def sort_key( x ): x_tags_manager = x.GetTagsManager() return [ x_tags_manager.GetComparableNamespaceSlice( ( namespace, ), tag_display_type ) for namespace in namespaces ] elif sort_metadata == 'rating': service_key = sort_data def sort_key( x ): x_ratings_manager = x.GetRatingsManager() rating = deal_with_none( x_ratings_manager.GetRating( service_key ) ) return rating reverse = self.sort_order == CC.SORT_DESC return ( sort_key, reverse ) def GetSortOrderStrings( self ): ( sort_metatype, sort_data ) = self.sort_type if sort_metatype == 'system': sort_string_lookup = {} sort_string_lookup[ CC.SORT_FILES_BY_APPROX_BITRATE ] = ( 'smallest first', 'largest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_FILESIZE ] = ( 'smallest first', 'largest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_DURATION ] = ( 'shortest first', 'longest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_FRAMERATE ] = ( 'slowest first', 'fastest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_NUM_COLLECTION_FILES ] = ( 'fewest first', 'most first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_NUM_FRAMES ] = ( 'smallest first', 'largest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_HAS_AUDIO ] = ( 'audio first', 'silent first', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_IMPORT_TIME ] = ( 'oldest first', 'newest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_FILE_MODIFIED_TIMESTAMP ] = ( 'oldest first', 'newest first', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_MIME ] = ( 'filetype', 'filetype', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_RANDOM ] = ( 'random', 'random', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_WIDTH ] = ( 'slimmest first', 'widest first', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_HEIGHT ] = ( 'shortest first', 'tallest first', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_RATIO ] = ( 'tallest first', 'widest first', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_NUM_PIXELS ] = ( 'ascending', 'descending', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_NUM_TAGS ] = ( 'ascending', 'descending', CC.SORT_ASC ) sort_string_lookup[ CC.SORT_FILES_BY_MEDIA_VIEWS ] = ( 'ascending', 'descending', CC.SORT_DESC ) sort_string_lookup[ CC.SORT_FILES_BY_MEDIA_VIEWTIME ] = ( 'ascending', 'descending', CC.SORT_DESC ) return sort_string_lookup[ sort_data ] elif sort_metatype == 'namespaces': return ( 'a-z', 'z-a', CC.SORT_ASC ) else: return ( 'ascending', 'descending', CC.SORT_DESC ) def GetSortTypeString( self ): ( sort_metatype, sort_data ) = self.sort_type sort_string = 'sort by ' if sort_metatype == 'system': sort_string += CC.sort_type_string_lookup[ sort_data ] elif sort_metatype == 'namespaces': ( namespaces, tag_display_type ) = sort_data sort_string += 'tags: ' + '-'.join( namespaces ) elif sort_metatype == 'rating': service_key = sort_data try: service = HG.client_controller.services_manager.GetService( service_key ) name = service.GetName() except HydrusExceptions.DataMissing: name = 'unknown service' sort_string += 'rating: {}'.format( name ) return sort_string def Sort( self, location_context: ClientLocation.LocationContext, media_results_list: "SortedList" ): ( sort_metadata, sort_data ) = self.sort_type if sort_data == CC.SORT_FILES_BY_RANDOM: media_results_list.random_sort() else: ( sort_key, reverse ) = self.GetSortKeyAndReverse( location_context ) media_results_list.sort( sort_key, reverse = reverse ) def ToString( self ): sort_type_string = self.GetSortTypeString() ( asc_string, desc_string, sort_gumpf ) = self.GetSortOrderStrings() sort_order_string = asc_string if self.sort_order == CC.SORT_ASC else desc_string return '{}, {}'.format( sort_type_string, sort_order_string ) HydrusSerialisable.SERIALISABLE_TYPES_TO_OBJECT_TYPES[ HydrusSerialisable.SERIALISABLE_TYPE_MEDIA_SORT ] = MediaSort class SortedList( object ): def __init__( self, initial_items = None ): if initial_items is None: initial_items = [] self._sort_key = None self._sort_reverse = False self._sorted_list = list( initial_items ) self._items_to_indices = {} self._indices_dirty = True def __contains__( self, item ): if self._indices_dirty: self._RecalcIndices() return self._items_to_indices.__contains__( item ) def __getitem__( self, value ): return self._sorted_list.__getitem__( value ) def __iter__( self ): return iter( self._sorted_list ) def __len__( self ): return len( self._sorted_list ) def _DirtyIndices( self ): self._indices_dirty = True self._items_to_indices = {} def _RecalcIndices( self ): self._items_to_indices = { item : index for ( index, item ) in enumerate( self._sorted_list ) } self._indices_dirty = False def append_items( self, items ): if self._indices_dirty is None: self._RecalcIndices() for ( i, item ) in enumerate( items, start = len( self._sorted_list ) ): self._items_to_indices[ item ] = i self._sorted_list.extend( items ) def index( self, item ): if self._indices_dirty: self._RecalcIndices() try: result = self._items_to_indices[ item ] except KeyError: raise HydrusExceptions.DataMissing() return result def insert_items( self, items ): self.append_items( items ) self.sort() def remove_items( self, items ): deletee_indices = [ self.index( item ) for item in items ] deletee_indices.sort( reverse = True ) for index in deletee_indices: del self._sorted_list[ index ] self._DirtyIndices() def random_sort( self ): def sort_key( x ): return random.random() self._sort_key = sort_key random.shuffle( self._sorted_list ) self._DirtyIndices() def sort( self, sort_key = None, reverse = False ): if sort_key is None: sort_key = self._sort_key reverse = self._sort_reverse else: self._sort_key = sort_key self._sort_reverse = reverse self._sorted_list.sort( key = sort_key, reverse = reverse ) self._DirtyIndices()