hydrus/hydrus/client/gui/ClientGUIFileSeedCache.py

953 lines
35 KiB
Python
Raw Normal View History

2017-07-27 00:47:13 +00:00
import os
2020-04-22 21:00:35 +00:00
2019-11-14 03:56:30 +00:00
from qtpy import QtCore as QC
from qtpy import QtWidgets as QW
from qtpy import QtGui as QG
2020-04-22 21:00:35 +00:00
from hydrus.core import HydrusConstants as HC
from hydrus.core import HydrusData
from hydrus.core import HydrusExceptions
from hydrus.core import HydrusGlobals as HG
from hydrus.core import HydrusPaths
from hydrus.core import HydrusText
2020-07-29 20:52:44 +00:00
2020-04-22 21:00:35 +00:00
from hydrus.client import ClientConstants as CC
from hydrus.client import ClientData
2022-01-19 21:28:59 +00:00
from hydrus.client import ClientLocation
2020-04-22 21:00:35 +00:00
from hydrus.client import ClientPaths
from hydrus.client import ClientSerialisable
from hydrus.client.gui import ClientGUIDialogsQuick
from hydrus.client.gui import ClientGUIMenus
from hydrus.client.gui import ClientGUISerialisable
from hydrus.client.gui import ClientGUIScrolledPanels
2020-04-29 21:44:12 +00:00
from hydrus.client.gui import ClientGUITopLevelWindowsPanels
2020-04-22 21:00:35 +00:00
from hydrus.client.gui import QtPorting as QP
2020-07-15 20:52:09 +00:00
from hydrus.client.gui.lists import ClientGUIListConstants as CGLC
from hydrus.client.gui.lists import ClientGUIListCtrl
2021-03-17 21:59:28 +00:00
from hydrus.client.gui.widgets import ClientGUICommon
2020-04-22 21:00:35 +00:00
from hydrus.client.importing import ClientImportFileSeeds
2021-11-24 21:59:58 +00:00
from hydrus.client.importing.options import PresentationImportOptions
from hydrus.client.metadata import ClientTagSorting
2017-07-27 00:47:13 +00:00
2022-04-06 20:40:17 +00:00
def ClearFileSeeds( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache, statuses_to_remove ):
message = 'Are you sure you want to delete all the ' + '/'.join( ( CC.status_string_lookup[ status ] for status in statuses_to_remove ) ) + ' file import items? This is useful for cleaning up and de-laggifying a very large list, but be careful you aren\'t removing something you would want to revisit or what watcher/subscription may be using for future check time calculations.'
result = ClientGUIDialogsQuick.GetYesNo( win, message )
if result == QW.QDialog.DialogCode.Accepted:
file_seed_cache.RemoveFileSeedsByStatus( statuses_to_remove )
2021-07-14 20:42:19 +00:00
def GetRetryIgnoredParam( window ):
choice_tuples = [
( 'retry all', None, 'retry all' ),
( 'retry 404s', '^404', 'retry all 404s' ),
( 'retry blacklisted', 'blacklisted!$', 'retry all blacklisted' )
]
return ClientGUIDialogsQuick.SelectFromListButtons( window, 'select what to retry', choice_tuples )
2022-04-06 20:40:17 +00:00
# TODO: I pulled this stuff out of the button to share it with the panel. TBH anything without Qt may be better as be FSC methods
def GetExportableSourcesString( file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
file_seeds = file_seed_cache.GetFileSeeds()
sources = [ file_seed.file_seed_data for file_seed in file_seeds ]
return os.linesep.join( sources )
def GetSourcesFromSourcesString( sources_string ):
sources = HydrusText.DeserialiseNewlinedTexts( sources_string )
return sources
def ExportToClipboard( file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
payload = GetExportableSourcesString( file_seed_cache )
HG.client_controller.pub( 'clipboard', 'text', payload )
def ExportToPNG( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
payload = GetExportableSourcesString( file_seed_cache )
with ClientGUITopLevelWindowsPanels.DialogNullipotent( win, 'export to png' ) as dlg:
panel = ClientGUISerialisable.PNGExportPanel( dlg, payload )
dlg.SetPanel( panel )
dlg.exec()
def ImportFromClipboard( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
try:
raw_text = HG.client_controller.GetClipboardText()
except HydrusExceptions.DataMissing as e:
QW.QMessageBox.critical( win, 'Error', str(e) )
return
sources = GetSourcesFromSourcesString( raw_text )
try:
ImportSources( file_seed_cache, sources )
except:
QW.QMessageBox.critical( win, 'Error', 'Could not import!' )
raise
def ImportFromPNG( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
with QP.FileDialog( win, 'select the png with the sources', wildcard = 'PNG (*.png)' ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
try:
payload_string = ClientSerialisable.LoadStringFromPNG( path )
sources = GetSourcesFromSourcesString( payload_string )
ImportSources( file_seed_cache, sources )
except:
QW.QMessageBox.critical( win, 'Error', 'Could not import!' )
raise
def ImportSources( file_seed_cache, sources ):
if sources[0].startswith( 'http' ):
file_seed_type = ClientImportFileSeeds.FILE_SEED_TYPE_URL
else:
file_seed_type = ClientImportFileSeeds.FILE_SEED_TYPE_HDD
file_seeds = [ ClientImportFileSeeds.FileSeed( file_seed_type, source ) for source in sources ]
file_seed_cache.AddFileSeeds( file_seeds )
def RetryErrors( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
message = 'Are you sure you want to retry all the files that encountered errors?'
result = ClientGUIDialogsQuick.GetYesNo( win, message )
if result == QW.QDialog.DialogCode.Accepted:
file_seed_cache.RetryFailed()
def RetryIgnored( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
try:
ignored_regex = GetRetryIgnoredParam( win )
except HydrusExceptions.CancelledException:
return
file_seed_cache.RetryIgnored( ignored_regex = ignored_regex )
2022-12-14 22:22:11 +00:00
def ReverseFileSeedCache( win: QW.QWidget, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
message = 'Reverse this file log? Any outstanding imports will process in the opposite order.'
result = ClientGUIDialogsQuick.GetYesNo( win, message )
if result == QW.QDialog.Accepted:
file_seed_cache.Reverse()
2022-04-06 20:40:17 +00:00
def ShowFilesInNewPage( file_seed_cache: ClientImportFileSeeds.FileSeedCache, show = 'all' ):
if show == 'all':
hashes = file_seed_cache.GetHashes()
elif show == 'new':
presentation_import_options = PresentationImportOptions.PresentationImportOptions()
presentation_import_options.SetPresentationStatus( PresentationImportOptions.PRESENTATION_STATUS_NEW_ONLY )
hashes = file_seed_cache.GetPresentedHashes( presentation_import_options )
if len( hashes ) > 0:
2022-05-25 21:30:53 +00:00
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
2022-04-06 20:40:17 +00:00
HG.client_controller.pub( 'new_page_query', location_context, initial_hashes = hashes )
def PopulateFileSeedCacheMenu( win: QW.QWidget, menu: QW.QMenu, file_seed_cache: ClientImportFileSeeds.FileSeedCache ):
num_successful = file_seed_cache.GetFileSeedCount( CC.STATUS_SUCCESSFUL_AND_NEW ) + file_seed_cache.GetFileSeedCount( CC.STATUS_SUCCESSFUL_BUT_REDUNDANT )
num_vetoed = file_seed_cache.GetFileSeedCount( CC.STATUS_VETOED )
num_deleted = file_seed_cache.GetFileSeedCount( CC.STATUS_DELETED )
num_errors = file_seed_cache.GetFileSeedCount( CC.STATUS_ERROR )
num_skipped = file_seed_cache.GetFileSeedCount( CC.STATUS_SKIPPED )
if num_errors > 0:
ClientGUIMenus.AppendMenuItem( menu, 'retry ' + HydrusData.ToHumanInt( num_errors ) + ' failures', 'Tell this log to reattempt all its error failures.', RetryErrors, win, file_seed_cache )
if num_vetoed > 0:
ClientGUIMenus.AppendMenuItem( menu, 'retry ' + HydrusData.ToHumanInt( num_vetoed ) + ' ignored', 'Tell this log to reattempt all its ignored/vetoed results.', RetryIgnored, win, file_seed_cache )
ClientGUIMenus.AppendSeparator( menu )
if num_successful > 0:
ClientGUIMenus.AppendMenuItem( menu, 'delete {} \'successful\' file import items from the queue'.format( HydrusData.ToHumanInt( num_successful ) ), 'Tell this log to clear out successful files, reducing the size of the queue.', ClearFileSeeds, win, file_seed_cache, ( CC.STATUS_SUCCESSFUL_AND_NEW, CC.STATUS_SUCCESSFUL_BUT_REDUNDANT, CC.STATUS_SUCCESSFUL_AND_CHILD_FILES ) )
if num_deleted > 0:
ClientGUIMenus.AppendMenuItem( menu, 'delete {} \'previously deleted\' file import items from the queue'.format( HydrusData.ToHumanInt( num_deleted ) ), 'Tell this log to clear out deleted files, reducing the size of the queue.', ClearFileSeeds, win, file_seed_cache, ( CC.STATUS_DELETED, ) )
if num_errors > 0:
ClientGUIMenus.AppendMenuItem( menu, 'delete {} \'failed\' file import items from the queue'.format( HydrusData.ToHumanInt( num_errors ) ), 'Tell this log to clear out errored files, reducing the size of the queue.', ClearFileSeeds, win, file_seed_cache, ( CC.STATUS_ERROR, ) )
if num_vetoed > 0:
ClientGUIMenus.AppendMenuItem( menu, 'delete {} \'ignored\' file import items from the queue'.format( HydrusData.ToHumanInt( num_vetoed ) ), 'Tell this log to clear out ignored files, reducing the size of the queue.', ClearFileSeeds, win, file_seed_cache, ( CC.STATUS_VETOED, ) )
if num_skipped > 0:
ClientGUIMenus.AppendMenuItem( menu, 'delete {} \'skipped\' file import items from the queue'.format( HydrusData.ToHumanInt( num_skipped ) ), 'Tell this log to clear out skipped files, reducing the size of the queue.', ClearFileSeeds, win, file_seed_cache, ( CC.STATUS_SKIPPED, ) )
ClientGUIMenus.AppendSeparator( menu )
if num_successful > 0:
ClientGUIMenus.AppendMenuItem( menu, 'show new files in a new page', 'Gather the new files in this import list and show them in a new page.', ShowFilesInNewPage, file_seed_cache, show = 'new' )
ClientGUIMenus.AppendMenuItem( menu, 'show all files in a new page', 'Gather the files in this import list and show them in a new page.', ShowFilesInNewPage, file_seed_cache )
ClientGUIMenus.AppendSeparator( menu )
if len( file_seed_cache ) > 0:
2022-12-14 22:22:11 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'reverse import order', 'Reverse the import list so outstanding imports process in the opposite order.', ReverseFileSeedCache, win, file_seed_cache )
ClientGUIMenus.AppendSeparator( menu )
2022-04-06 20:40:17 +00:00
submenu = QW.QMenu( menu )
ClientGUIMenus.AppendMenuItem( submenu, 'to clipboard', 'Copy all the sources in this list to the clipboard.', ExportToClipboard, file_seed_cache )
ClientGUIMenus.AppendMenuItem( submenu, 'to png', 'Export all the sources in this list to a png file.', ExportToPNG, win, file_seed_cache )
ClientGUIMenus.AppendMenu( menu, submenu, 'export all sources' )
submenu = QW.QMenu( menu )
ClientGUIMenus.AppendMenuItem( submenu, 'from clipboard', 'Import new urls or paths to this list from the clipboard.', ImportFromClipboard, win, file_seed_cache )
ClientGUIMenus.AppendMenuItem( submenu, 'from png', 'Import new urls or paths to this list from a png file.', ImportFromPNG, win, file_seed_cache )
ClientGUIMenus.AppendMenu( menu, submenu, 'import new sources' )
2018-06-27 19:27:05 +00:00
class EditFileSeedCachePanel( ClientGUIScrolledPanels.EditPanel ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
def __init__( self, parent, controller, file_seed_cache ):
2017-07-27 00:47:13 +00:00
ClientGUIScrolledPanels.EditPanel.__init__( self, parent )
self._controller = controller
2018-06-27 19:27:05 +00:00
self._file_seed_cache = file_seed_cache
2017-07-27 00:47:13 +00:00
self._text = ClientGUICommon.BetterStaticText( self, 'initialising' )
2018-06-27 19:27:05 +00:00
# add index control row here, hide it if needed and hook into showing/hiding and postsizechangedevent on file_seed add/remove
2017-07-27 00:47:13 +00:00
2020-07-15 20:52:09 +00:00
self._list_ctrl = ClientGUIListCtrl.BetterListCtrl( self, CGLC.COLUMN_LIST_FILE_SEED_CACHE.ID, 30, self._ConvertFileSeedToListCtrlTuples, activation_callback = self._ShowSelectionInNewPage, delete_key_callback = self._DeleteSelected )
2017-07-27 00:47:13 +00:00
#
2018-06-27 19:27:05 +00:00
self._list_ctrl.AddDatas( self._file_seed_cache.GetFileSeeds() )
2017-07-27 00:47:13 +00:00
2020-07-15 20:52:09 +00:00
self._list_ctrl.Sort()
2017-08-30 20:27:47 +00:00
2017-07-27 00:47:13 +00:00
#
2019-11-14 03:56:30 +00:00
vbox = QP.VBoxLayout()
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
QP.AddToLayout( vbox, self._text, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, self._list_ctrl, CC.FLAGS_EXPAND_BOTH_WAYS )
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
self.widget().setLayout( vbox )
2017-07-27 00:47:13 +00:00
2018-09-19 21:54:51 +00:00
self._list_ctrl.AddMenuCallable( self._GetListCtrlMenu )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._controller.sub( self, 'NotifyFileSeedsUpdated', 'file_seed_cache_file_seeds_updated' )
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
QP.CallAfter( self._UpdateText )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
def _ConvertFileSeedToListCtrlTuples( self, file_seed ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
try:
file_seed_index = self._file_seed_cache.GetFileSeedIndex( file_seed )
2018-11-14 23:10:55 +00:00
pretty_file_seed_index = HydrusData.ToHumanInt( file_seed_index )
2018-06-27 19:27:05 +00:00
except:
file_seed_index = '--'
2018-11-14 23:10:55 +00:00
pretty_file_seed_index = '--'
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
file_seed_data = file_seed.file_seed_data
status = file_seed.status
added = file_seed.created
modified = file_seed.modified
source_time = file_seed.source_time
note = file_seed.note
2017-07-27 00:47:13 +00:00
2019-01-09 22:59:03 +00:00
pretty_file_seed_data = str( file_seed_data )
2017-07-27 00:47:13 +00:00
pretty_status = CC.status_string_lookup[ status ]
pretty_added = ClientData.TimestampToPrettyTimeDelta( added )
pretty_modified = ClientData.TimestampToPrettyTimeDelta( modified )
2017-09-27 21:52:54 +00:00
2017-11-29 21:48:23 +00:00
if source_time is None:
2017-09-27 21:52:54 +00:00
pretty_source_time = 'unknown'
else:
pretty_source_time = ClientData.TimestampToPrettyTimeDelta( source_time )
2017-09-27 21:52:54 +00:00
2019-01-09 22:59:03 +00:00
sort_source_time = ClientGUIListCtrl.SafeNoneInt( source_time )
2017-07-27 00:47:13 +00:00
pretty_note = note.split( os.linesep )[0]
2018-06-27 19:27:05 +00:00
display_tuple = ( pretty_file_seed_index, pretty_file_seed_data, pretty_status, pretty_added, pretty_modified, pretty_source_time, pretty_note )
2019-01-09 22:59:03 +00:00
sort_tuple = ( file_seed_index, file_seed_data, status, added, modified, sort_source_time, note )
2017-07-27 00:47:13 +00:00
return ( display_tuple, sort_tuple )
def _CopySelectedNotes( self ):
notes = []
2018-06-27 19:27:05 +00:00
for file_seed in self._list_ctrl.GetData( only_selected = True ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
note = file_seed.note
2017-07-27 00:47:13 +00:00
if note != '':
notes.append( note )
if len( notes ) > 0:
separator = os.linesep * 2
text = separator.join( notes )
HG.client_controller.pub( 'clipboard', 'text', text )
2018-06-27 19:27:05 +00:00
def _CopySelectedFileSeedData( self ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
file_seeds = self._list_ctrl.GetData( only_selected = True )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if len( file_seeds ) > 0:
2017-07-27 00:47:13 +00:00
separator = os.linesep * 2
2018-06-27 19:27:05 +00:00
text = separator.join( ( file_seed.file_seed_data for file_seed in file_seeds ) )
2017-07-27 00:47:13 +00:00
HG.client_controller.pub( 'clipboard', 'text', text )
def _DeleteSelected( self ):
2018-06-27 19:27:05 +00:00
file_seeds_to_delete = self._list_ctrl.GetData( only_selected = True )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if len( file_seeds_to_delete ) > 0:
2017-07-27 00:47:13 +00:00
2022-01-12 22:14:50 +00:00
message = 'Are you sure you want to delete the {} selected entries?'.format( HydrusData.ToHumanInt( len( file_seeds_to_delete ) ) )
2017-07-27 00:47:13 +00:00
2019-09-05 00:05:32 +00:00
result = ClientGUIDialogsQuick.GetYesNo( self, message )
2019-11-14 03:56:30 +00:00
if result == QW.QDialog.Accepted:
2017-07-27 00:47:13 +00:00
2019-09-05 00:05:32 +00:00
self._file_seed_cache.RemoveFileSeeds( file_seeds_to_delete )
2017-07-27 00:47:13 +00:00
2018-09-19 21:54:51 +00:00
def _GetListCtrlMenu( self ):
selected_file_seeds = self._list_ctrl.GetData( only_selected = True )
2022-04-06 20:40:17 +00:00
menu = QW.QMenu()
2018-09-19 21:54:51 +00:00
if len( selected_file_seeds ) == 0:
2022-04-06 20:40:17 +00:00
PopulateFileSeedCacheMenu( self, menu, self._file_seed_cache )
return menu
2018-09-19 21:54:51 +00:00
2022-04-06 20:40:17 +00:00
ClientGUIMenus.AppendSeparator( menu )
2018-09-19 21:54:51 +00:00
can_show_files_in_new_page = True in ( file_seed.HasHash() for file_seed in selected_file_seeds )
if can_show_files_in_new_page:
2019-11-14 03:56:30 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'open selected import files in a new page', 'Show all the known selected files in a new thumbnail page. This is complicated, so cannot always be guaranteed, even if the import says \'success\'.', self._ShowSelectionInNewPage )
2018-09-19 21:54:51 +00:00
ClientGUIMenus.AppendSeparator( menu )
2021-06-23 21:11:38 +00:00
2019-11-14 03:56:30 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'copy sources', 'Copy all the selected sources to clipboard.', self._CopySelectedFileSeedData )
ClientGUIMenus.AppendMenuItem( menu, 'copy notes', 'Copy all the selected notes to clipboard.', self._CopySelectedNotes )
2018-09-19 21:54:51 +00:00
if len( selected_file_seeds ) == 1:
ClientGUIMenus.AppendSeparator( menu )
( selected_file_seed, ) = selected_file_seeds
hash_types_to_hashes = selected_file_seed.GetHashTypesToHashes()
if len( hash_types_to_hashes ) == 0:
ClientGUIMenus.AppendMenuLabel( menu, 'no hashes yet' )
else:
hash_submenu = QW.QMenu( menu )
for hash_type in ( 'sha256', 'md5', 'sha1', 'sha512' ):
if hash_type in hash_types_to_hashes:
h = hash_types_to_hashes[ hash_type ]
ClientGUIMenus.AppendMenuLabel( hash_submenu, '{}:{}'.format( hash_type, h.hex() ) )
ClientGUIMenus.AppendMenu( menu, hash_submenu, 'hashes' )
#
if selected_file_seed.IsURLFileImport():
2021-10-13 20:16:57 +00:00
referral_url = selected_file_seed.GetReferralURL()
primary_urls = sorted( selected_file_seed.GetPrimaryURLs() )
source_urls = sorted( selected_file_seed.GetSourceURLs() )
2021-10-13 20:16:57 +00:00
if referral_url is None and len( primary_urls ) + len( source_urls ) == 0:
2021-10-13 20:16:57 +00:00
ClientGUIMenus.AppendMenuLabel( menu, 'no additional urls' )
else:
url_submenu = QW.QMenu( menu )
2021-10-13 20:16:57 +00:00
if referral_url is not None:
2021-10-13 20:16:57 +00:00
ClientGUIMenus.AppendMenuLabel( url_submenu, 'referral url:' )
ClientGUIMenus.AppendMenuLabel( url_submenu, referral_url )
2021-10-13 20:16:57 +00:00
if len( primary_urls ) > 0:
ClientGUIMenus.AppendSeparator( url_submenu )
ClientGUIMenus.AppendMenuLabel( url_submenu, 'primary urls:' )
for url in primary_urls:
ClientGUIMenus.AppendMenuLabel( url_submenu, url )
if len( source_urls ) > 0:
ClientGUIMenus.AppendSeparator( url_submenu )
ClientGUIMenus.AppendMenuLabel( url_submenu, 'source urls:' )
for url in source_urls:
ClientGUIMenus.AppendMenuLabel( url_submenu, url )
ClientGUIMenus.AppendMenu( menu, url_submenu, 'additional urls' )
#
tags = list( selected_file_seed.GetExternalTags() )
tag_sort = ClientTagSorting.TagSort( sort_type = ClientTagSorting.SORT_BY_HUMAN_TAG, sort_order = CC.SORT_ASC )
ClientTagSorting.SortTags( tag_sort, tags )
if len( tags ) == 0:
ClientGUIMenus.AppendMenuLabel( menu, 'no parsed tags' )
else:
tag_submenu = QW.QMenu( menu )
for tag in tags:
ClientGUIMenus.AppendMenuLabel( tag_submenu, tag )
ClientGUIMenus.AppendMenu( menu, tag_submenu, 'parsed tags' )
2018-09-19 21:54:51 +00:00
ClientGUIMenus.AppendSeparator( menu )
2021-06-23 21:11:38 +00:00
2019-11-14 03:56:30 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'open sources', 'Open all the selected sources in your file explorer or web browser.', self._OpenSelectedFileSeedData )
2018-09-19 21:54:51 +00:00
ClientGUIMenus.AppendSeparator( menu )
2021-06-23 21:11:38 +00:00
2019-11-14 03:56:30 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'try again', 'Reset the progress of all the selected imports.', HydrusData.Call( self._SetSelected, CC.STATUS_UNKNOWN ) )
2021-06-23 21:11:38 +00:00
2019-11-14 03:56:30 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'skip', 'Skip all the selected imports.', HydrusData.Call( self._SetSelected, CC.STATUS_SKIPPED ) )
2021-06-23 21:11:38 +00:00
ClientGUIMenus.AppendMenuItem( menu, 'delete from list', 'Remove all the selected imports.', HydrusData.Call( self._DeleteSelected ) )
2018-09-19 21:54:51 +00:00
2022-04-06 20:40:17 +00:00
ClientGUIMenus.AppendSeparator( menu )
submenu = QW.QMenu( menu )
PopulateFileSeedCacheMenu( self, submenu, self._file_seed_cache )
ClientGUIMenus.AppendMenu( menu, submenu, 'whole log' )
2018-09-19 21:54:51 +00:00
return menu
2018-06-27 19:27:05 +00:00
def _OpenSelectedFileSeedData( self ):
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
file_seeds = self._list_ctrl.GetData( only_selected = True )
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
if len( file_seeds ) > 0:
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
if len( file_seeds ) > 10:
2017-12-13 22:33:07 +00:00
message = 'You have many objects selected--are you sure you want to open them all?'
2019-09-05 00:05:32 +00:00
result = ClientGUIDialogsQuick.GetYesNo( self, message )
2019-11-14 03:56:30 +00:00
if result != QW.QDialog.Accepted:
2017-12-13 22:33:07 +00:00
2019-09-05 00:05:32 +00:00
return
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
if file_seeds[0].file_seed_data.startswith( 'http' ):
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
for file_seed in file_seeds:
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
ClientPaths.LaunchURLInWebBrowser( file_seed.file_seed_data )
2017-12-13 22:33:07 +00:00
else:
try:
2018-06-27 19:27:05 +00:00
for file_seed in file_seeds:
2017-12-13 22:33:07 +00:00
2018-06-27 19:27:05 +00:00
HydrusPaths.OpenFileLocation( file_seed.file_seed_data )
2017-12-13 22:33:07 +00:00
except Exception as e:
2019-11-14 03:56:30 +00:00
QW.QMessageBox.critical( self, 'Error', str(e) )
2017-12-13 22:33:07 +00:00
2017-07-27 00:47:13 +00:00
def _SetSelected( self, status_to_set ):
2018-06-27 19:27:05 +00:00
file_seeds = self._list_ctrl.GetData( only_selected = True )
2017-11-29 21:48:23 +00:00
2018-10-03 21:00:15 +00:00
if status_to_set == CC.STATUS_UNKNOWN:
2018-10-24 21:34:02 +00:00
deleted_and_clearable_file_seeds = [ file_seed for file_seed in file_seeds if file_seed.IsDeleted() and file_seed.HasHash() ]
2018-10-03 21:00:15 +00:00
2018-10-24 21:34:02 +00:00
if len( deleted_and_clearable_file_seeds ) > 0:
2018-10-03 21:00:15 +00:00
message = 'One or more of these files did not import due to being previously deleted. They will likely fail again unless you erase those deletion records. Would you like to do this now?'
2019-09-05 00:05:32 +00:00
result = ClientGUIDialogsQuick.GetYesNo( self, message )
2019-11-14 03:56:30 +00:00
if result == QW.QDialog.DialogCode.Accepted:
2018-10-03 21:00:15 +00:00
2019-09-05 00:05:32 +00:00
deletee_hashes = { file_seed.GetHash() for file_seed in deleted_and_clearable_file_seeds }
2021-04-28 21:43:16 +00:00
from hydrus.client.gui import ClientGUIMediaActions
ClientGUIMediaActions.UndeleteFiles( deletee_hashes )
2022-12-21 22:00:27 +00:00
content_update_erase_record = HydrusData.ContentUpdate( HC.CONTENT_TYPE_FILES, HC.CONTENT_UPDATE_CLEAR_DELETE_RECORD, deletee_hashes )
2019-09-05 00:05:32 +00:00
2021-04-28 21:43:16 +00:00
service_keys_to_content_updates = { CC.COMBINED_LOCAL_FILE_SERVICE_KEY : [ content_update_erase_record ] }
2019-09-05 00:05:32 +00:00
HG.client_controller.WriteSynchronous( 'content_updates', service_keys_to_content_updates )
2018-10-03 21:00:15 +00:00
2018-06-27 19:27:05 +00:00
for file_seed in file_seeds:
2017-11-29 21:48:23 +00:00
2018-06-27 19:27:05 +00:00
file_seed.SetStatus( status_to_set )
2017-11-29 21:48:23 +00:00
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._file_seed_cache.NotifyFileSeedsUpdated( file_seeds )
2017-07-27 00:47:13 +00:00
2018-05-09 20:23:00 +00:00
def _ShowSelectionInNewPage( self ):
hashes = []
2018-06-27 19:27:05 +00:00
for file_seed in self._list_ctrl.GetData( only_selected = True ):
2018-05-09 20:23:00 +00:00
2018-06-27 19:27:05 +00:00
if file_seed.HasHash():
2018-05-09 20:23:00 +00:00
2018-06-27 19:27:05 +00:00
hashes.append( file_seed.GetHash() )
2018-05-09 20:23:00 +00:00
if len( hashes ) > 0:
2022-05-25 21:30:53 +00:00
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
2022-01-19 21:28:59 +00:00
HG.client_controller.pub( 'new_page_query', location_context, initial_hashes = hashes )
2018-05-09 20:23:00 +00:00
2018-06-27 19:27:05 +00:00
def _UpdateListCtrl( self, file_seeds ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
file_seeds_to_add = []
file_seeds_to_update = []
file_seeds_to_delete = []
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
for file_seed in file_seeds:
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if self._file_seed_cache.HasFileSeed( file_seed ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if self._list_ctrl.HasData( file_seed ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
file_seeds_to_update.append( file_seed )
2017-07-27 00:47:13 +00:00
else:
2018-06-27 19:27:05 +00:00
file_seeds_to_add.append( file_seed )
2017-07-27 00:47:13 +00:00
else:
2018-06-27 19:27:05 +00:00
if self._list_ctrl.HasData( file_seed ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
file_seeds_to_delete.append( file_seed )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._list_ctrl.DeleteDatas( file_seeds_to_delete )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if len( file_seeds_to_add ) > 0:
2018-06-20 20:20:22 +00:00
2018-06-27 19:27:05 +00:00
self._list_ctrl.AddDatas( file_seeds_to_add )
2018-06-20 20:20:22 +00:00
2018-06-27 19:27:05 +00:00
# if file_seeds are inserted, then all subsequent indices need to be shuffled up, hence just update all here
2018-06-20 20:20:22 +00:00
self._list_ctrl.UpdateDatas()
else:
2018-06-27 19:27:05 +00:00
self._list_ctrl.UpdateDatas( file_seeds_to_update )
2018-06-20 20:20:22 +00:00
2017-07-27 00:47:13 +00:00
def _UpdateText( self ):
2020-06-11 12:01:08 +00:00
file_seed_cache_status = self._file_seed_cache.GetStatus()
2017-07-27 00:47:13 +00:00
2020-06-11 12:01:08 +00:00
self._text.setText( file_seed_cache_status.GetStatusText() )
2017-07-27 00:47:13 +00:00
def GetValue( self ):
2018-06-27 19:27:05 +00:00
return self._file_seed_cache
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
def NotifyFileSeedsUpdated( self, file_seed_cache_key, file_seeds ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
if file_seed_cache_key == self._file_seed_cache.GetFileSeedCacheKey():
2017-07-27 00:47:13 +00:00
self._UpdateText()
2018-06-27 19:27:05 +00:00
self._UpdateListCtrl( file_seeds )
2017-07-27 00:47:13 +00:00
2021-12-22 22:31:23 +00:00
class FileSeedCacheButton( ClientGUICommon.ButtonWithMenuArrow ):
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
def __init__( self, parent, controller, file_seed_cache_get_callable, file_seed_cache_set_callable = None ):
2017-10-18 19:41:25 +00:00
self._controller = controller
2018-06-27 19:27:05 +00:00
self._file_seed_cache_get_callable = file_seed_cache_get_callable
self._file_seed_cache_set_callable = file_seed_cache_set_callable
2017-10-18 19:41:25 +00:00
2021-12-22 22:31:23 +00:00
action = QW.QAction()
action.setText( 'file log' )
action.setToolTip( 'open detailed file log' )
action.triggered.connect( self._ShowFileSeedCacheFrame )
ClientGUICommon.ButtonWithMenuArrow.__init__( self, parent, action )
def _PopulateMenu( self, menu ):
file_seed_cache = self._file_seed_cache_get_callable()
2022-04-06 20:40:17 +00:00
PopulateFileSeedCacheMenu( self, menu, file_seed_cache )
2021-07-14 20:42:19 +00:00
2018-07-11 20:23:51 +00:00
2018-06-27 19:27:05 +00:00
def _ShowFileSeedCacheFrame( self ):
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
file_seed_cache = self._file_seed_cache_get_callable()
2017-10-18 19:41:25 +00:00
2019-12-11 23:18:37 +00:00
tlw = self.window()
2017-10-18 19:41:25 +00:00
2021-07-07 20:48:57 +00:00
title = 'file log'
2019-12-11 23:18:37 +00:00
if isinstance( tlw, QP.Dialog ):
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
if self._file_seed_cache_set_callable is None: # throw up a dialog that edits the file_seed cache in place
2017-10-18 19:41:25 +00:00
2021-07-07 20:48:57 +00:00
with ClientGUITopLevelWindowsPanels.DialogNullipotent( self, title ) as dlg:
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
panel = EditFileSeedCachePanel( dlg, self._controller, file_seed_cache )
2017-10-18 19:41:25 +00:00
dlg.SetPanel( panel )
2019-11-14 03:56:30 +00:00
dlg.exec()
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
else: # throw up a dialog that edits the file_seed cache but can be cancelled
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
dupe_file_seed_cache = file_seed_cache.Duplicate()
2017-10-18 19:41:25 +00:00
2021-07-07 20:48:57 +00:00
with ClientGUITopLevelWindowsPanels.DialogEdit( self, title ) as dlg:
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
panel = EditFileSeedCachePanel( dlg, self._controller, dupe_file_seed_cache )
2017-10-18 19:41:25 +00:00
dlg.SetPanel( panel )
2019-11-14 03:56:30 +00:00
if dlg.exec() == QW.QDialog.Accepted:
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
self._file_seed_cache_set_callable( dupe_file_seed_cache )
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
else: # throw up a frame that edits the file_seed cache in place
2017-10-18 19:41:25 +00:00
frame_key = 'file_import_status'
2020-04-29 21:44:12 +00:00
frame = ClientGUITopLevelWindowsPanels.FrameThatTakesScrollablePanel( self, title, frame_key )
2017-10-18 19:41:25 +00:00
2018-06-27 19:27:05 +00:00
panel = EditFileSeedCachePanel( frame, self._controller, file_seed_cache )
2017-10-18 19:41:25 +00:00
frame.SetPanel( panel )
2019-11-14 03:56:30 +00:00
class FileSeedCacheStatusControl( QW.QFrame ):
2017-07-27 00:47:13 +00:00
2018-05-09 20:23:00 +00:00
def __init__( self, parent, controller, page_key = None ):
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
QW.QFrame.__init__( self, parent )
self.setFrameStyle( QW.QFrame.Box | QW.QFrame.Raised )
2017-07-27 00:47:13 +00:00
self._controller = controller
2018-05-09 20:23:00 +00:00
self._page_key = page_key
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._file_seed_cache = None
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
self._import_summary_st = ClientGUICommon.BetterStaticText( self, ellipsize_end = True )
self._progress_st = ClientGUICommon.BetterStaticText( self, ellipsize_end = True )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._file_seed_cache_button = FileSeedCacheButton( self, self._controller, self._GetFileSeedCache )
2017-07-27 00:47:13 +00:00
self._progress_gauge = ClientGUICommon.Gauge( self )
#
self._Update()
#
2019-11-14 03:56:30 +00:00
hbox = QP.HBoxLayout()
2017-07-27 00:47:13 +00:00
2020-07-29 20:52:44 +00:00
QP.AddToLayout( hbox, self._progress_st, CC.FLAGS_CENTER_PERPENDICULAR_EXPAND_DEPTH )
QP.AddToLayout( hbox, self._file_seed_cache_button, CC.FLAGS_CENTER_PERPENDICULAR )
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
vbox = QP.VBoxLayout()
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
QP.AddToLayout( vbox, self._import_summary_st, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
QP.AddToLayout( vbox, self._progress_gauge, CC.FLAGS_EXPAND_PERPENDICULAR )
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
self.setLayout( vbox )
2017-07-27 00:47:13 +00:00
#
2018-02-14 21:47:18 +00:00
HG.client_controller.gui.RegisterUIUpdateWindow( self )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
def _GetFileSeedCache( self ):
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
return self._file_seed_cache
2017-07-27 00:47:13 +00:00
def _Update( self ):
2018-06-27 19:27:05 +00:00
if self._file_seed_cache is None:
2017-07-27 00:47:13 +00:00
2020-12-23 23:07:58 +00:00
self._import_summary_st.clear()
self._progress_st.clear()
2017-07-27 00:47:13 +00:00
self._progress_gauge.SetRange( 1 )
self._progress_gauge.SetValue( 0 )
2019-11-14 03:56:30 +00:00
if self._file_seed_cache_button.isEnabled():
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
self._file_seed_cache_button.setEnabled( False )
2017-07-27 00:47:13 +00:00
else:
2020-06-11 12:01:08 +00:00
file_seed_cache_status = self._file_seed_cache.GetStatus()
2017-07-27 00:47:13 +00:00
2020-06-11 12:01:08 +00:00
( num_done, num_to_do ) = file_seed_cache_status.GetValueRange()
self._import_summary_st.setText( file_seed_cache_status.GetStatusText() )
2017-07-27 00:47:13 +00:00
if num_to_do == 0:
2020-12-23 23:07:58 +00:00
self._progress_st.clear()
2017-07-27 00:47:13 +00:00
else:
2019-11-14 03:56:30 +00:00
self._progress_st.setText( HydrusData.ConvertValueRangeToPrettyString(num_done,num_to_do) )
2017-07-27 00:47:13 +00:00
self._progress_gauge.SetRange( num_to_do )
self._progress_gauge.SetValue( num_done )
2019-11-14 03:56:30 +00:00
if not self._file_seed_cache_button.isEnabled():
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
self._file_seed_cache_button.setEnabled( True )
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
def SetFileSeedCache( self, file_seed_cache ):
2017-07-27 00:47:13 +00:00
2019-11-14 03:56:30 +00:00
if not self or not QP.isValid( self ):
2017-07-27 00:47:13 +00:00
2018-02-14 21:47:18 +00:00
return
2017-07-27 00:47:13 +00:00
2018-06-27 19:27:05 +00:00
self._file_seed_cache = file_seed_cache
2018-02-14 21:47:18 +00:00
2017-07-27 00:47:13 +00:00
2018-02-14 21:47:18 +00:00
def TIMERUIUpdate( self ):
2017-07-27 00:47:13 +00:00
2018-05-09 20:23:00 +00:00
do_it_anyway = False
2018-06-27 19:27:05 +00:00
if self._file_seed_cache is not None:
2018-05-09 20:23:00 +00:00
2020-06-11 12:01:08 +00:00
file_seed_cache_status = self._file_seed_cache.GetStatus()
( num_done, num_to_do ) = file_seed_cache_status.GetValueRange()
2018-05-09 20:23:00 +00:00
( old_num_done, old_num_to_do ) = self._progress_gauge.GetValueRange()
if old_num_done != num_done or old_num_to_do != num_to_do:
if self._page_key is not None:
do_it_anyway = True # to update the gauge
HG.client_controller.pub( 'refresh_page_name', self._page_key )
if self._controller.gui.IShouldRegularlyUpdate( self ) or do_it_anyway:
2017-07-27 00:47:13 +00:00
self._Update()