hydrus/hydrus/client/gui/ClientGUIScrolledPanelsRevi...

4623 lines
180 KiB
Python

import collections
import os
import queue
import threading
import time
import typing
from PIL import ExifTags
from qtpy import QtCore as QC
from qtpy import QtWidgets as QW
from qtpy import QtGui as QG
from hydrus.core import HydrusCompression
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 HydrusSerialisable
from hydrus.core import HydrusTags
from hydrus.core import HydrusTagArchive
from hydrus.core import HydrusText
from hydrus.core import HydrusThreading
from hydrus.core import HydrusTime
from hydrus.core.files import HydrusFileHandling
from hydrus.client import ClientConstants as CC
from hydrus.client import ClientData
from hydrus.client import ClientFiles
from hydrus.client import ClientLocation
from hydrus.client import ClientMigration
from hydrus.client import ClientParsing
from hydrus.client import ClientFilesPhysical
from hydrus.client import ClientRendering
from hydrus.client import ClientSerialisable
from hydrus.client import ClientThreading
from hydrus.client.gui import ClientGUIAsync
from hydrus.client.gui import ClientGUICharts
from hydrus.client.gui import ClientGUIDialogsMessage
from hydrus.client.gui import ClientGUIDialogsQuick
from hydrus.client.gui import ClientGUIDragDrop
from hydrus.client.gui import ClientGUIFunctions
from hydrus.client.gui import ClientGUIScrolledPanels
from hydrus.client.gui import ClientGUITags
from hydrus.client.gui import ClientGUITopLevelWindowsPanels
from hydrus.client.gui import QtPorting as QP
from hydrus.client.gui.importing import ClientGUIImport
from hydrus.client.gui.importing import ClientGUIImportOptions
from hydrus.client.gui.lists import ClientGUIListConstants as CGLC
from hydrus.client.gui.lists import ClientGUIListCtrl
from hydrus.client.gui.search import ClientGUIACDropdown
from hydrus.client.gui.search import ClientGUILocation
from hydrus.client.gui.widgets import ClientGUICommon
from hydrus.client.gui.widgets import ClientGUIControls
from hydrus.client.gui.widgets import ClientGUIMenuButton
from hydrus.client.importing.options import FileImportOptions
from hydrus.client.metadata import ClientTags
from hydrus.client.networking import ClientNetworkingContexts
from hydrus.client.networking import ClientNetworkingDomain
from hydrus.client.networking import ClientNetworkingGUG
from hydrus.client.networking import ClientNetworkingLogin
from hydrus.client.networking import ClientNetworkingURLClass
from hydrus.client.search import ClientSearch
class AboutPanel( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, name, version, description, license_text, developers, site ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
icon_label = ClientGUICommon.BetterStaticText( self )
icon_label.setPixmap( HG.client_controller.frame_icon_pixmap )
name_label = ClientGUICommon.BetterStaticText( self, name )
name_label_font = name_label.font()
name_label_font.setBold( True )
name_label.setFont( name_label_font )
version_label = ClientGUICommon.BetterStaticText( self, version )
tabwidget = QW.QTabWidget( self )
desc_panel = QW.QWidget( self )
desc_label = ClientGUICommon.BetterStaticText( self, description )
desc_label.setAlignment( QC.Qt.AlignHCenter | QC.Qt.AlignVCenter )
url_label = ClientGUICommon.BetterHyperLink( self, site, site )
credits = QW.QTextEdit( self )
credits.setPlainText( 'Created by ' + ', '.join( developers ) )
credits.setReadOnly( True )
credits.setAlignment( QC.Qt.AlignHCenter )
license_textedit = QW.QTextEdit( self )
license_textedit.setPlainText( license_text )
license_textedit.setReadOnly( True )
tabwidget.addTab( desc_panel, 'Description' )
tabwidget.addTab( credits, 'Credits' )
tabwidget.addTab( license_textedit, 'License' )
tabwidget.setCurrentIndex( 0 )
desc_layout = QP.VBoxLayout()
QP.AddToLayout( desc_layout, desc_label, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( desc_layout, url_label, CC.FLAGS_CENTER_PERPENDICULAR )
desc_panel.setLayout( desc_layout )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, icon_label, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( vbox, name_label, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( vbox, version_label, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( vbox, tabwidget, CC.FLAGS_CENTER_PERPENDICULAR )
self.widget().setLayout( vbox )
class MigrateDatabasePanel( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, controller ):
self._controller = controller
self._new_options = self._controller.new_options
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._client_files_subfolders = HG.client_controller.Read( 'client_files_subfolders' )
( self._media_base_locations, self._ideal_thumbnails_base_location_override ) = self._controller.Read( 'ideal_client_files_locations' )
service_info = HG.client_controller.Read( 'service_info', CC.COMBINED_LOCAL_FILE_SERVICE_KEY )
self._all_local_files_total_size = service_info[ HC.SERVICE_INFO_TOTAL_SIZE ]
self._all_local_files_total_num = service_info[ HC.SERVICE_INFO_NUM_FILES ]
menu_items = []
page_func = HydrusData.Call( ClientGUIDialogsQuick.OpenDocumentation, self, HC.DOCUMENTATION_DATABASE_MIGRATION )
menu_items.append( ( 'normal', 'open the html migration help', 'Open the help page for database migration in your web browser.', page_func ) )
help_button = ClientGUIMenuButton.MenuBitmapButton( self, CC.global_pixmaps().help, menu_items )
help_hbox = ClientGUICommon.WrapInText( help_button, self, 'help for this panel -->', object_name = 'HydrusIndeterminate' )
#
info_panel = ClientGUICommon.StaticBox( self, 'database components' )
self._current_install_path_st = ClientGUICommon.BetterStaticText( info_panel )
self._current_db_path_st = ClientGUICommon.BetterStaticText( info_panel )
self._current_media_paths_st = ClientGUICommon.BetterStaticText( info_panel )
file_locations_panel = ClientGUICommon.StaticBox( self, 'media file locations' )
current_media_base_locations_listctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( file_locations_panel )
self._current_media_base_locations_listctrl = ClientGUIListCtrl.BetterListCtrl( current_media_base_locations_listctrl_panel, CGLC.COLUMN_LIST_DB_MIGRATION_LOCATIONS.ID, 8, self._ConvertLocationToListCtrlTuples, activation_callback = self._SetMaxNumBytes )
self._current_media_base_locations_listctrl.setSelectionMode( QW.QAbstractItemView.SingleSelection )
self._current_media_base_locations_listctrl.Sort()
current_media_base_locations_listctrl_panel.SetListCtrl( self._current_media_base_locations_listctrl )
current_media_base_locations_listctrl_panel.AddButton( 'add new location for files', self._SelectPathToAdd )
current_media_base_locations_listctrl_panel.AddButton( 'increase location weight', self._IncreaseWeight, enabled_check_func = self._CanIncreaseWeight )
current_media_base_locations_listctrl_panel.AddButton( 'decrease location weight', self._DecreaseWeight, enabled_check_func = self._CanDecreaseWeight )
current_media_base_locations_listctrl_panel.AddButton( 'set max size', self._SetMaxNumBytes, enabled_check_func = self._CanSetMaxNumBytes )
current_media_base_locations_listctrl_panel.AddButton( 'remove location', self._RemoveSelectedBaseLocation, enabled_check_func = self._CanRemoveLocation )
self._thumbnails_location = QW.QLineEdit( file_locations_panel )
self._thumbnails_location.setEnabled( False )
self._thumbnails_location_set = ClientGUICommon.BetterButton( file_locations_panel, 'set', self._SetThumbnailLocation )
self._thumbnails_location_clear = ClientGUICommon.BetterButton( file_locations_panel, 'clear', self._ClearThumbnailLocation )
self._rebalance_status_st = ClientGUICommon.BetterStaticText( file_locations_panel )
self._rebalance_status_st.setAlignment( QC.Qt.AlignRight | QC.Qt.AlignVCenter )
self._rebalance_button = ClientGUICommon.BetterButton( file_locations_panel, 'move files now', self._Rebalance )
#
info_panel.Add( self._current_install_path_st, CC.FLAGS_EXPAND_PERPENDICULAR )
info_panel.Add( self._current_db_path_st, CC.FLAGS_EXPAND_PERPENDICULAR )
info_panel.Add( self._current_media_paths_st, CC.FLAGS_EXPAND_PERPENDICULAR )
t_hbox = QP.HBoxLayout()
QP.AddToLayout( t_hbox, ClientGUICommon.BetterStaticText( file_locations_panel, 'thumbnail location override' ), CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( t_hbox, self._thumbnails_location, CC.FLAGS_CENTER_PERPENDICULAR_EXPAND_DEPTH )
QP.AddToLayout( t_hbox, self._thumbnails_location_set, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( t_hbox, self._thumbnails_location_clear, CC.FLAGS_CENTER_PERPENDICULAR )
rebalance_hbox = QP.HBoxLayout()
QP.AddToLayout( rebalance_hbox, self._rebalance_status_st, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( rebalance_hbox, self._rebalance_button, CC.FLAGS_CENTER_PERPENDICULAR )
file_locations_panel.Add( current_media_base_locations_listctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
file_locations_panel.Add( t_hbox, CC.FLAGS_EXPAND_PERPENDICULAR )
file_locations_panel.Add( rebalance_hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
#
vbox = QP.VBoxLayout()
message = 'THIS IS ADVANCED. DO NOT CHANGE ANYTHING HERE UNLESS YOU KNOW WHAT IT DOES!'
warning_st = ClientGUICommon.BetterStaticText( self, label = message )
warning_st.setObjectName( 'HydrusWarning' )
QP.AddToLayout( vbox, warning_st, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, help_hbox, CC.FLAGS_ON_RIGHT )
QP.AddToLayout( vbox, info_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, file_locations_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( vbox )
min_width = ClientGUIFunctions.ConvertTextToPixelWidth( self, 100 )
self.setMinimumWidth( min_width )
self._Update()
def _AddBaseLocation( self, base_location ):
if base_location in self._media_base_locations:
ClientGUIDialogsMessage.ShowWarning( self, 'You already have that location entered!' )
return
if self._ideal_thumbnails_base_location_override is not None and base_location.path == self._ideal_thumbnails_base_location_override.path:
ClientGUIDialogsMessage.ShowWarning( self, 'That path is already used as the special thumbnail location--please choose another.' )
return
self._media_base_locations.add( base_location )
self._SaveToDB()
def _AdjustWeight( self, amount ):
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
new_amount = base_location.ideal_weight + amount
if new_amount > 0:
base_location.ideal_weight = new_amount
self._SaveToDB()
elif new_amount <= 0:
self._RemoveBaseLocation( base_location )
else:
if amount > 0:
new_base_location = ClientFilesPhysical.FilesStorageBaseLocation( base_location.path, amount )
if base_location != self._ideal_thumbnails_base_location_override:
self._AddBaseLocation( new_base_location )
def _CanDecreaseWeight( self ):
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
is_big = base_location.ideal_weight > 1
others_can_take_slack = len( self._media_base_locations ) > 1
if is_big or others_can_take_slack:
return True
return False
def _CanIncreaseWeight( self ):
( locations_to_file_weights, locations_to_thumb_weights ) = self._GetLocationsToCurrentWeights()
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
if len( self._media_base_locations ) > 1:
return True
elif base_location in locations_to_file_weights:
return True
return False
def _CanRemoveLocation( self ):
( locations_to_file_weights, locations_to_thumb_weights ) = self._GetLocationsToCurrentWeights()
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
others_can_take_slack = len( self._media_base_locations ) > 1
if others_can_take_slack:
return True
return False
def _CanSetMaxNumBytes( self ):
only_one_location_is_limitless = len( [ 1 for base_location in self._media_base_locations if base_location.max_num_bytes is None ] ) == 1
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
if base_location.max_num_bytes is None and only_one_location_is_limitless:
return False
return True
return False
def _ClearThumbnailLocation( self ):
message = 'Clear the custom thumbnail location? This will schedule all the thumbnail files to migrate back into the regular file locations.'
result = ClientGUIDialogsQuick.GetYesNo( self, message )
if result != QW.QDialog.Accepted:
return
self._ideal_thumbnails_base_location_override = None
self._SaveToDB()
def _ConvertLocationToListCtrlTuples( self, base_location: ClientFilesPhysical.FilesStorageBaseLocation ):
f_space = self._all_local_files_total_size
( t_space_min, t_space_max ) = self._GetThumbnailSizeEstimates()
# current
( locations_to_file_weights, locations_to_thumb_weights ) = self._GetLocationsToCurrentWeights()
#
path = base_location.path
if os.path.exists( path ):
pretty_location = path
try:
free_space = HydrusPaths.GetFreeSpace( path )
pretty_free_space = HydrusData.ToHumanBytes( free_space )
except Exception as e:
message = 'There was a problem finding the free space for "{path}"! Perhaps this location does not exist?'
HydrusData.Print( message )
HydrusData.PrintException( e )
free_space = 0
pretty_free_space = 'problem finding free space'
else:
pretty_location = 'DOES NOT EXIST: {}'.format( path )
free_space = 0
pretty_free_space = 'DOES NOT EXIST'
portable_location = HydrusPaths.ConvertAbsPathToPortablePath( base_location.path )
portable = not os.path.isabs( portable_location )
if portable:
pretty_portable = 'yes'
else:
pretty_portable = 'no'
fp = locations_to_file_weights[ base_location ]
tp = locations_to_thumb_weights[ base_location ]
p = HydrusData.ConvertFloatToPercentage
current_bytes = fp * f_space + tp * t_space_max
current_usage = ( fp, tp )
usages = []
if fp > 0:
usages.append( p( fp ) + ' files' )
if tp > 0:
usages.append( p( tp ) + ' thumbnails' )
if len( usages ) > 0:
if fp == tp:
usages = [ p( fp ) + ' everything' ]
pretty_current_usage = HydrusData.ToHumanBytes( current_bytes ) + ' - ' + ','.join( usages )
else:
pretty_current_usage = 'nothing'
#
if base_location in self._media_base_locations:
ideal_weight = base_location.ideal_weight
pretty_ideal_weight = str( int( ideal_weight ) )
else:
ideal_weight = 0
if base_location in locations_to_file_weights:
pretty_ideal_weight = '0'
else:
pretty_ideal_weight = 'n/a'
if base_location.max_num_bytes is None:
pretty_max_num_bytes = 'n/a'
sort_max_num_bytes = -1
else:
pretty_max_num_bytes = HydrusData.ToHumanBytes( base_location.max_num_bytes )
sort_max_num_bytes = base_location.max_num_bytes
ideal_fp = self._media_base_locations_to_ideal_usage.get( base_location, 0.0 )
if self._ideal_thumbnails_base_location_override is None:
ideal_tp = ideal_fp
else:
if base_location == self._ideal_thumbnails_base_location_override:
ideal_tp = 1.0
else:
ideal_tp = 0.0
ideal_bytes = ideal_fp * f_space + ideal_tp * t_space_max
ideal_usage = ( ideal_fp, ideal_tp )
usages = []
if ideal_fp > 0:
usages.append( p( ideal_fp ) + ' files' )
if ideal_tp > 0:
usages.append( p( ideal_tp ) + ' thumbnails' )
if len( usages ) > 0:
if ideal_fp == ideal_tp:
usages = [ p( ideal_fp ) + ' everything' ]
pretty_ideal_usage = HydrusData.ToHumanBytes( ideal_bytes ) + ' - ' + ','.join( usages )
else:
pretty_ideal_usage = 'nothing'
display_tuple = ( pretty_location, pretty_portable, pretty_free_space, pretty_current_usage, pretty_ideal_weight, pretty_max_num_bytes, pretty_ideal_usage )
sort_tuple = ( pretty_location, portable, free_space, ideal_weight, ideal_usage, sort_max_num_bytes, current_usage )
return ( display_tuple, sort_tuple )
def _DecreaseWeight( self ):
self._AdjustWeight( -1 )
def _GetLocationsToCurrentWeights( self ):
locations_to_file_weights = collections.Counter()
locations_to_thumb_weights = collections.Counter()
for client_files_subfolder in self._client_files_subfolders:
prefix = client_files_subfolder.prefix
base_location = client_files_subfolder.base_location
subfolder_weight = client_files_subfolder.GetNormalisedWeight()
if prefix.startswith( 'f' ):
locations_to_file_weights[ base_location ] += subfolder_weight
if prefix.startswith( 't' ):
locations_to_thumb_weights[ base_location ] += subfolder_weight
return ( locations_to_file_weights, locations_to_thumb_weights )
def _GetListCtrlLocations( self ):
# current
( locations_to_file_weights, locations_to_thumb_weights ) = self._GetLocationsToCurrentWeights()
#
all_locations = set()
all_locations.update( self._media_base_locations )
if self._ideal_thumbnails_base_location_override is not None:
all_locations.add( self._ideal_thumbnails_base_location_override )
all_locations.update( locations_to_file_weights.keys() )
all_locations.update( locations_to_thumb_weights.keys() )
all_locations = list( all_locations )
return all_locations
def _GetThumbnailSizeEstimates( self ):
( t_width, t_height ) = HG.client_controller.options[ 'thumbnail_dimensions' ]
typical_thumb_num_pixels = 320 * 240
typical_thumb_size = 36 * 1024
our_thumb_num_pixels = t_width * t_height
our_thumb_size_estimate = typical_thumb_size * ( our_thumb_num_pixels / typical_thumb_num_pixels )
our_total_thumb_size_estimate = self._all_local_files_total_num * our_thumb_size_estimate
return ( int( our_total_thumb_size_estimate * 0.8 ), int( our_total_thumb_size_estimate * 1.4 ) )
def _IncreaseWeight( self ):
self._AdjustWeight( 1 )
def _Rebalance( self ):
for base_location in self._GetListCtrlLocations():
if not os.path.exists( base_location.path ):
message = 'The path "{}" does not exist! Please ensure all the locations on this dialog are valid before trying to rebalance your files.'.format( base_location )
ClientGUIDialogsMessage.ShowWarning( self, message )
return
message = 'Moving files can be a slow and slightly laggy process, with the UI intermittently hanging, which sometimes makes manually stopping a large ongoing job difficult. Would you like to set a max runtime on this job?'
yes_tuples = []
yes_tuples.append( ( 'run for 10 minutes', 600 ) )
yes_tuples.append( ( 'run for 30 minutes', 1800 ) )
yes_tuples.append( ( 'run for 1 hour', 3600 ) )
yes_tuples.append( ( 'run indefinitely', None ) )
try:
result = ClientGUIDialogsQuick.GetYesYesNo( self, message, yes_tuples = yes_tuples, no_label = 'forget it' )
except HydrusExceptions.CancelledException:
return
if result is None:
stop_time = None
else:
stop_time = HydrusTime.GetNow() + result
job_status = ClientThreading.JobStatus( cancellable = True, stop_time = stop_time )
HG.client_controller.pub( 'do_file_storage_rebalance', job_status )
self._OKParent()
def _RemoveBaseLocation( self, base_location ):
( locations_to_file_weights, locations_to_thumb_weights ) = self._GetLocationsToCurrentWeights()
removees = set()
if base_location not in self._media_base_locations:
ClientGUIDialogsMessage.ShowWarning( self, 'Please select a location with weight.' )
return
if len( self._media_base_locations ) == 1:
ClientGUIDialogsMessage.ShowWarning( self, 'You cannot empty every single current file location--please add a new place for the files to be moved to and then try again.' )
if os.path.exists( base_location.path ):
if base_location in locations_to_file_weights:
message = 'Are you sure you want to remove this location? This will schedule all of the files it is currently responsible for to be moved elsewhere.'
else:
message = 'Are you sure you want to remove this location?'
else:
if base_location in locations_to_file_weights:
message = 'This path does not exist, but it seems to have files. This could be a critical error that has occurred while the client is open (maybe a drive unmounting?). I recommend you do not remove it here and instead shut the client down immediately and fix the problem, then restart it, which will run the \'recover missing file locations\' routine if needed.'
else:
message = 'This path does not exist. Just checking, but I recommend you remove it.'
result = ClientGUIDialogsQuick.GetYesNo( self, message )
if result == QW.QDialog.Accepted:
self._media_base_locations.remove( base_location )
self._SaveToDB()
def _RemoveSelectedBaseLocation( self ):
locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( locations ) > 0:
location = locations[0]
self._RemoveBaseLocation( location )
def _SaveToDB( self ):
self._controller.Write( 'ideal_client_files_locations', self._media_base_locations, self._ideal_thumbnails_base_location_override )
self._controller.client_files_manager.Reinit()
self._Update()
def _SelectPathToAdd( self ):
with QP.DirDialog( self, 'Select location' ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
base_location = ClientFilesPhysical.FilesStorageBaseLocation( path, 1 )
self._AddBaseLocation( base_location )
def _SetMaxNumBytes( self ):
if not self._CanSetMaxNumBytes():
return
base_locations = self._current_media_base_locations_listctrl.GetData( only_selected = True )
if len( base_locations ) > 0:
base_location = base_locations[0]
if base_location in self._media_base_locations:
max_num_bytes = base_location.max_num_bytes
with ClientGUITopLevelWindowsPanels.DialogEdit( self, 'edit max size' ) as dlg:
message = 'If a location goes over its set size, it will schedule to migrate some files to other locations. At least one media location must have no limit.'
message += '\n' * 2
message += 'This is not precise (it works on average size, and thumbnails can add a bit), so give it some padding. Also, in general, remember it is not healthy to fill any hard drive more than 90% full.'
message += '\n' * 2
message += 'Also, this feature is under active development. The client will go over this limit if your collection grows significantly--the only way things rebalance atm are if you click "move files now"--but in future I will have things automatically migrate in the background to ensure limits are obeyed. This is just for advanced users to play with for now!'
panel = ClientGUIScrolledPanels.EditSingleCtrlPanel( dlg, message = message )
control = ClientGUIControls.NoneableBytesControl( panel, initial_value = 100 * ( 1024 ** 3 ) )
control.SetValue( max_num_bytes )
panel.SetControl( control )
dlg.SetPanel( panel )
if dlg.exec() == QW.QDialog.Accepted:
max_num_bytes = control.GetValue()
base_location.max_num_bytes = max_num_bytes
self._SaveToDB()
def _SetThumbnailLocation( self ):
with QP.DirDialog( self, 'Select thumbnail location' ) as dlg:
if self._ideal_thumbnails_base_location_override is not None:
dlg.setDirectory( self._ideal_thumbnails_base_location_override.path )
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
all_paths = { base_location.path for base_location in self._media_base_locations }
if path in all_paths:
ClientGUIDialogsMessage.ShowWarning( self, 'That path already exists as a regular file location! Please choose another.' )
else:
self._ideal_thumbnails_base_location_override = ClientFilesPhysical.FilesStorageBaseLocation( path, 1 )
self._SaveToDB()
def _Update( self ):
self._client_files_subfolders = HG.client_controller.Read( 'client_files_subfolders' )
( self._media_base_locations, self._ideal_thumbnails_base_location_override ) = self._controller.Read( 'ideal_client_files_locations' )
self._media_base_locations_to_ideal_usage = ClientFilesPhysical.FilesStorageBaseLocation.STATICGetIdealWeights( self._all_local_files_total_size, self._media_base_locations )
approx_total_db_size = self._controller.db.GetApproxTotalFileSize()
self._current_db_path_st.setText( 'database (about '+HydrusData.ToHumanBytes(approx_total_db_size)+'): '+self._controller.GetDBDir() )
self._current_install_path_st.setText( 'install: ' + HC.BASE_DIR )
approx_total_client_files = self._all_local_files_total_size
( approx_total_thumbnails_min, approx_total_thumbnails_max ) = self._GetThumbnailSizeEstimates()
label = 'media is {}, thumbnails are estimated at {}-{}'.format( HydrusData.ToHumanBytes( approx_total_client_files ), HydrusData.ToHumanBytes( approx_total_thumbnails_min ), HydrusData.ToHumanBytes( approx_total_thumbnails_max ) )
self._current_media_paths_st.setText( label )
self._current_media_paths_st.setToolTip( 'Precise thumbnail sizes are not tracked, so this is an estimate based on your current thumbnail dimensions.' )
base_locations = self._GetListCtrlLocations()
self._current_media_base_locations_listctrl.SetData( base_locations )
#
if self._ideal_thumbnails_base_location_override is None:
self._thumbnails_location.setText( 'none set' )
self._thumbnails_location_set.setEnabled( True )
self._thumbnails_location_clear.setEnabled( False )
else:
self._thumbnails_location.setText( self._ideal_thumbnails_base_location_override.path )
self._thumbnails_location_set.setEnabled( False )
self._thumbnails_location_clear.setEnabled( True )
#
if self._controller.client_files_manager.RebalanceWorkToDo():
self._rebalance_button.setEnabled( True )
self._rebalance_status_st.setText( 'files need to be moved' )
self._rebalance_status_st.setObjectName( 'HydrusInvalid' )
else:
self._rebalance_button.setEnabled( False )
self._rebalance_status_st.setText( 'all files are in their ideal locations' )
self._rebalance_status_st.setObjectName( 'HydrusValid' )
self._rebalance_status_st.style().polish( self._rebalance_status_st )
class MigrateTagsPanel( ClientGUIScrolledPanels.ReviewPanel ):
COPY = 0
DELETE = 1
DELETE_DELETED = 2
DELETE_FOR_DELETED_FILES = 3
ALL_MAPPINGS = 0
SPECIFIC_MAPPINGS = 1
SPECIFIC_NAMESPACE = 2
NAMESPACED = 3
UNNAMESPACED = 4
HTA_SERVICE_KEY = b'hydrus tag archive'
HTPA_SERVICE_KEY = b'hydrus tag pair archive'
LOCATION_CONTEXT_LOCATION = 0
HASHES_LOCATION = 1
def __init__( self, parent, service_key, hashes = None ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._service_key = service_key
self._hashes = hashes
self._source_archive_path = None
self._source_archive_hash_type = None
self._dest_archive_path = None
self._dest_archive_hash_type_override = None
try:
service = HG.client_controller.services_manager.GetService( self._service_key )
except HydrusExceptions.DataMissing:
services = HG.client_controller.services_manager.GetServices( ( HC.LOCAL_TAG, ) )
service = next( iter( services ) )
self._service_key = service.GetServiceKey()
#
self._migration_panel = ClientGUICommon.StaticBox( self, 'migration' )
self._migration_content_type = ClientGUICommon.BetterChoice( self._migration_panel )
for content_type in ( HC.CONTENT_TYPE_MAPPINGS, HC.CONTENT_TYPE_TAG_PARENTS, HC.CONTENT_TYPE_TAG_SIBLINGS ):
self._migration_content_type.addItem( HC.content_type_string_lookup[ content_type ], content_type )
self._migration_content_type.setToolTip( 'Sets what will be migrated.' )
self._migration_source = ClientGUICommon.BetterChoice( self._migration_panel )
location_context = HG.client_controller.new_options.GetDefaultLocalLocationContext()
self._migration_source_hash_type_st = ClientGUICommon.BetterStaticText( self._migration_panel, 'hash type: unknown' )
self._migration_source_hash_type_st.setToolTip( 'If this is something other than sha256, this will only work for files the client has ever previously imported.' )
self._migration_source_content_status_filter = ClientGUICommon.BetterChoice( self._migration_panel )
self._migration_source_content_status_filter.setToolTip( 'This filters which status of tags will be migrated.' )
self._migration_source_file_filtering_type = ClientGUICommon.BetterChoice( self._migration_panel )
self._migration_source_file_filtering_type.addItem( 'file domain', self.LOCATION_CONTEXT_LOCATION )
if self._hashes is not None:
self._migration_source_file_filtering_type.addItem( '{} files'.format( HydrusData.ToHumanInt( len( self._hashes ) ) ), self.HASHES_LOCATION )
self._migration_source_file_filtering_type.SetValue( self.HASHES_LOCATION )
self._migration_source_file_filtering_type.setToolTip( 'Choose whether to do this operation for the files you launched the dialog on, or a whole file domain.' )
self._migration_source_location_context_button = ClientGUILocation.LocationSearchContextButton( self._migration_panel, location_context )
self._migration_source_location_context_button.setToolTip( 'Set which files should be eligible. This can get as complicated as you like!' )
self._migration_source_archive_path_button = ClientGUICommon.BetterButton( self._migration_panel, 'no path set', self._SetSourceArchivePath )
message = 'Tags that pass this filter will be applied to the destination with the chosen action.'
message += os.linesep * 2
message += 'For instance, if you whitelist the \'series\' namespace, only series: tags from the source will be added to/deleted from the destination.'
tag_filter = HydrusTags.TagFilter()
self._migration_source_tag_filter = ClientGUITags.TagFilterButton( self._migration_panel, message, tag_filter, label_prefix = 'tags taken: ' )
message = 'The left side of a tag sibling/parent pair must pass this filter for the pair to be included in the migration.'
message += os.linesep * 2
message += 'For instance, if you whitelist the \'character\' namespace, only pairs from the source with character: tags on the left will be added to/deleted from the destination.'
tag_filter = HydrusTags.TagFilter()
self._migration_source_left_tag_pair_filter = ClientGUITags.TagFilterButton( self._migration_panel, message, tag_filter, label_prefix = 'left: ' )
message = 'The right side of a tag sibling/parent pair must pass this filter for the pair to be included in the migration.'
message += os.linesep * 2
message += 'For instance, if you whitelist the \'series\' namespace, only pairs from the source with series: tags on the right will be added to/deleted from the destination.'
tag_filter = HydrusTags.TagFilter()
self._migration_source_right_tag_pair_filter = ClientGUITags.TagFilterButton( self._migration_panel, message, tag_filter, label_prefix = 'right: ' )
self._migration_destination = ClientGUICommon.BetterChoice( self._migration_panel )
self._migration_destination_archive_path_button = ClientGUICommon.BetterButton( self._migration_panel, 'no path set', self._SetDestinationArchivePath )
self._migration_destination_hash_type_choice_st = ClientGUICommon.BetterStaticText( self._migration_panel, 'hash type: ' )
self._migration_destination_hash_type_choice = ClientGUICommon.BetterChoice( self._migration_panel )
for hash_type in ( 'sha256', 'md5', 'sha1', 'sha512' ):
self._migration_destination_hash_type_choice.addItem( hash_type, hash_type )
self._migration_destination_hash_type_choice.setToolTip( 'If you set something other than sha256, this will only work for files the client has ever previously imported.' )
self._migration_action = ClientGUICommon.BetterChoice( self._migration_panel )
self._migration_go = ClientGUICommon.BetterButton( self._migration_panel, 'Go!', self._MigrationGo )
#
file_left_vbox = QP.VBoxLayout()
QP.AddToLayout( file_left_vbox, self._migration_source_file_filtering_type, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( file_left_vbox, self._migration_source_location_context_button, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( file_left_vbox, self._migration_source_left_tag_pair_filter, CC.FLAGS_EXPAND_BOTH_WAYS )
tag_right_vbox = QP.VBoxLayout()
QP.AddToLayout( tag_right_vbox, self._migration_source_tag_filter, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( tag_right_vbox, self._migration_source_right_tag_pair_filter, CC.FLAGS_EXPAND_BOTH_WAYS )
dest_hash_type_hbox = QP.HBoxLayout()
QP.AddToLayout( dest_hash_type_hbox, self._migration_destination_hash_type_choice_st, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( dest_hash_type_hbox, self._migration_destination_hash_type_choice, CC.FLAGS_EXPAND_BOTH_WAYS )
#
gridbox = QP.GridLayout( 6 )
gridbox.setColumnStretch( 0, 1 )
gridbox.setColumnStretch( 1, 1 )
gridbox.setColumnStretch( 2, 1 )
gridbox.setColumnStretch( 3, 1 )
QP.AddToLayout( gridbox, ClientGUICommon.BetterStaticText( self._migration_panel, 'content' ), CC.FLAGS_CENTER )
QP.AddToLayout( gridbox, ClientGUICommon.BetterStaticText( self._migration_panel, 'source' ), CC.FLAGS_CENTER )
QP.AddToLayout( gridbox, ClientGUICommon.BetterStaticText( self._migration_panel, 'filter' ), CC.FLAGS_CENTER )
QP.AddToLayout( gridbox, ClientGUICommon.BetterStaticText( self._migration_panel, 'action' ), CC.FLAGS_CENTER )
QP.AddToLayout( gridbox, ClientGUICommon.BetterStaticText( self._migration_panel, 'destination' ), CC.FLAGS_CENTER )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
QP.AddToLayout( gridbox, self._migration_content_type, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, self._migration_source, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, self._migration_source_content_status_filter, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, self._migration_action, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, self._migration_destination, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, self._migration_go, CC.FLAGS_EXPAND_BOTH_WAYS )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
QP.AddToLayout( gridbox, self._migration_source_archive_path_button, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( gridbox, file_left_vbox, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
QP.AddToLayout( gridbox, self._migration_destination_archive_path_button, CC.FLAGS_EXPAND_BOTH_WAYS )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
QP.AddToLayout( gridbox, self._migration_source_hash_type_st, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( gridbox, tag_right_vbox, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
QP.AddToLayout( gridbox, dest_hash_type_hbox, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
ClientGUICommon.AddGridboxStretchSpacer( self._migration_panel, gridbox )
self._migration_panel.Add( gridbox )
#
message = 'The content from the SOURCE that the FILTER ALLOWS is applied using the ACTION to the DESTINATION.'
message += os.linesep * 2
message += 'To delete content en masse from one location, select what you want to delete with the filter and set the source and destination the same.'
message += os.linesep * 2
message += 'These migrations can be powerful, so be very careful that you understand what you are doing and choose what you want. Large jobs may have a significant initial setup time, during which case the client may hang briefly, but once they start they are pausable or cancellable. If you do want to perform a large action, it is a good idea to back up your database first, just in case you get a result you did not intend.'
message += os.linesep * 2
message += 'You may need to restart your client to see their effect.'
st = ClientGUICommon.BetterStaticText( self, message )
st.setWordWrap( True )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, st, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( vbox, self._migration_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
#vbox.addStretch( 1 )
self.widget().setLayout( vbox )
#
self._migration_content_type.SetValue( HC.CONTENT_TYPE_MAPPINGS )
self._UpdateMigrationControlsNewType()
self._migration_content_type.activated.connect( self._UpdateMigrationControlsNewType )
self._migration_source.activated.connect( self._UpdateMigrationControlsNewSource )
self._migration_destination.activated.connect( self._UpdateMigrationControlsNewDestination )
self._migration_source_content_status_filter.activated.connect( self._UpdateMigrationControlsActions )
self._migration_source_file_filtering_type.activated.connect( self._UpdateMigrationControlsFileFilter )
def _MigrationGo( self ):
extra_info = ''
source_content_statuses_strings = {
( HC.CONTENT_STATUS_CURRENT, ) : 'current',
( HC.CONTENT_STATUS_CURRENT, HC.CONTENT_STATUS_PENDING ) : 'current and pending',
( HC.CONTENT_STATUS_PENDING, ) : 'pending',
( HC.CONTENT_STATUS_DELETED, ) : 'deleted'
}
destination_action_strings = {
HC.CONTENT_UPDATE_ADD : 'adding them to',
HC.CONTENT_UPDATE_DELETE : 'deleting them from',
HC.CONTENT_UPDATE_CLEAR_DELETE_RECORD : 'clearing their deletion record from',
HC.CONTENT_UPDATE_PEND : 'pending them to',
HC.CONTENT_UPDATE_PETITION : 'petitioning them for removal from'
}
content_type = self._migration_content_type.GetValue()
content_statuses = self._migration_source_content_status_filter.GetValue()
destination_service_key = self._migration_destination.GetValue()
source_service_key = self._migration_source.GetValue()
if content_type == HC.CONTENT_TYPE_MAPPINGS:
if destination_service_key == self.HTA_SERVICE_KEY:
if self._dest_archive_path is None:
ClientGUIDialogsMessage.ShowWarning( self, 'Please set a path for the destination Hydrus Tag Archive.' )
return
content_action = HC.CONTENT_UPDATE_ADD
desired_hash_type = self._migration_destination_hash_type_choice.GetValue()
destination = ClientMigration.MigrationDestinationHTA( HG.client_controller, self._dest_archive_path, desired_hash_type )
else:
content_action = self._migration_action.GetValue()
desired_hash_type = 'sha256'
destination = ClientMigration.MigrationDestinationTagServiceMappings( HG.client_controller, destination_service_key, content_action )
file_filtering_type = self._migration_source_file_filtering_type.GetValue()
if file_filtering_type == self.HASHES_LOCATION:
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_FILE_SERVICE_KEY )
hashes = self._hashes
extra_info = ' for {} files'.format( HydrusData.ToHumanInt( len( hashes ) ) )
else:
location_context = self._migration_source_location_context_button.GetValue()
hashes = None
if location_context.IsAllKnownFiles():
extra_info = ' for all known files'
else:
extra_info = ' for files in "{}"'.format( location_context.ToString( HG.client_controller.services_manager.GetName ) )
tag_filter = self._migration_source_tag_filter.GetValue()
extra_info += ' and for tags "{}"'.format( HydrusText.ElideText( tag_filter.ToPermittedString(), 96 ) )
if source_service_key == self.HTA_SERVICE_KEY:
if self._source_archive_path is None:
ClientGUIDialogsMessage.ShowWarning( self, 'Please set a path for the source Hydrus Tag Archive.' )
return
source = ClientMigration.MigrationSourceHTA( HG.client_controller, self._source_archive_path, location_context, desired_hash_type, hashes, tag_filter )
else:
source = ClientMigration.MigrationSourceTagServiceMappings( HG.client_controller, source_service_key, location_context, desired_hash_type, hashes, tag_filter, content_statuses )
else:
if destination_service_key == self.HTPA_SERVICE_KEY:
if self._dest_archive_path is None:
ClientGUIDialogsMessage.ShowWarning( self, 'Please set a path for the destination Hydrus Tag Pair Archive.' )
return
content_action = HC.CONTENT_UPDATE_ADD
destination = ClientMigration.MigrationDestinationHTPA( HG.client_controller, self._dest_archive_path, content_type )
else:
content_action = self._migration_action.GetValue()
destination = ClientMigration.MigrationDestinationTagServicePairs( HG.client_controller, destination_service_key, content_action, content_type )
left_tag_pair_filter = self._migration_source_left_tag_pair_filter.GetValue()
right_tag_pair_filter = self._migration_source_right_tag_pair_filter.GetValue()
left_s = left_tag_pair_filter.ToPermittedString()
right_s = right_tag_pair_filter.ToPermittedString()
if left_s == right_s:
extra_info = ' for "{}" on both sides'.format( left_s )
else:
extra_info = ' for "{}" on the left and "{}" on the right'.format( left_s, right_s )
if source_service_key == self.HTPA_SERVICE_KEY:
if self._source_archive_path is None:
ClientGUIDialogsMessage.ShowWarning( self, 'Please set a path for the source Hydrus Tag Archive.' )
return
source = ClientMigration.MigrationSourceHTPA( HG.client_controller, self._source_archive_path, left_tag_pair_filter, right_tag_pair_filter )
else:
source = ClientMigration.MigrationSourceTagServicePairs( HG.client_controller, source_service_key, content_type, left_tag_pair_filter, right_tag_pair_filter, content_statuses )
title = 'taking {} {}{} from "{}" and {} "{}"'.format( source_content_statuses_strings[ content_statuses ], HC.content_type_string_lookup[ content_type ], extra_info, source.GetName(), destination_action_strings[ content_action ], destination.GetName() )
message = 'Migrations can make huge changes. They can be cancelled early, but any work they do cannot always be undone. Please check that this summary looks correct:'
message += os.linesep * 2
message += title
message += os.linesep * 2
message += 'If you plan to make a very big change (especially a mass delete), I recommend making a backup of your database before going ahead, just in case something unexpected happens.'
result = ClientGUIDialogsQuick.GetYesNo( self, message, yes_label = 'looks good', no_label = 'go back' )
if result == QW.QDialog.Accepted:
if HG.client_controller.new_options.GetBoolean( 'advanced_mode' ):
do_it = True
else:
message = 'Are you absolutely sure you set up the filters and everything how you wanted? Last chance to turn back.'
result = ClientGUIDialogsQuick.GetYesNo( self, message )
do_it = result == QW.QDialog.Accepted
if do_it:
migration_job = ClientMigration.MigrationJob( HG.client_controller, title, source, destination )
HG.client_controller.CallToThread( migration_job.Run )
def _SetDestinationArchivePath( self ):
message = 'Select the destination location for the Archive. Existing Archives are also ok, and will be appended to.'
with QP.FileDialog( self, message = message, acceptMode = QW.QFileDialog.AcceptSave, default_filename = 'archive.db', fileMode = QW.QFileDialog.AnyFile ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
if os.path.exists( path ):
content_type = self._migration_content_type.GetValue()
if content_type == HC.CONTENT_TYPE_MAPPINGS:
try:
hta = HydrusTagArchive.HydrusTagArchive( path )
except Exception as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem with HTA!', f'Could not load that path as an Archive!\n\n{e}' )
try:
hash_type = hta.GetHashType()
self._dest_archive_hash_type_override = hash_type
self._migration_destination_hash_type_choice.SetValue( hash_type )
self._migration_destination_hash_type_choice.setEnabled( False )
except:
pass
else:
try:
hta = HydrusTagArchive.HydrusTagPairArchive( path )
pair_type = hta.GetPairType()
except Exception as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem!', f'Could not load that file as a Hydrus Tag Pair Archive! {e}' )
return
if content_type == HC.CONTENT_TYPE_TAG_PARENTS and pair_type != HydrusTagArchive.TAG_PAIR_TYPE_PARENTS:
ClientGUIDialogsMessage.ShowWarning( self, 'This Hydrus Tag Pair Archive is not a tag parents archive!' )
return
elif content_type == HC.CONTENT_TYPE_TAG_SIBLINGS and pair_type != HydrusTagArchive.TAG_PAIR_TYPE_SIBLINGS:
ClientGUIDialogsMessage.ShowWarning( self, 'This Hydrus Tag Pair Archive is not a tag siblings archive!' )
return
else:
self._migration_destination_hash_type_choice.setEnabled( True )
self._dest_archive_path = path
filename = os.path.basename( self._dest_archive_path )
self._migration_destination_archive_path_button.setText( filename )
self._migration_destination_archive_path_button.setToolTip( path )
def _SetSourceArchivePath( self ):
message = 'Select the Archive to pull data from.'
with QP.FileDialog( self, message = message, acceptMode = QW.QFileDialog.AcceptOpen ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
content_type = self._migration_content_type.GetValue()
if content_type == HC.CONTENT_TYPE_MAPPINGS:
try:
hta = HydrusTagArchive.HydrusTagArchive( path )
hash_type = hta.GetHashType()
except Exception as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem!', f'Could not load that file as a Hydrus Tag Archive! {e}' )
return
hash_type_str = HydrusTagArchive.hash_type_to_str_lookup[ hash_type ]
self._migration_source_hash_type_st.setText( 'hash type: {}'.format(hash_type_str) )
else:
try:
hta = HydrusTagArchive.HydrusTagPairArchive( path )
pair_type = hta.GetPairType()
except Exception as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem!', f'Could not load that file as a Hydrus Tag Pair Archive! {e}' )
return
if pair_type is None:
ClientGUIDialogsMessage.ShowWarning( self, 'This Hydrus Tag Pair Archive does not have a pair type set!' )
return
if content_type == HC.CONTENT_TYPE_TAG_PARENTS and pair_type != HydrusTagArchive.TAG_PAIR_TYPE_PARENTS:
ClientGUIDialogsMessage.ShowWarning( self, 'This Hydrus Tag Pair Archive is not a tag parents archive!' )
return
elif content_type == HC.CONTENT_TYPE_TAG_SIBLINGS and pair_type != HydrusTagArchive.TAG_PAIR_TYPE_SIBLINGS:
ClientGUIDialogsMessage.ShowWarning( self, 'This Hydrus Tag Pair Archive is not a tag siblings archive!' )
return
self._source_archive_path = path
filename = os.path.basename( self._source_archive_path )
self._migration_source_archive_path_button.setText( filename )
self._migration_source_archive_path_button.setToolTip( path )
def _UpdateMigrationControlsActions( self ):
content_type = self._migration_content_type.GetValue()
self._migration_action.clear()
source = self._migration_source.GetValue()
destination = self._migration_destination.GetValue()
tag_status = self._migration_source_content_status_filter.GetValue()
source_and_destination_the_same = source == destination
pulling_and_pushing_existing = source_and_destination_the_same and HC.CONTENT_STATUS_CURRENT in tag_status
pulling_and_pushing_deleted = source_and_destination_the_same and HC.CONTENT_STATUS_DELETED in tag_status
actions = []
if destination in ( self.HTA_SERVICE_KEY, self.HTPA_SERVICE_KEY ):
actions.append( HC.CONTENT_UPDATE_ADD )
else:
destination_service = HG.client_controller.services_manager.GetService( destination )
destination_service_type = destination_service.GetServiceType()
if destination_service_type == HC.LOCAL_TAG:
if not pulling_and_pushing_existing:
actions.append( HC.CONTENT_UPDATE_ADD )
if not pulling_and_pushing_deleted:
actions.append( HC.CONTENT_UPDATE_DELETE )
if not pulling_and_pushing_existing and content_type == HC.CONTENT_TYPE_MAPPINGS:
actions.append( HC.CONTENT_UPDATE_CLEAR_DELETE_RECORD )
elif destination_service_type == HC.TAG_REPOSITORY:
if not pulling_and_pushing_existing:
actions.append( HC.CONTENT_UPDATE_PEND )
if not pulling_and_pushing_deleted:
actions.append( HC.CONTENT_UPDATE_PETITION )
for action in actions:
self._migration_action.addItem( HC.content_update_string_lookup[ action ], action )
if len( actions ) > 1:
self._migration_action.setEnabled( True )
else:
self._migration_action.setEnabled( False )
def _UpdateMigrationControlsFileFilter( self ):
self._migration_source_file_filtering_type.setVisible( self._migration_source_file_filtering_type.count() > 1 )
self._migration_source_location_context_button.setVisible( self._migration_source_file_filtering_type.GetValue() == self.LOCATION_CONTEXT_LOCATION )
def _UpdateMigrationControlsNewDestination( self ):
destination = self._migration_destination.GetValue()
if destination in ( self.HTA_SERVICE_KEY, self.HTPA_SERVICE_KEY ):
self._migration_destination_archive_path_button.show()
if destination == self.HTA_SERVICE_KEY:
self._migration_destination_hash_type_choice_st.show()
self._migration_destination_hash_type_choice.show()
else:
self._migration_destination_hash_type_choice_st.hide()
self._migration_destination_hash_type_choice.hide()
else:
self._migration_destination_archive_path_button.hide()
self._migration_destination_hash_type_choice_st.hide()
self._migration_destination_hash_type_choice.hide()
self._UpdateMigrationControlsActions()
def _UpdateMigrationControlsNewSource( self ):
source = self._migration_source.GetValue()
self._migration_source_content_status_filter.clear()
if source in ( self.HTA_SERVICE_KEY, self.HTPA_SERVICE_KEY ):
self._migration_source_archive_path_button.show()
if source == self.HTA_SERVICE_KEY:
self._migration_source_hash_type_st.show()
else:
self._migration_source_hash_type_st.hide()
self._migration_source_content_status_filter.addItem( 'current', ( HC.CONTENT_STATUS_CURRENT, ) )
self._migration_source_content_status_filter.setEnabled( False )
else:
self._migration_source_archive_path_button.hide()
self._migration_source_hash_type_st.hide()
source_service = HG.client_controller.services_manager.GetService( source )
source_service_type = source_service.GetServiceType()
self._migration_source_content_status_filter.addItem( 'current', ( HC.CONTENT_STATUS_CURRENT, ) )
if source_service_type == HC.TAG_REPOSITORY:
self._migration_source_content_status_filter.addItem( 'current and pending', ( HC.CONTENT_STATUS_CURRENT, HC.CONTENT_STATUS_PENDING ) )
self._migration_source_content_status_filter.addItem( 'pending', ( HC.CONTENT_STATUS_PENDING, ) )
self._migration_source_content_status_filter.addItem( 'deleted', ( HC.CONTENT_STATUS_DELETED, ) )
self._migration_source_content_status_filter.setEnabled( True )
self._UpdateMigrationControlsActions()
def _UpdateMigrationControlsNewType( self ):
content_type = self._migration_content_type.GetValue()
self._migration_source.clear()
self._migration_destination.clear()
for service in HG.client_controller.services_manager.GetServices( HC.REAL_TAG_SERVICES ):
self._migration_source.addItem( service.GetName(), service.GetServiceKey() )
self._migration_destination.addItem( service.GetName(), service.GetServiceKey() )
self._source_archive_path = None
self._source_archive_hash_type = None
self._dest_archive_path = None
self._dest_archive_hash_type_override = None
self._migration_source_hash_type_st.setText( 'hash type: unknown' )
self._migration_destination_hash_type_choice.SetValue( 'sha256' )
self._migration_destination_hash_type_choice.setEnabled( True )
self._migration_source_archive_path_button.setText( 'no path set' )
self._migration_source_archive_path_button.setToolTip( '' )
self._migration_destination_archive_path_button.setText( 'no path set' )
self._migration_destination_archive_path_button.setToolTip( '' )
if content_type == HC.CONTENT_TYPE_MAPPINGS:
self._migration_source.addItem( 'Hydrus Tag Archive', self.HTA_SERVICE_KEY )
self._migration_destination.addItem( 'Hydrus Tag Archive', self.HTA_SERVICE_KEY )
self._migration_source_tag_filter.show()
self._migration_source_left_tag_pair_filter.hide()
self._migration_source_right_tag_pair_filter.hide()
self._UpdateMigrationControlsFileFilter()
elif content_type in ( HC.CONTENT_TYPE_TAG_SIBLINGS, HC.CONTENT_TYPE_TAG_PARENTS ):
self._migration_source.addItem( 'Hydrus Tag Pair Archive', self.HTPA_SERVICE_KEY )
self._migration_destination.addItem( 'Hydrus Tag Pair Archive', self.HTPA_SERVICE_KEY )
self._migration_source_file_filtering_type.hide()
self._migration_source_location_context_button.hide()
self._migration_source_tag_filter.hide()
self._migration_source_left_tag_pair_filter.show()
self._migration_source_right_tag_pair_filter.show()
self._migration_source.SetValue( self._service_key )
self._migration_destination.SetValue( self._service_key )
self._UpdateMigrationControlsNewSource()
self._UpdateMigrationControlsNewDestination()
class ReviewDownloaderImport( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, network_engine ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._network_engine = network_engine
vbox = QP.VBoxLayout()
menu_items = []
page_func = HydrusData.Call( ClientGUIDialogsQuick.OpenDocumentation, self, HC.DOCUMENTATION_ADDING_NEW_DOWNLOADERS )
menu_items.append( ( 'normal', 'open the easy downloader import help', 'Open the help page for easily importing downloaders in your web browser.', page_func ) )
help_button = ClientGUIMenuButton.MenuBitmapButton( self, CC.global_pixmaps().help, menu_items )
help_hbox = ClientGUICommon.WrapInText( help_button, self, 'help -->', object_name = 'HydrusIndeterminate' )
self._repo_link = ClientGUICommon.BetterHyperLink( self, 'get user-made downloaders here', 'https://github.com/CuddleBear92/Hydrus-Presets-and-Scripts/tree/master/Downloaders' )
self._paste_button = ClientGUICommon.BetterBitmapButton( self, CC.global_pixmaps().paste, self._Paste )
self._paste_button.setToolTip( 'Or you can paste bitmaps from clipboard!' )
st = ClientGUICommon.BetterStaticText( self, label = 'Drop downloader-encoded pngs onto Lain to import.' )
lain_path = os.path.join( HC.STATIC_DIR, 'lain.jpg' )
lain_qt_pixmap = ClientRendering.GenerateHydrusBitmap( lain_path, HC.IMAGE_JPEG ).GetQtPixmap()
win = ClientGUICommon.BufferedWindowIcon( self, lain_qt_pixmap )
win.setCursor( QG.QCursor( QC.Qt.PointingHandCursor ) )
self._select_from_list = QW.QCheckBox( self )
if HG.client_controller.new_options.GetBoolean( 'advanced_mode' ):
self._select_from_list.setChecked( True )
QP.AddToLayout( vbox, help_hbox, CC.FLAGS_ON_RIGHT )
QP.AddToLayout( vbox, self._repo_link, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, st, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, self._paste_button, CC.FLAGS_ON_RIGHT )
QP.AddToLayout( vbox, win, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, ClientGUICommon.WrapInText( self._select_from_list, self, 'select objects from list' ), CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
self.widget().setLayout( vbox )
#
win.installEventFilter( ClientGUIDragDrop.FileDropTarget( win, filenames_callable = self.ImportFromDragDrop ) )
self._lain_event_filter = QP.WidgetEventFilter( win )
self._lain_event_filter.EVT_LEFT_DOWN( self.EventLainClick )
def _ImportPaths( self, paths ):
payloads = []
for path in paths:
try:
payload = ClientSerialisable.LoadFromPNG( path )
except Exception as e:
HydrusData.PrintException( e )
ClientGUIDialogsMessage.ShowCritical( self, 'Problem loading PNG!', str(e) )
return
payloads.append( ( 'file "{}"'.format( path ), payload ) )
self._ImportPayloads( payloads )
def _ImportPayloads( self, payloads ):
have_shown_load_error = False
gugs = []
url_classes = []
parsers = []
domain_metadatas = []
login_scripts = []
num_misc_objects = 0
bandwidth_manager = self._network_engine.bandwidth_manager
domain_manager = self._network_engine.domain_manager
login_manager = self._network_engine.login_manager
for ( payload_description, payload ) in payloads:
try:
obj_list = HydrusSerialisable.CreateFromNetworkBytes( payload, raise_error_on_future_version = True )
except HydrusExceptions.SerialisationException as e:
if not have_shown_load_error:
message = str( e )
if len( payloads ) > 1:
message += os.linesep * 2
message += 'If there are more unloadable objects in this import, they will be skipped silently.'
ClientGUIDialogsMessage.ShowCritical( self, 'Serialisation Object Load Error!', message )
have_shown_load_error = True
continue
except Exception as e:
HydrusData.PrintException( e )
ClientGUIDialogsMessage.ShowCritical( self, 'Error', 'I could not understand what was encoded in {}!'.format( payload_description ) )
continue
if isinstance( obj_list, ( ClientNetworkingGUG.GalleryURLGenerator, ClientNetworkingGUG.NestedGalleryURLGenerator, ClientNetworkingURLClass.URLClass, ClientParsing.PageParser, ClientNetworkingDomain.DomainMetadataPackage, ClientNetworkingLogin.LoginScriptDomain ) ):
obj_list = HydrusSerialisable.SerialisableList( [ obj_list ] )
if not isinstance( obj_list, HydrusSerialisable.SerialisableList ):
ClientGUIDialogsMessage.ShowWarning( self, 'Unfortunately, {} did not look like a package of download data! Instead, it looked like: {}'.format( payload_description, obj_list.SERIALISABLE_NAME ) )
continue
for obj in obj_list:
if isinstance( obj, ( ClientNetworkingGUG.GalleryURLGenerator, ClientNetworkingGUG.NestedGalleryURLGenerator ) ):
gugs.append( obj )
elif isinstance( obj, ClientNetworkingURLClass.URLClass ):
url_classes.append( obj )
elif isinstance( obj, ClientParsing.PageParser ):
parsers.append( obj )
elif isinstance( obj, ClientNetworkingDomain.DomainMetadataPackage ):
domain_metadatas.append( obj )
elif isinstance( obj, ClientNetworkingLogin.LoginScriptDomain ):
login_scripts.append( obj )
else:
num_misc_objects += 1
if len( gugs ) + len( url_classes ) + len( parsers ) + len( domain_metadatas ) + len( login_scripts ) == 0:
if num_misc_objects > 0:
ClientGUIDialogsMessage.ShowWarning( self, 'I found '+HydrusData.ToHumanInt(num_misc_objects)+' misc objects in that png, but nothing downloader related.' )
return
# url matches first
url_class_names_seen = set()
dupe_url_classes = []
num_exact_dupe_url_classes = 0
new_url_classes = []
for url_class in url_classes:
if url_class.GetName() in url_class_names_seen:
continue
if domain_manager.AlreadyHaveExactlyThisURLClass( url_class ):
dupe_url_classes.append( url_class )
num_exact_dupe_url_classes += 1
else:
new_url_classes.append( url_class )
url_class_names_seen.add( url_class.GetName() )
# now gugs
gug_names_seen = set()
num_exact_dupe_gugs = 0
new_gugs = []
for gug in gugs:
if gug.GetName() in gug_names_seen:
continue
if domain_manager.AlreadyHaveExactlyThisGUG( gug ):
num_exact_dupe_gugs += 1
else:
new_gugs.append( gug )
gug_names_seen.add( gug.GetName() )
# now parsers
parser_names_seen = set()
num_exact_dupe_parsers = 0
new_parsers = []
for parser in parsers:
if parser.GetName() in parser_names_seen:
continue
if domain_manager.AlreadyHaveExactlyThisParser( parser ):
num_exact_dupe_parsers += 1
else:
new_parsers.append( parser )
parser_names_seen.add( parser.GetName() )
# now login scripts
login_script_names_seen = set()
num_exact_dupe_login_scripts = 0
new_login_scripts = []
for login_script in login_scripts:
if login_script.GetName() in login_script_names_seen:
continue
if login_manager.AlreadyHaveExactlyThisLoginScript( login_script ):
num_exact_dupe_login_scripts += 1
else:
new_login_scripts.append( login_script )
login_script_names_seen.add( login_script.GetName() )
# now domain metadata
domains_seen = set()
num_exact_dupe_domain_metadatas = 0
new_domain_metadatas = []
for domain_metadata in domain_metadatas:
# check if the headers or rules are new, discarding any dupe content
domain = domain_metadata.GetDomain()
headers_list = None
bandwidth_rules = None
if domain in domains_seen:
continue
nc = ClientNetworkingContexts.NetworkContext( CC.NETWORK_CONTEXT_DOMAIN, domain )
if domain_metadata.HasHeaders():
headers_list = domain_metadata.GetHeaders()
if domain_manager.AlreadyHaveExactlyTheseHeaders( nc, headers_list ):
headers_list = None
if domain_metadata.HasBandwidthRules():
bandwidth_rules = domain_metadata.GetBandwidthRules()
if bandwidth_manager.AlreadyHaveExactlyTheseBandwidthRules( nc, bandwidth_rules ):
bandwidth_rules = None
if headers_list is None and bandwidth_rules is None:
num_exact_dupe_domain_metadatas += 1
else:
new_dm = ClientNetworkingDomain.DomainMetadataPackage( domain = domain, headers_list = headers_list, bandwidth_rules = bandwidth_rules )
new_domain_metadatas.append( new_dm )
domains_seen.add( domain )
#
total_num_dupes = num_exact_dupe_gugs + num_exact_dupe_url_classes + num_exact_dupe_parsers + num_exact_dupe_domain_metadatas + num_exact_dupe_login_scripts
if len( new_gugs ) + len( new_url_classes ) + len( new_parsers ) + len( new_domain_metadatas ) + len( new_login_scripts ) == 0:
ClientGUIDialogsMessage.ShowInformation( self, f'All {HydrusData.ToHumanInt( total_num_dupes )} downloader objects in that package appeared to already be in the client, so nothing need be added.' )
return
# let's start selecting what we want
if self._select_from_list.isChecked():
choice_tuples = []
choice_tuples.extend( [ ( 'GUG: ' + gug.GetName(), gug, True ) for gug in new_gugs ] )
choice_tuples.extend( [ ( 'URL Class: ' + url_class.GetName(), url_class, True ) for url_class in new_url_classes ] )
choice_tuples.extend( [ ( 'Parser: ' + parser.GetName(), parser, True ) for parser in new_parsers ] )
choice_tuples.extend( [ ( 'Login Script: ' + login_script.GetName(), login_script, True ) for login_script in new_login_scripts ] )
choice_tuples.extend( [ ( 'Domain Metadata: ' + domain_metadata.GetDomain(), domain_metadata, True ) for domain_metadata in new_domain_metadatas ] )
try:
new_objects = ClientGUIDialogsQuick.SelectMultipleFromList( self, 'select objects to add', choice_tuples )
except HydrusExceptions.CancelledException:
return
new_gugs = [ obj for obj in new_objects if isinstance( obj, ( ClientNetworkingGUG.GalleryURLGenerator, ClientNetworkingGUG.NestedGalleryURLGenerator ) ) ]
new_url_classes = [ obj for obj in new_objects if isinstance( obj, ClientNetworkingURLClass.URLClass ) ]
new_parsers = [ obj for obj in new_objects if isinstance( obj, ClientParsing.PageParser ) ]
new_login_scripts = [ obj for obj in new_objects if isinstance( obj, ClientNetworkingLogin.LoginScriptDomain ) ]
new_domain_metadatas = [ obj for obj in new_objects if isinstance( obj, ClientNetworkingDomain.DomainMetadataPackage ) ]
# final ask
new_gugs.sort( key = lambda o: o.GetName() )
new_url_classes.sort( key = lambda o: o.GetName() )
new_parsers.sort( key = lambda o: o.GetName() )
new_login_scripts.sort( key = lambda o: o.GetName() )
new_domain_metadatas.sort( key = lambda o: o.GetDomain() )
if len( new_domain_metadatas ) > 0:
message = 'Before the final import confirmation, I will now show the domain metadata in detail. Give it a cursory look just to check it seems good. If not, click no on the final dialog!'
TOO_MANY_DM = 8
if len( new_domain_metadatas ) > TOO_MANY_DM:
message += os.linesep * 2
message += 'There are more than ' + HydrusData.ToHumanInt( TOO_MANY_DM ) + ' domain metadata objects. So I do not give you dozens of preview windows, I will only show you these first ' + HydrusData.ToHumanInt( TOO_MANY_DM ) + '.'
new_domain_metadatas_to_show = new_domain_metadatas[:TOO_MANY_DM]
for new_dm in new_domain_metadatas_to_show:
ClientGUIDialogsMessage.ShowInformation( self, new_dm.GetDetailedSafeSummary() )
all_to_add = list( new_gugs )
all_to_add.extend( new_url_classes )
all_to_add.extend( new_parsers )
all_to_add.extend( new_login_scripts )
all_to_add.extend( new_domain_metadatas )
message = 'The client is about to add and link these objects:'
message += os.linesep * 2
message += os.linesep.join( ( obj.GetSafeSummary() for obj in all_to_add[:20] ) )
if len( all_to_add ) > 20:
message += os.linesep
message += '(and ' + HydrusData.ToHumanInt( len( all_to_add ) - 20 ) + ' others)'
message += os.linesep * 2
message += 'Does that sound good?'
result = ClientGUIDialogsQuick.GetYesNo( self, message )
if result != QW.QDialog.Accepted:
return
# ok, do it
if len( new_gugs ) > 0:
for gug in new_gugs:
gug.RegenerateGUGKey()
domain_manager.AddGUGs( new_gugs )
domain_manager.AutoAddURLClassesAndParsers( new_url_classes, dupe_url_classes, new_parsers )
bandwidth_manager.AutoAddDomainMetadatas( new_domain_metadatas )
domain_manager.AutoAddDomainMetadatas( new_domain_metadatas, approved = ClientNetworkingDomain.VALID_APPROVED )
login_manager.AutoAddLoginScripts( new_login_scripts )
num_new_gugs = len( new_gugs )
num_aux = len( new_url_classes ) + len( new_parsers ) + len( new_login_scripts ) + len( new_domain_metadatas )
final_message = 'Successfully added ' + HydrusData.ToHumanInt( num_new_gugs ) + ' new downloaders and ' + HydrusData.ToHumanInt( num_aux ) + ' auxiliary objects.'
if total_num_dupes > 0:
final_message += ' ' + HydrusData.ToHumanInt( total_num_dupes ) + ' duplicate objects were not added (but some additional metadata may have been merged).'
ClientGUIDialogsMessage.ShowInformation( self, final_message )
def _Paste( self ):
if HG.client_controller.ClipboardHasImage():
try:
qt_image = HG.client_controller.GetClipboardImage()
payload_description = 'clipboard image data'
payload = ClientSerialisable.LoadFromQtImage( qt_image )
except Exception as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem pasting image!', f'Sorry, seemed to be a problem: {e}' )
return
else:
try:
raw_text = HG.client_controller.GetClipboardText()
payload_description = 'clipboard text data'
payload = HydrusCompression.CompressStringToBytes( raw_text )
except HydrusExceptions.DataMissing as e:
ClientGUIDialogsMessage.ShowCritical( self, 'Problem pasting text!', f'Sorry, seemed to be a problem: {e}' )
return
payloads = [ ( payload_description, payload ) ]
self._ImportPayloads( payloads )
def EventLainClick( self, event ):
with QP.FileDialog( self, 'Select the pngs to add.', acceptMode = QW.QFileDialog.AcceptOpen, fileMode = QW.QFileDialog.ExistingFiles ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
paths = dlg.GetPaths()
self._ImportPaths( paths )
def ImportFromDragDrop( self, paths ):
self._ImportPaths( paths )
class ReviewFileEmbeddedMetadata( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, exif_dict: typing.Optional[ dict ], file_text: typing.Optional[ str ], extra_rows: typing.List[ typing.Tuple[ str, str ] ] ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
#
exif_panel = ClientGUICommon.StaticBox( self, 'EXIF' )
label = 'Double-click a row to copy its value to clipboard.'
st = ClientGUICommon.BetterStaticText( exif_panel, label = label )
st.setWordWrap( True )
st.setAlignment( QC.Qt.AlignCenter )
self._exif_listctrl = ClientGUIListCtrl.BetterListCtrl( exif_panel, CGLC.COLUMN_LIST_EXIF_DATA.ID, 16, self._ConvertEXIFToListCtrlTuples, activation_callback = self._CopyRow )
exif_panel.Add( st, CC.FLAGS_EXPAND_PERPENDICULAR )
exif_panel.Add( self._exif_listctrl, CC.FLAGS_EXPAND_BOTH_WAYS )
#
text_panel = ClientGUICommon.StaticBox( self, 'embedded text' )
self._text = QW.QPlainTextEdit( text_panel )
self._text.setReadOnly( True )
text_panel.Add( self._text, CC.FLAGS_EXPAND_BOTH_WAYS )
#
extra_rows_panel = ClientGUICommon.StaticBox( self, 'extra info' )
rows = [ ( f'{key}: ', ClientGUICommon.BetterStaticText( extra_rows_panel, label = value ) ) for ( key, value ) in extra_rows ]
gridbox = ClientGUICommon.WrapInGrid( self, rows )
extra_rows_panel.Add( gridbox, CC.FLAGS_EXPAND_PERPENDICULAR )
#
if exif_dict is None:
exif_panel.setVisible( False )
else:
datas = []
for ( exif_id, value ) in exif_dict.items():
if isinstance( value, dict ):
datas.extend( value.items() )
else:
datas.append( ( exif_id, value ) )
self._exif_listctrl.AddDatas( datas )
if file_text is None:
text_panel.setVisible( False )
else:
self._text.setPlainText( file_text )
if len( extra_rows ) == 0:
extra_rows_panel.setVisible( False )
#
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, exif_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( vbox, text_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( vbox, extra_rows_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
self.widget().setLayout( vbox )
def _ConvertEXIFToListCtrlTuples( self, exif_tuple ):
( exif_id, raw_value ) = exif_tuple
if exif_id in ExifTags.TAGS:
label = ExifTags.TAGS[ exif_id ]
pretty_label = label
elif exif_id in ExifTags.GPSTAGS:
label = ExifTags.GPSTAGS[ exif_id ]
pretty_label = label
else:
label = 'zzz'
pretty_label = 'Unknown'
pretty_id = str( exif_id )
if isinstance( raw_value, bytes ):
value = raw_value.hex()
pretty_value = '{}: {}'.format( HydrusData.ToHumanBytes( len( raw_value ) ), value )
else:
value = str( raw_value )
if HydrusText.NULL_CHARACTER in value:
value = value.replace( HydrusText.NULL_CHARACTER, '[null]' )
pretty_value = value
display_tuple = ( pretty_id, pretty_label, pretty_value )
sort_tuple = ( exif_id, label, value )
return ( display_tuple, sort_tuple )
def _CopyRow( self ):
selected_exif_tuples = self._exif_listctrl.GetData( only_selected = True )
if len( selected_exif_tuples ) == 0:
return
( first_row_id, first_row_value ) = selected_exif_tuples[0]
if isinstance( first_row_value, bytes ):
copy_text = first_row_value.hex()
else:
copy_text = str( first_row_value )
HG.client_controller.pub( 'clipboard', 'text', copy_text )
class ReviewFileHistory( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._job_status = ClientThreading.JobStatus()
#
self._search_panel = QW.QWidget( self )
# TODO: ok add 'num_steps' as a control, and a date range
panel_vbox = QP.VBoxLayout()
file_search_context = ClientSearch.FileSearchContext(
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
)
page_key = b'mr bones placeholder'
self._tag_autocomplete = ClientGUIACDropdown.AutoCompleteDropdownTagsRead(
self._search_panel,
page_key,
file_search_context,
allow_all_known_files = False,
force_system_everything = True,
fixed_results_list_height = 8
)
self._loading_text = ClientGUICommon.BetterStaticText( self._search_panel )
self._loading_text.setAlignment( QC.Qt.AlignVCenter | QC.Qt.AlignRight )
self._cancel_button = ClientGUICommon.BetterBitmapButton( self._search_panel, CC.global_pixmaps().stop, self._CancelCurrentSearch )
self._refresh_button = ClientGUICommon.BetterBitmapButton( self._search_panel, CC.global_pixmaps().refresh, self._RefreshSearch )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._loading_text, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( hbox, self._cancel_button, CC.FLAGS_CENTER )
QP.AddToLayout( hbox, self._refresh_button, CC.FLAGS_CENTER )
QP.AddToLayout( panel_vbox, self._tag_autocomplete, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( panel_vbox, hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
panel_vbox.addStretch( 1 )
self._search_panel.setLayout( panel_vbox )
#
self._file_history_chart_panel = QW.QWidget( self )
self._file_history_vbox = QP.VBoxLayout()
self._status_st = ClientGUICommon.BetterStaticText( self._file_history_chart_panel, label = f'loading{HC.UNICODE_ELLIPSIS}' )
self._flip_deleted = QW.QCheckBox( 'show deleted', self._file_history_chart_panel )
self._flip_deleted.setChecked( True )
self._file_history_chart = QW.QWidget( self._file_history_chart_panel )
QP.AddToLayout( self._file_history_vbox, self._status_st, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( self._file_history_vbox, self._flip_deleted, CC.FLAGS_CENTER )
QP.AddToLayout( self._file_history_vbox, self._file_history_chart, CC.FLAGS_EXPAND_BOTH_WAYS )
self._file_history_chart_panel.setLayout( self._file_history_vbox )
#
self._hbox = QP.HBoxLayout()
QP.AddToLayout( self._hbox, self._search_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( self._hbox, self._file_history_chart_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( self._hbox )
self._tag_autocomplete.searchChanged.connect( self._RefreshSearch )
self._RefreshSearch()
def _CancelCurrentSearch( self ):
self._job_status.Cancel()
self._cancel_button.setEnabled( False )
def _RefreshSearch( self ):
def work_callable():
try:
file_history = HG.client_controller.Read( 'file_history', num_steps, file_search_context = file_search_context, job_status = job_status )
except HydrusExceptions.DBException as e:
if isinstance( e.db_e, HydrusExceptions.TooComplicatedM8 ):
return -1
raise
return file_history
def publish_callable( file_history ):
try:
if file_history == -1:
self._status_st.setText( 'Sorry, this domain is too complicated to calculate for now, try something simpler!' )
return
elif job_status.IsCancelled():
self._status_st.setText( 'Cancelled!' )
return
self._file_history_vbox.removeWidget( self._file_history_chart )
# TODO: presumably the thing here is to have SetValue on this widget so we can simply clear/set it rather than the mickey-mouse replace
self._file_history_chart = ClientGUICharts.FileHistory( self._file_history_chart_panel, file_history )
self._file_history_chart.setMinimumSize( 720, 480 )
self._flip_deleted.clicked.connect( self._file_history_chart.FlipDeletedVisible )
QP.AddToLayout( self._file_history_vbox, self._file_history_chart, CC.FLAGS_EXPAND_BOTH_WAYS )
self._status_st.setVisible( False )
self._flip_deleted.setVisible( True )
finally:
self._cancel_button.setEnabled( False )
self._refresh_button.setEnabled( True )
if not self._tag_autocomplete.IsSynchronised():
self._refresh_button.setEnabled( False )
return
file_search_context = self._tag_autocomplete.GetFileSearchContext()
num_steps = 7680
self._cancel_button.setEnabled( True )
self._refresh_button.setEnabled( False )
self._status_st.setText( 'loading' + HC.UNICODE_ELLIPSIS )
self._status_st.setVisible( True )
self._flip_deleted.setVisible( False )
self._file_history_chart.setVisible( False )
self._job_status.Cancel()
job_status = ClientThreading.JobStatus()
self._job_status = job_status
self._update_job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
self._update_job.start()
class ReviewFileMaintenance( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, stats ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._hash_ids = None
self._job_types_to_due_counts = {}
self._job_types_to_not_due_counts = {}
self._notebook = ClientGUICommon.BetterNotebook( self )
#
self._current_work_panel = QW.QWidget( self._notebook )
jobs_listctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self._current_work_panel )
self._jobs_listctrl = ClientGUIListCtrl.BetterListCtrl( jobs_listctrl_panel, CGLC.COLUMN_LIST_FILE_MAINTENANCE_JOBS.ID, 8, self._ConvertJobTypeToListCtrlTuples )
jobs_listctrl_panel.SetListCtrl( self._jobs_listctrl )
jobs_listctrl_panel.AddButton( 'clear', self._DeleteWork, enabled_only_on_selection = True )
jobs_listctrl_panel.AddButton( 'do work', self._DoWork, enabled_check_func = self._WorkToDoOnSelection )
jobs_listctrl_panel.AddButton( 'do all work', self._DoAllWork, enabled_check_func = self._WorkToDo )
jobs_listctrl_panel.AddButton( 'refresh', self._RefreshWorkDue )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, jobs_listctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self._current_work_panel.setLayout( vbox )
#
self._new_work_panel = QW.QWidget( self._notebook )
#
self._search_panel = ClientGUICommon.StaticBox( self._new_work_panel, 'select files by search' )
page_key = HydrusData.GenerateKey()
default_location_context = HG.client_controller.new_options.GetDefaultLocalLocationContext()
file_search_context = ClientSearch.FileSearchContext( location_context = default_location_context )
self._tag_autocomplete = ClientGUIACDropdown.AutoCompleteDropdownTagsRead( self._search_panel, page_key, file_search_context, allow_all_known_files = False, force_system_everything = True )
self._run_search_st = ClientGUICommon.BetterStaticText( self._search_panel, label = 'no results yet' )
self._run_search = ClientGUICommon.BetterButton( self._search_panel, 'run this search', self._RunSearch )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._run_search_st, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( hbox, self._run_search, CC.FLAGS_CENTER_PERPENDICULAR )
self._search_panel.Add( self._tag_autocomplete, CC.FLAGS_EXPAND_PERPENDICULAR )
self._search_panel.Add( hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
#
self._button_panel = ClientGUICommon.StaticBox( self._new_work_panel, 'easy select' )
self._select_all_media_files = ClientGUICommon.BetterButton( self._button_panel, 'all media files', self._SelectAllMediaFiles )
self._select_repo_files = ClientGUICommon.BetterButton( self._button_panel, 'all repository update files', self._SelectRepoUpdateFiles )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._select_all_media_files, CC.FLAGS_CENTER )
QP.AddToLayout( hbox, self._select_repo_files, CC.FLAGS_CENTER )
self._button_panel.Add( hbox, CC.FLAGS_ON_RIGHT )
#
self._action_panel = ClientGUICommon.StaticBox( self._new_work_panel, 'add job' )
self._selected_files_st = ClientGUICommon.BetterStaticText( self._action_panel, label = 'no files selected yet' )
self._action_selector = ClientGUICommon.BetterChoice( self._action_panel )
for job_type in ClientFiles.ALL_REGEN_JOBS_IN_HUMAN_ORDER:
self._action_selector.addItem( ClientFiles.regen_file_enum_to_str_lookup[ job_type ], job_type )
self._description_button = ClientGUICommon.BetterButton( self._action_panel, 'see description', self._SeeDescription )
self._add_new_job = ClientGUICommon.BetterButton( self._action_panel, 'add job', self._AddJob )
self._add_new_job.setEnabled( False )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._selected_files_st, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( hbox, self._action_selector, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( hbox, self._description_button, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( hbox, self._add_new_job, CC.FLAGS_CENTER_PERPENDICULAR )
self._action_panel.Add( hbox, CC.FLAGS_EXPAND_BOTH_WAYS )
#
vbox = QP.VBoxLayout()
label = 'First, select which files you wish to queue up for the job using a normal search. Hit \'run this search\' to select those files.'
label += os.linesep
label += 'Then select the job type and click \'add job\'.'
QP.AddToLayout( vbox, ClientGUICommon.BetterStaticText(self._new_work_panel,label=label), CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, self._search_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, self._button_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, self._action_panel, CC.FLAGS_EXPAND_PERPENDICULAR )
self._new_work_panel.setLayout( vbox )
#
self._notebook.addTab( self._current_work_panel, 'scheduled work' )
self._notebook.addTab( self._new_work_panel, 'add new work' )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, self._notebook, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( vbox )
self._RefreshWorkDue()
HG.client_controller.sub( self, '_RefreshWorkDue', 'notify_files_maintenance_done' )
def _AddJob( self ):
hash_ids = self._hash_ids
job_type = self._action_selector.GetValue()
if len( hash_ids ) > 1000:
message = 'Are you sure you want to schedule "{}" on {} files?'.format( ClientFiles.regen_file_enum_to_str_lookup[ job_type ], HydrusData.ToHumanInt( len( hash_ids ) ) )
result = ClientGUIDialogsQuick.GetYesNo( self, message, yes_label = 'do it', no_label = 'forget it' )
if result != QW.QDialog.Accepted:
return
self._add_new_job.setEnabled( False )
def work_callable():
HG.client_controller.files_maintenance_manager.ScheduleJobHashIds( hash_ids, job_type )
return True
def publish_callable( result ):
ClientGUIDialogsMessage.ShowInformation( self, 'Jobs added!' )
self._add_new_job.setEnabled( True )
self._RefreshWorkDue()
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _ConvertJobTypeToListCtrlTuples( self, job_type ):
pretty_job_type = ClientFiles.regen_file_enum_to_str_lookup[ job_type ]
sort_job_type = pretty_job_type
if job_type in self._job_types_to_due_counts:
num_to_do = self._job_types_to_due_counts[ job_type ]
else:
num_to_do = 0
pretty_num_to_do = HydrusData.ToHumanInt( num_to_do )
not_due_num_to_do = self._job_types_to_not_due_counts[ job_type ]
if not_due_num_to_do > 0:
pretty_num_to_do = '{} ({} is not yet due)'.format( pretty_num_to_do, HydrusData.ToHumanInt( not_due_num_to_do ) )
display_tuple = ( pretty_job_type, pretty_num_to_do )
sort_tuple = ( sort_job_type, num_to_do )
return ( display_tuple, sort_tuple )
def _DeleteWork( self ):
message = 'Clear all the selected scheduled work?'
result = ClientGUIDialogsQuick.GetYesNo( self, message )
if result != QW.QDialog.Accepted:
return
job_types = self._jobs_listctrl.GetData( only_selected = True )
def work_callable():
for job_type in job_types:
HG.client_controller.files_maintenance_manager.CancelJobs( job_type )
return True
def publish_callable( result ):
self._RefreshWorkDue()
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _DoAllWork( self ):
HG.client_controller.CallToThread( HG.client_controller.files_maintenance_manager.ForceMaintenance )
def _DoWork( self ):
job_types = self._jobs_listctrl.GetData( only_selected = True )
if len( job_types ) == 0:
return
HG.client_controller.CallToThread( HG.client_controller.files_maintenance_manager.ForceMaintenance, mandated_job_types = job_types )
def _RefreshWorkDue( self ):
def work_callable():
job_types_to_counts = HG.client_controller.Read( 'file_maintenance_get_job_counts' )
return job_types_to_counts
def publish_callable( job_types_to_counts ):
job_types = set()
self._job_types_to_due_counts = collections.Counter()
self._job_types_to_not_due_counts = collections.Counter()
for ( job_type, ( due_count, not_due_count ) ) in job_types_to_counts.items():
job_types.add( job_type )
self._job_types_to_due_counts[ job_type ] = due_count
self._job_types_to_not_due_counts[ job_type ] = not_due_count
self._jobs_listctrl.SetData( job_types )
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _RunSearch( self ):
self._run_search_st.setText( 'loading' + HC.UNICODE_ELLIPSIS )
self._run_search.setEnabled( False )
file_search_context = self._tag_autocomplete.GetFileSearchContext()
def work_callable():
query_hash_ids = HG.client_controller.Read( 'file_query_ids', file_search_context, apply_implicit_limit = False )
return query_hash_ids
def publish_callable( hash_ids ):
self._run_search_st.setText( '{} files found'.format( HydrusData.ToHumanInt( len( hash_ids ) ) ) )
self._run_search.setEnabled( True )
self._SetHashIds( hash_ids )
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _SeeDescription( self ):
job_type = self._action_selector.GetValue()
message = ClientFiles.regen_file_enum_to_description_lookup[ job_type ]
message += os.linesep * 2
message += 'This job has weight {}, where a normalised unit of file work has value {}.'.format( HydrusData.ToHumanInt( ClientFiles.regen_file_enum_to_job_weight_lookup[ job_type ] ), HydrusData.ToHumanInt( ClientFiles.NORMALISED_BIG_JOB_WEIGHT ) )
ClientGUIDialogsMessage.ShowInformation( self, message )
def _SelectAllMediaFiles( self ):
self._select_all_media_files.setEnabled( False )
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
file_search_context = ClientSearch.FileSearchContext( location_context = location_context )
def work_callable():
query_hash_ids = HG.client_controller.Read( 'file_query_ids', file_search_context, apply_implicit_limit = False )
return query_hash_ids
def publish_callable( hash_ids ):
self._select_all_media_files.setEnabled( True )
self._SetHashIds( hash_ids )
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _SelectRepoUpdateFiles( self ):
self._select_repo_files.setEnabled( False )
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.LOCAL_UPDATE_SERVICE_KEY )
file_search_context = ClientSearch.FileSearchContext( location_context = location_context )
def work_callable():
query_hash_ids = HG.client_controller.Read( 'file_query_ids', file_search_context, apply_implicit_limit = False )
return query_hash_ids
def publish_callable( hash_ids ):
self._select_repo_files.setEnabled( True )
self._SetHashIds( hash_ids )
job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
job.start()
def _SetHashIds( self, hash_ids ):
if hash_ids is None:
hash_ids = set()
self._hash_ids = hash_ids
self._selected_files_st.setText( '{} files selected'.format(HydrusData.ToHumanInt(len(hash_ids))) )
if len( hash_ids ) == 0:
self._add_new_job.setEnabled( False )
else:
self._add_new_job.setEnabled( True )
def _WorkToDo( self ):
return sum( self._job_types_to_due_counts.values() ) > 0
def _WorkToDoOnSelection( self ):
job_types = self._jobs_listctrl.GetData( only_selected = True )
for job_type in job_types:
if self._job_types_to_due_counts[ job_type ] > 0:
return True
return False
class ReviewHowBonedAmI( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._update_job = None
self._job_status = ClientThreading.JobStatus()
vbox = QP.VBoxLayout()
self._mr_bones_text = ClientGUICommon.BetterStaticText( self )
boned_path = os.path.join( HC.STATIC_DIR, 'boned.jpg' )
boned_qt_pixmap = ClientRendering.GenerateHydrusBitmap( boned_path, HC.IMAGE_JPEG ).GetQtPixmap()
self._mr_bones_image = ClientGUICommon.BufferedWindowIcon( self, boned_qt_pixmap )
QP.AddToLayout( vbox, self._mr_bones_image, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, self._mr_bones_text, CC.FLAGS_CENTER )
self._notebook = ClientGUICommon.BetterNotebook( self )
#
self._search_panel = QW.QWidget( self )
self._files_panel = QW.QWidget( self._notebook )
self._views_panel = QW.QWidget( self._notebook )
self._duplicates_panel = QW.QWidget( self._notebook )
self._notebook.addTab( self._files_panel, 'files' )
self._notebook.addTab( self._views_panel, 'views' )
self._notebook.addTab( self._duplicates_panel, 'duplicates' )
#
self._files_content_panel = QW.QWidget( self._files_panel )
self._files_content_vbox = QP.VBoxLayout( margin = 0, spacing = 0 )
QP.AddToLayout( self._files_content_vbox, self._files_content_panel, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
self._files_panel.setLayout( self._files_content_vbox )
#
panel_vbox = QP.VBoxLayout()
self._media_views_st = ClientGUICommon.BetterStaticText( self._views_panel )
self._preview_views_st = ClientGUICommon.BetterStaticText( self._views_panel )
QP.AddToLayout( panel_vbox, self._media_views_st, CC.FLAGS_CENTER )
QP.AddToLayout( panel_vbox, self._preview_views_st, CC.FLAGS_CENTER )
panel_vbox.addStretch( 1 )
self._views_panel.setLayout( panel_vbox )
#
panel_vbox = QP.VBoxLayout()
self._potentials_st = ClientGUICommon.BetterStaticText( self._duplicates_panel )
self._duplicates_st = ClientGUICommon.BetterStaticText( self._duplicates_panel )
self._alternates_st = ClientGUICommon.BetterStaticText( self._duplicates_panel )
QP.AddToLayout( panel_vbox, self._potentials_st, CC.FLAGS_CENTER )
QP.AddToLayout( panel_vbox, self._duplicates_st, CC.FLAGS_CENTER )
QP.AddToLayout( panel_vbox, self._alternates_st, CC.FLAGS_CENTER )
panel_vbox.addStretch( 1 )
self._duplicates_panel.setLayout( panel_vbox )
#
panel_vbox = QP.VBoxLayout()
file_search_context = ClientSearch.FileSearchContext(
location_context = ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
)
page_key = b'mr bones placeholder'
self._tag_autocomplete = ClientGUIACDropdown.AutoCompleteDropdownTagsRead(
self._search_panel,
page_key,
file_search_context,
allow_all_known_files = False,
force_system_everything = True,
fixed_results_list_height = 8
)
self._loading_text = ClientGUICommon.BetterStaticText( self._search_panel )
self._loading_text.setAlignment( QC.Qt.AlignVCenter | QC.Qt.AlignRight )
self._cancel_button = ClientGUICommon.BetterBitmapButton( self._search_panel, CC.global_pixmaps().stop, self._CancelCurrentSearch )
self._refresh_button = ClientGUICommon.BetterBitmapButton( self._search_panel, CC.global_pixmaps().refresh, self._RefreshSearch )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._loading_text, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( hbox, self._cancel_button, CC.FLAGS_CENTER )
QP.AddToLayout( hbox, self._refresh_button, CC.FLAGS_CENTER )
QP.AddToLayout( panel_vbox, self._tag_autocomplete, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( panel_vbox, hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
panel_vbox.addStretch( 1 )
self._search_panel.setLayout( panel_vbox )
#
QP.AddToLayout( vbox, self._notebook, CC.FLAGS_EXPAND_BOTH_WAYS )
big_hbox = QP.HBoxLayout()
QP.AddToLayout( big_hbox, self._search_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( big_hbox, vbox, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( big_hbox )
self._tag_autocomplete.searchChanged.connect( self._RefreshSearch )
self._RefreshSearch()
def _CancelCurrentSearch( self ):
self._job_status.Cancel()
self._cancel_button.setEnabled( False )
def _RefreshSearch( self ):
def work_callable():
boned_stats = HG.client_controller.Read( 'boned_stats', file_search_context = file_search_context, job_status = job_status )
return boned_stats
def publish_callable( boned_stats ):
if job_status.IsCancelled():
self._SetCancelled()
else:
self._SetBones( boned_stats )
self._loading_text.setText( '' )
self._cancel_button.setEnabled( False )
self._refresh_button.setEnabled( True )
if not self._tag_autocomplete.IsSynchronised():
self._refresh_button.setEnabled( False )
return
file_search_context = self._tag_autocomplete.GetFileSearchContext()
self._SetToLoading()
self._job_status.Cancel()
job_status = ClientThreading.JobStatus()
self._job_status = job_status
self._update_job = ClientGUIAsync.AsyncQtJob( self, work_callable, publish_callable )
self._update_job.start()
def _SetBones( self, boned_stats: dict ):
self._SetTheMasterOfCeremonies( boned_stats )
self._SetFilesPanel( boned_stats )
self._SetViewsPanel( boned_stats )
self._SetDuplicatesPanel( boned_stats )
def _SetCancelled( self ):
self._loading_text.setText( 'cancelled!' )
def _SetDuplicatesPanel( self, boned_stats: dict ):
total_alternate_files = boned_stats[ 'total_alternate_files' ]
total_duplicate_files = boned_stats[ 'total_duplicate_files' ]
#total_potential_pairs = boned_stats[ 'total_potential_pairs' ]
#potentials_label = f'Total duplicate potential pairs: {HydrusData.ToHumanInt( total_potential_pairs )}'
potentials_label = f'Total duplicate potential pairs: disabled for now'
duplicates_label = f'Total files set duplicate: {HydrusData.ToHumanInt( total_duplicate_files )}'
alternates_label = f'Total duplicate file groups set alternate: {HydrusData.ToHumanInt( total_alternate_files )}'
self._potentials_st.setText( potentials_label )
self._duplicates_st.setText( duplicates_label )
self._alternates_st.setText( alternates_label )
def _SetFilesPanel( self, boned_stats: dict ):
self._files_content_vbox.removeWidget( self._files_content_panel )
self._files_content_panel.deleteLater()
self._files_content_panel = QW.QWidget( self._files_panel )
num_inbox = boned_stats[ 'num_inbox' ]
num_archive = boned_stats[ 'num_archive' ]
size_inbox = boned_stats[ 'size_inbox' ]
size_archive = boned_stats[ 'size_archive' ]
num_total = num_archive + num_inbox
size_total = size_archive + size_inbox
panel_vbox = QP.VBoxLayout()
if num_total == 0 or size_total == 0:
divide_by_zero_m8 = ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'No files!' )
QP.AddToLayout( panel_vbox, divide_by_zero_m8, CC.FLAGS_CENTER )
else:
current_average_filesize = size_total // num_total
inbox_num_percent = num_inbox / num_total
inbox_size_percent = size_inbox / size_total
if num_inbox > 0:
inbox_average_filesize = size_inbox // num_inbox
else:
inbox_average_filesize = 0
archive_num_percent = num_archive / num_total
archive_size_percent = size_archive / size_total
if num_archive > 0:
archive_average_filesize = size_archive // num_archive
else:
archive_average_filesize = 0
if 'num_deleted' in boned_stats:
num_deleted = boned_stats[ 'num_deleted' ]
size_deleted = boned_stats[ 'size_deleted' ]
num_supertotal = num_total + num_deleted
size_supertotal = size_total + size_deleted
supertotal_average_filesize = size_supertotal // num_supertotal
current_num_percent = num_total / num_supertotal
current_size_percent = size_total / size_supertotal
deleted_num_percent = num_deleted / num_supertotal
deleted_size_percent = size_deleted / size_supertotal
if num_deleted > 0:
deleted_average_filesize = size_deleted // num_deleted
else:
deleted_average_filesize = 0
# spacing=0 to make the weird unicode characters join up neater
text_table_layout = QP.GridLayout( cols = 6, spacing = 0 )
text_table_layout.setHorizontalSpacing( ClientGUIFunctions.ConvertTextToPixelWidth( self, 2 ) )
text_table_layout.setColumnStretch( 0, 1 )
#
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Files' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '%' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Size' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '%' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Average' ), CC.FLAGS_CENTER )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Total Ever Imported:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_supertotal ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_supertotal ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( supertotal_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Current:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_total ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( current_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_total ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( current_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( current_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Deleted:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_deleted ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( deleted_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_deleted ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( deleted_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( deleted_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Inbox:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_inbox ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( inbox_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_inbox ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( inbox_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( inbox_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Archive:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_archive ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( archive_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_archive ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( archive_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( archive_average_filesize ) ), CC.FLAGS_ON_RIGHT )
else:
panel_vbox = QP.VBoxLayout()
# spacing=0 to make the weird unicode characters join up neater
text_table_layout = QP.GridLayout( cols = 6, spacing = 0 )
text_table_layout.setHorizontalSpacing( ClientGUIFunctions.ConvertTextToPixelWidth( self, 2 ) )
text_table_layout.setColumnStretch( 0, 1 )
#
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Files' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '%' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Size' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '%' ), CC.FLAGS_CENTER )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Average' ), CC.FLAGS_CENTER )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Current:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_total ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_total ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( current_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = '' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, QW.QWidget( self._files_content_panel ), CC.FLAGS_ON_LEFT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Inbox:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_inbox ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( inbox_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_inbox ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( inbox_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( inbox_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = 'Archive:' ), CC.FLAGS_ON_LEFT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanInt( num_archive ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( archive_num_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( size_archive ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = ClientData.ConvertZoomToPercentage( archive_size_percent ) ), CC.FLAGS_ON_RIGHT )
QP.AddToLayout( text_table_layout, ClientGUICommon.BetterStaticText( self._files_content_panel, label = HydrusData.ToHumanBytes( archive_average_filesize ) ), CC.FLAGS_ON_RIGHT )
#
QP.AddToLayout( panel_vbox, text_table_layout, CC.FLAGS_EXPAND_PERPENDICULAR )
if 'earliest_import_time' in boned_stats:
panel_vbox.addSpacing( ClientGUIFunctions.ConvertTextToPixelWidth( self, 2 ) )
eit = boned_stats[ 'earliest_import_time' ]
eit_label = 'Earliest file import: {} ({})'.format( HydrusTime.TimestampToPrettyTime( eit ), HydrusTime.TimestampToPrettyTimeDelta( eit ) )
eit_st = ClientGUICommon.BetterStaticText( self._files_content_panel, label = eit_label )
QP.AddToLayout( panel_vbox, eit_st, CC.FLAGS_CENTER )
panel_vbox.addStretch( 1 )
self._files_content_panel.setLayout( panel_vbox )
QP.AddToLayout( self._files_content_vbox, self._files_content_panel, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
def _SetTheMasterOfCeremonies( self, boned_stats: dict ):
num_inbox = boned_stats[ 'num_inbox' ]
num_archive = boned_stats[ 'num_archive' ]
num_total = num_archive + num_inbox
special_message = None
num_deleted = boned_stats.get( 'num_deleted', 0 )
current_fsc = self._tag_autocomplete.GetFileSearchContext()
special_message_is_appropriate = len( current_fsc.GetPredicates() ) == 0 and current_fsc.GetLocationContext() == ClientLocation.LocationContext.STATICCreateSimple( CC.COMBINED_LOCAL_MEDIA_SERVICE_KEY )
if special_message_is_appropriate:
if num_total == 0 and num_deleted == 0:
special_message = 'You have yet to board the ride.'
elif num_total + num_deleted < 1000:
special_message = 'I hope you enjoy my software. You might like to check out the downloaders! :^)'
elif num_inbox <= num_archive / 99:
special_message = 'CONGRATULATIONS. YOU APPEAR TO BE UNBONED--BUT REMAIN EVER VIGILANT'
if special_message is None:
self._mr_bones_text.setVisible( False )
self._mr_bones_image.setVisible( True )
else:
self._mr_bones_image.setVisible( False )
self._mr_bones_text.setVisible( True )
self._mr_bones_text.setText( special_message )
def _SetToLoading( self ):
self._loading_text.setText( 'loading' + HC.UNICODE_ELLIPSIS )
self._files_content_vbox.removeWidget( self._files_content_panel )
self._files_content_panel.deleteLater()
self._files_content_panel = QW.QWidget( self._files_panel )
QP.AddToLayout( self._files_content_vbox, self._files_content_panel, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
self._potentials_st.setText( '' )
self._duplicates_st.setText( '' )
self._alternates_st.setText( '' )
self._media_views_st.setText( '' )
self._preview_views_st.setText( '' )
self._refresh_button.setEnabled( False )
self._cancel_button.setEnabled( True )
def _SetViewsPanel( self, boned_stats: dict ):
total_viewtime = boned_stats[ 'total_viewtime' ]
( media_views, media_viewtime, preview_views, preview_viewtime ) = total_viewtime
media_label = 'Total media views: ' + HydrusData.ToHumanInt( media_views ) + ', totalling ' + HydrusTime.TimeDeltaToPrettyTimeDelta( media_viewtime )
preview_label = 'Total preview views: ' + HydrusData.ToHumanInt( preview_views ) + ', totalling ' + HydrusTime.TimeDeltaToPrettyTimeDelta( preview_viewtime )
self._media_views_st.setText( media_label )
self._preview_views_st.setText( preview_label )
class ReviewLocalFileImports( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, paths = None ):
if paths is None:
paths = []
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self.widget().installEventFilter( ClientGUIDragDrop.FileDropTarget( self.widget(), filenames_callable = self._AddPathsToList ) )
listctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self )
self._paths_list = ClientGUIListCtrl.BetterListCtrl( listctrl_panel, CGLC.COLUMN_LIST_INPUT_LOCAL_FILES.ID, 12, self._ConvertListCtrlDataToTuple, delete_key_callback = self.RemovePaths )
listctrl_panel.SetListCtrl( self._paths_list )
listctrl_panel.AddButton( 'add files', self.AddPaths )
listctrl_panel.AddButton( 'add folder', self.AddFolder )
listctrl_panel.AddButton( 'remove files', self.RemovePaths, enabled_only_on_selection = True )
self._progress = ClientGUICommon.TextAndGauge( self )
self._progress_pause = ClientGUICommon.BetterBitmapButton( self, CC.global_pixmaps().pause, self.PauseProgress )
self._progress_pause.setEnabled( False )
self._progress_cancel = ClientGUICommon.BetterBitmapButton( self, CC.global_pixmaps().stop, self.StopProgress )
self._progress_cancel.setEnabled( False )
file_import_options = FileImportOptions.FileImportOptions()
file_import_options.SetIsDefault( True )
show_downloader_options = False
allow_default_selection = True
self._import_options_button = ClientGUIImportOptions.ImportOptionsButton( self, show_downloader_options, allow_default_selection )
self._import_options_button.SetFileImportOptions( file_import_options )
menu_items = []
check_manager = ClientGUICommon.CheckboxManagerOptions( 'do_human_sort_on_hdd_file_import_paths' )
menu_items.append( ( 'check', 'sort paths as they are added', 'If checked, paths will be sorted in a numerically human-friendly (e.g. "page 9.jpg" comes before "page 10.jpg") way.', check_manager ) )
self._cog_button = ClientGUIMenuButton.MenuBitmapButton( self, CC.global_pixmaps().cog, menu_items )
self._delete_after_success_st = ClientGUICommon.BetterStaticText( self )
self._delete_after_success_st.setAlignment( QC.Qt.AlignRight | QC.Qt.AlignVCenter )
self._delete_after_success_st.setObjectName( 'HydrusWarning' )
self._delete_after_success = QW.QCheckBox( 'delete original files after successful import', self )
self._delete_after_success.clicked.connect( self.EventDeleteAfterSuccessCheck )
self._add_button = ClientGUICommon.BetterButton( self, 'import now', self._DoImport )
self._add_button.setObjectName( 'HydrusAccept' )
self._tag_button = ClientGUICommon.BetterButton( self, 'add tags/urls with the import >>', self._AddTags )
self._tag_button.setObjectName( 'HydrusAccept' )
self._tag_button.setToolTip( 'You can add specific tags to these files, import from sidecar files, or generate them based on filename. Don\'t be afraid to experiment!' )
gauge_sizer = QP.HBoxLayout()
QP.AddToLayout( gauge_sizer, self._progress, CC.FLAGS_EXPAND_SIZER_BOTH_WAYS )
QP.AddToLayout( gauge_sizer, self._progress_pause, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( gauge_sizer, self._progress_cancel, CC.FLAGS_CENTER_PERPENDICULAR )
delete_hbox = QP.HBoxLayout()
QP.AddToLayout( delete_hbox, self._delete_after_success_st, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( delete_hbox, self._delete_after_success, CC.FLAGS_CENTER_PERPENDICULAR )
import_options_buttons = QP.HBoxLayout()
QP.AddToLayout( import_options_buttons, self._import_options_button, CC.FLAGS_CENTER )
QP.AddToLayout( import_options_buttons, self._cog_button, CC.FLAGS_CENTER )
buttons = QP.HBoxLayout()
QP.AddToLayout( buttons, self._add_button, CC.FLAGS_CENTER_PERPENDICULAR )
QP.AddToLayout( buttons, self._tag_button, CC.FLAGS_CENTER_PERPENDICULAR )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, listctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
QP.AddToLayout( vbox, gauge_sizer, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
QP.AddToLayout( vbox, delete_hbox, CC.FLAGS_EXPAND_SIZER_PERPENDICULAR )
QP.AddToLayout( vbox, import_options_buttons, CC.FLAGS_ON_RIGHT )
QP.AddToLayout( vbox, buttons, CC.FLAGS_ON_RIGHT )
self.widget().setLayout( vbox )
self._lock = threading.Lock()
self._current_path_data = {}
self._job_status = ClientThreading.JobStatus()
self._unparsed_paths_queue = queue.Queue()
self._currently_parsing = threading.Event()
self._work_to_do = threading.Event()
self._parsed_path_queue = queue.Queue()
self._pause_event = threading.Event()
self._cancel_event = threading.Event()
self._progress_updater = ClientGUIAsync.FastThreadToGUIUpdater( self._progress, self._progress.SetValue )
if len( paths ) > 0:
self._AddPathsToList( paths )
HG.client_controller.gui.RegisterUIUpdateWindow( self )
HG.client_controller.CallToThreadLongRunning( self.THREADParseImportablePaths, self._unparsed_paths_queue, self._currently_parsing, self._work_to_do, self._parsed_path_queue, self._progress_updater, self._pause_event, self._cancel_event )
def _AddPathsToList( self, paths ):
if self._cancel_event.is_set():
message = 'Please wait for the cancel to clear.'
ClientGUIDialogsMessage.ShowWarning( self, message )
return
self._unparsed_paths_queue.put( paths )
def _AddTags( self ):
# TODO: convert this class to have a filenametaggingoptions and the structure for 'tags for these files', which is separate
# then make this button not start the import. just edit the options and routers and return
# if needed, we convert to paths_to_additional_tags on ultimate ok, or we convert the hdd import to just hold service_keys_to_filenametaggingoptions, like an import folder does
paths = self._paths_list.GetData()
if len( paths ) > 0:
file_import_options = self._import_options_button.GetFileImportOptions()
with ClientGUITopLevelWindowsPanels.DialogEdit( self, 'filename tagging', frame_key = 'local_import_filename_tagging' ) as dlg:
panel = ClientGUIImport.EditLocalImportFilenameTaggingPanel( dlg, paths )
dlg.SetPanel( panel )
if dlg.exec() == QW.QDialog.Accepted:
( metadata_routers, paths_to_additional_service_keys_to_tags ) = panel.GetValue()
delete_after_success = self._delete_after_success.isChecked()
HG.client_controller.pub( 'new_hdd_import', paths, file_import_options, metadata_routers, paths_to_additional_service_keys_to_tags, delete_after_success )
self._OKParent()
def _ConvertListCtrlDataToTuple( self, path ):
( index, mime, size ) = self._current_path_data[ path ]
pretty_index = HydrusData.ToHumanInt( index )
pretty_mime = HC.mime_string_lookup[ mime ]
pretty_size = HydrusData.ToHumanBytes( size )
display_tuple = ( pretty_index, path, pretty_mime, pretty_size )
sort_tuple = ( index, path, pretty_mime, size )
return ( display_tuple, sort_tuple )
def _DoImport( self ):
paths = self._paths_list.GetData()
if len( paths ) > 0:
file_import_options = self._import_options_button.GetFileImportOptions()
metadata_routers = []
paths_to_additional_service_keys_to_tags = collections.defaultdict( ClientTags.ServiceKeysToTags )
delete_after_success = self._delete_after_success.isChecked()
HG.client_controller.pub( 'new_hdd_import', paths, file_import_options, metadata_routers, paths_to_additional_service_keys_to_tags, delete_after_success )
self._OKParent()
def _TidyUp( self ):
self._pause_event.set()
def AddFolder( self ):
with QP.DirDialog( self, 'Select a folder to add.' ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
path = dlg.GetPath()
self._AddPathsToList( ( path, ) )
def AddPaths( self ):
with QP.FileDialog( self, 'Select the files to add.', fileMode = QW.QFileDialog.ExistingFiles ) as dlg:
if dlg.exec() == QW.QDialog.Accepted:
paths = dlg.GetPaths()
self._AddPathsToList( paths )
def CleanBeforeDestroy( self ):
ClientGUIScrolledPanels.ReviewPanel.CleanBeforeDestroy( self )
self._TidyUp()
def EventDeleteAfterSuccessCheck( self ):
if self._delete_after_success.isChecked():
self._delete_after_success_st.setText( 'YOUR ORIGINAL FILES WILL BE DELETED' )
else:
self._delete_after_success_st.clear()
def PauseProgress( self ):
if self._pause_event.is_set():
self._pause_event.clear()
else:
self._pause_event.set()
def StopProgress( self ):
self._cancel_event.set()
def RemovePaths( self ):
text = 'Remove all selected?'
result = ClientGUIDialogsQuick.GetYesNo( self, text )
if result == QW.QDialog.Accepted:
paths_to_delete = self._paths_list.GetData( only_selected = True )
self._paths_list.DeleteSelected()
for path in paths_to_delete:
del self._current_path_data[ path ]
flat_path_data = sorted( ( ( index, path, mime, size ) for ( path, ( index, mime, size ) ) in self._current_path_data.items() ) )
new_index = 1
for ( old_index, path, mime, size ) in flat_path_data:
self._current_path_data[ path ] = ( new_index, mime, size )
new_index += 1
self._paths_list.UpdateDatas()
def THREADParseImportablePaths( self, unparsed_paths_queue, currently_parsing, work_to_do, parsed_path_queue, progress_updater, pause_event, cancel_event ):
unparsed_paths = collections.deque()
num_files_done = 0
num_good_files = 0
num_empty_files = 0
num_missing_files = 0
num_unimportable_mime_files = 0
num_occupied_files = 0
num_sidecars = 0
while not HG.started_shutdown:
if not self or not QP.isValid( self ):
return
if len( unparsed_paths ) > 0:
work_to_do.set()
else:
work_to_do.clear()
currently_parsing.clear()
# ready to start, let's update ui on status
total_paths = num_files_done + len( unparsed_paths )
if num_good_files == 0:
if num_files_done == 0:
message = 'waiting for paths to parse'
else:
message = 'none of the ' + HydrusData.ToHumanInt( total_paths ) + ' files parsed successfully'
else:
message = HydrusData.ConvertValueRangeToPrettyString( num_good_files, total_paths ) + ' files parsed successfully'
if num_empty_files + num_missing_files + num_unimportable_mime_files + num_occupied_files + num_sidecars > 0:
if num_good_files == 0:
message += ': '
else:
message += ', but '
bad_comments = []
if num_sidecars > 0:
bad_comments.append( '{} looked like txt/json/xml sidecars'.format( HydrusData.ToHumanInt( num_sidecars ) ) )
if num_empty_files > 0:
bad_comments.append( HydrusData.ToHumanInt( num_empty_files ) + ' were empty' )
if num_missing_files > 0:
bad_comments.append( HydrusData.ToHumanInt( num_missing_files ) + ' were missing' )
if num_unimportable_mime_files > 0:
bad_comments.append( HydrusData.ToHumanInt( num_unimportable_mime_files ) + ' had unsupported file types' )
if num_occupied_files > 0:
bad_comments.append( HydrusData.ToHumanInt( num_occupied_files ) + ' were inaccessible (maybe in use by another process)' )
message += ' and '.join( bad_comments )
message += '.'
progress_updater.Update( message, num_files_done, total_paths )
# status updated, lets see what work there is to do
if cancel_event.is_set():
while not unparsed_paths_queue.empty():
try:
unparsed_paths_queue.get( block = False )
except queue.Empty:
pass
unparsed_paths = collections.deque()
cancel_event.clear()
pause_event.clear()
continue
if pause_event.is_set():
time.sleep( 1 )
continue
# let's see if there is anything to parse
# first we'll flesh out unparsed_paths with anything new to look at
do_human_sort = HG.client_controller.new_options.GetBoolean( 'do_human_sort_on_hdd_file_import_paths' )
while not unparsed_paths_queue.empty():
try:
raw_paths = unparsed_paths_queue.get( block = False )
( paths, num_sidecars_this_loop ) = ClientFiles.GetAllFilePaths( raw_paths, do_human_sort = do_human_sort ) # convert any dirs to subpaths
num_sidecars += num_sidecars_this_loop
unparsed_paths.extend( paths )
except queue.Empty:
pass
# if unparsed_paths still has nothing, we'll nonetheless sleep on new paths
if len( unparsed_paths ) == 0:
try:
raw_paths = unparsed_paths_queue.get( timeout = 5 )
( paths, num_sidecars_this_loop ) = ClientFiles.GetAllFilePaths( raw_paths, do_human_sort = do_human_sort ) # convert any dirs to subpaths
num_sidecars += num_sidecars_this_loop
unparsed_paths.extend( paths )
except queue.Empty:
pass
continue # either we added some or didn't--in any case, restart the cycle
path = unparsed_paths.popleft()
currently_parsing.set()
# we are now dealing with a file. let's clear out quick no-gos
num_files_done += 1
if path.endswith( os.path.sep + 'Thumbs.db' ) or path.endswith( os.path.sep + 'thumbs.db' ):
HydrusData.Print( 'In import parse, skipping thumbs.db: ' + path )
num_unimportable_mime_files += 1
continue
if not os.path.exists( path ):
HydrusData.Print( 'Missing file: ' + path )
num_missing_files += 1
continue
if not HydrusPaths.PathIsFree( path ):
HydrusData.Print( 'File currently in use: ' + path )
num_occupied_files += 1
continue
size = os.path.getsize( path )
if size == 0:
HydrusData.Print( 'Empty file: ' + path )
num_empty_files += 1
continue
# looks good, let's burn some CPU
try:
mime = HydrusFileHandling.GetMime( path )
except Exception as e:
HydrusData.Print( 'Problem parsing mime for: ' + path )
HydrusData.PrintException( e )
mime = HC.APPLICATION_UNKNOWN
if mime in HC.ALLOWED_MIMES:
num_good_files += 1
parsed_path_queue.put( ( path, mime, size ) )
else:
HydrusData.Print( 'Unparsable file: ' + path )
num_unimportable_mime_files += 1
def TIMERUIUpdate( self ):
good_paths = list()
while not self._parsed_path_queue.empty():
( path, mime, size ) = self._parsed_path_queue.get()
if not self._paths_list.HasData( path ):
good_paths.append( path )
index = len( self._current_path_data ) + 1
self._current_path_data[ path ] = ( index, mime, size )
if len( good_paths ) > 0:
self._paths_list.AddDatas( good_paths )
#
files_in_list = len( self._current_path_data ) > 0
paused = self._pause_event.is_set()
no_work_in_queue = not self._work_to_do.is_set()
working_on_a_file_now = self._currently_parsing.is_set()
can_import = files_in_list and ( no_work_in_queue or paused ) and not working_on_a_file_now
if no_work_in_queue:
self._progress_pause.setEnabled( False )
self._progress_cancel.setEnabled( False )
else:
self._progress_pause.setEnabled( True )
self._progress_cancel.setEnabled( True )
if can_import:
self._add_button.setEnabled( True )
self._tag_button.setEnabled( True )
else:
self._add_button.setEnabled( False )
self._tag_button.setEnabled( False )
if paused:
ClientGUIFunctions.SetBitmapButtonBitmap( self._progress_pause, CC.global_pixmaps().play )
else:
ClientGUIFunctions.SetBitmapButtonBitmap( self._progress_pause, CC.global_pixmaps().pause )
class JobSchedulerPanel( QW.QWidget ):
def __init__( self, parent, controller, scheduler_name ):
self._controller = controller
self._scheduler_name = scheduler_name
QW.QWidget.__init__( self, parent )
self._list_ctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self )
self._list_ctrl = ClientGUIListCtrl.BetterListCtrl( self._list_ctrl_panel, CGLC.COLUMN_LIST_JOB_SCHEDULER_REVIEW.ID, 20, self._ConvertDataToListCtrlTuples )
self._list_ctrl_panel.SetListCtrl( self._list_ctrl )
# maybe some buttons to control behaviour here for debug
self._list_ctrl_panel.AddButton( 'refresh snapshot', self._RefreshSnapshot )
#
self._list_ctrl.Sort()
self._RefreshSnapshot()
#
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, self._list_ctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self.setLayout( vbox )
def _ConvertDataToListCtrlTuples( self, job: HydrusThreading.SchedulableJob ):
job_type = job.PRETTY_CLASS_NAME
job_call = job.GetPrettyJob()
due = job.GetNextWorkTime()
pretty_job_type = job_type
pretty_job_call = job_call
pretty_due = job.GetDueString()
display_tuple = ( pretty_job_type, pretty_job_call, pretty_due )
sort_tuple = ( job_type, job_call, due )
return ( display_tuple, sort_tuple )
def _RefreshSnapshot( self ):
jobs = self._controller.GetJobSchedulerSnapshot( self._scheduler_name )
self._list_ctrl.SetData( jobs )
class ThreadsPanel( QW.QWidget ):
def __init__( self, parent, controller ):
self._controller = controller
QW.QWidget.__init__( self, parent )
self._list_ctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self )
self._list_ctrl = ClientGUIListCtrl.BetterListCtrl( self._list_ctrl_panel, CGLC.COLUMN_LIST_THREADS_REVIEW.ID, 20, self._ConvertDataToListCtrlTuples )
self._list_ctrl_panel.SetListCtrl( self._list_ctrl )
# maybe some buttons to control behaviour here for debug
self._list_ctrl_panel.AddButton( 'refresh snapshot', self._RefreshSnapshot )
#
self._list_ctrl.Sort()
self._RefreshSnapshot()
#
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, self._list_ctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self.setLayout( vbox )
def _ConvertDataToListCtrlTuples( self, thread ):
name = thread.GetName()
thread_type = repr( type( thread ) )
current_job = repr( thread.GetCurrentJobSummary() )
pretty_name = name
pretty_thread_type = thread_type
pretty_current_job = current_job
display_tuple = ( pretty_name, pretty_thread_type, pretty_current_job )
sort_tuple = ( name, thread_type, current_job )
return ( display_tuple, sort_tuple )
def _RefreshSnapshot( self ):
threads = self._controller.GetThreadsSnapshot()
self._list_ctrl.SetData( threads )
class ReviewDeferredDeleteTableData( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, controller ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._controller = controller
#
info_message = '''When large database objects are no longer needed, they are not deleted immediately, but incrementally shrunk in the background over time. This is the queue of pending deferred delete database tables and approximate sizes.'''
st = ClientGUICommon.BetterStaticText( self, label = info_message )
st.setWordWrap( True )
deferred_delete_listctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self )
self._names_to_num_rows = {}
self._deferred_delete_listctrl = ClientGUIListCtrl.BetterListCtrl( deferred_delete_listctrl_panel, CGLC.COLUMN_LIST_DEFERRED_DELETE_TABLE_DATA.ID, 24, self._ConvertRowToListCtrlTuples )
deferred_delete_listctrl_panel.SetListCtrl( self._deferred_delete_listctrl )
self._refresh_button = ClientGUICommon.BetterBitmapButton( self, CC.global_pixmaps().refresh, self._RefreshData )
self._work_hard_button = ClientGUICommon.BetterButton( self,'work hard now', self._FlipWorkingHard )
#
#
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, st, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, deferred_delete_listctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
hbox = QP.HBoxLayout()
QP.AddToLayout( hbox, self._refresh_button, CC.FLAGS_CENTER )
QP.AddToLayout( hbox, self._work_hard_button, CC.FLAGS_CENTER )
QP.AddToLayout( vbox, hbox, CC.FLAGS_ON_RIGHT )
self.widget().setLayout( vbox )
self._updater = self._InitialiseUpdater()
self._RefreshData()
self._UpdateButtonLabel()
self._controller.sub( self, 'NotifyRefresh', 'notify_deferred_delete_database_maintenance_work_complete' )
self._controller.sub( self, 'NotifyStateChange', 'notify_deferred_delete_database_maintenance_state_change' )
self._controller.sub( self, 'NotifyRefresh', 'notify_deferred_delete_database_maintenance_new_work' )
def _ConvertRowToListCtrlTuples( self, name ):
num_rows = self._names_to_num_rows.get( name, None )
sort_name = name
pretty_name = name
if num_rows is None:
sort_num_rows = -1
pretty_num_rows = 'unknown'
else:
sort_num_rows = num_rows
pretty_num_rows = HydrusData.ToHumanInt( sort_num_rows )
display_tuple = ( pretty_name, pretty_num_rows )
sort_tuple = ( sort_name, sort_num_rows )
return ( display_tuple, sort_tuple )
def _FlipWorkingHard( self ):
self._controller.database_maintenance_manager.FlipWorkingHard()
self._UpdateButtonLabel()
def _InitialiseUpdater( self ) -> ClientGUIAsync.AsyncQtUpdater:
def loading_callable():
self._refresh_button.setEnabled( False )
def work_callable( args ):
deferred_delete_data = self._controller.Read( 'deferred_delete_data' )
return deferred_delete_data
def publish_callable( deferred_delete_data ):
self._names_to_num_rows = dict( deferred_delete_data )
self._deferred_delete_listctrl.SetData( list( self._names_to_num_rows.keys() ) )
self._deferred_delete_listctrl.Sort()
self._refresh_button.setEnabled( True )
self._UpdateButtonLabel()
return ClientGUIAsync.AsyncQtUpdater( self, loading_callable, work_callable, publish_callable )
def _RefreshData( self ):
self._updater.update()
def _UpdateButtonLabel( self ):
if self._controller.database_maintenance_manager.IsWorkingHard():
label = 'slow down'
enabled = True
else:
work_to_do = len( self._deferred_delete_listctrl.GetData() ) > 0
if work_to_do:
label = 'work hard'
else:
label = 'all work done!'
enabled = work_to_do
self._work_hard_button.setText( label )
self._work_hard_button.setEnabled( enabled )
def NotifyRefresh( self ):
self._RefreshData()
def NotifyStateChange( self ):
self._UpdateButtonLabel()
class ReviewThreads( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, controller ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._notebook = ClientGUICommon.BetterNotebook( self )
self._thread_panel = ThreadsPanel( self._notebook, controller )
self._fast_scheduler_panel = JobSchedulerPanel( self._notebook, controller, 'fast' )
self._slow_scheduler_panel = JobSchedulerPanel( self._notebook, controller, 'slow' )
self._notebook.addTab( self._thread_panel, 'threads' )
self._notebook.addTab( self._fast_scheduler_panel, 'fast scheduler' )
self._notebook.addTab( self._slow_scheduler_panel, 'slow scheduler' )
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, self._notebook, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( vbox )
class ReviewVacuumData( ClientGUIScrolledPanels.ReviewPanel ):
def __init__( self, parent, controller, vacuum_data ):
ClientGUIScrolledPanels.ReviewPanel.__init__( self, parent )
self._controller = controller
self._vacuum_data = vacuum_data
self._currently_vacuuming = False
#
info_message = '''Vacuuming is essentially an aggressive defrag of a database file. The entire database is copied contiguously to a new file, which then has tightly packed pages and no empty 'free' pages.
Because the new database is tightly packed, it will generally be smaller than the original file. This is currently the only way to truncate a hydrus database file.
Vacuuming is an expensive operation. It requires lots of free space on your drive(s) (including a full copy in your temp directory!), hydrus cannot operate while it is going on, and it tends to run quite slow, about 1-40MB/s. The main benefit is in truncating the database files after you delete a lot of data, so I recommend you only do it on files with a lot of free space.'''
st = ClientGUICommon.BetterStaticText( self, label = info_message )
st.setWordWrap( True )
vacuum_listctrl_panel = ClientGUIListCtrl.BetterListCtrlPanel( self )
self._vacuum_listctrl = ClientGUIListCtrl.BetterListCtrl( vacuum_listctrl_panel, CGLC.COLUMN_LIST_VACUUM_DATA.ID, 6, self._ConvertNameToListCtrlTuples )
vacuum_listctrl_panel.SetListCtrl( self._vacuum_listctrl )
vacuum_listctrl_panel.AddButton( 'vacuum', self._Vacuum, enabled_check_func = self._CanVacuum )
#
self._vacuum_listctrl.SetData( list( self._vacuum_data.keys() ) )
self._vacuum_listctrl.Sort()
#
vbox = QP.VBoxLayout()
QP.AddToLayout( vbox, st, CC.FLAGS_EXPAND_PERPENDICULAR )
QP.AddToLayout( vbox, vacuum_listctrl_panel, CC.FLAGS_EXPAND_BOTH_WAYS )
self.widget().setLayout( vbox )
def _CanVacuumName( self, name ):
from hydrus.core import HydrusDB
try:
vacuum_dict = self._vacuum_data[ name ]
path = vacuum_dict[ 'path' ]
page_size = vacuum_dict[ 'page_size' ]
page_count = vacuum_dict[ 'page_count' ]
freelist_count = vacuum_dict[ 'freelist_count' ]
HydrusDB.CheckCanVacuumData( path, page_size, page_count, freelist_count )
except Exception as e:
return ( False, str( e ) )
return ( True, 'yes!' )
def _CanVacuum( self ):
names = self._vacuum_listctrl.GetData( only_selected = True )
if len( names ) == 0:
return False
for name in names:
( result, info ) = self._CanVacuumName( name )
if not result:
return False
return True
def _ConvertNameToListCtrlTuples( self, name ):
vacuum_dict = self._vacuum_data[ name ]
page_size = vacuum_dict[ 'page_size' ]
sort_name = name
pretty_name = name
sort_total_size = page_size * vacuum_dict[ 'page_count' ]
pretty_total_size = HydrusData.ToHumanBytes( sort_total_size )
sort_free_size = page_size * vacuum_dict[ 'freelist_count' ]
pretty_free_size = HydrusData.ToHumanBytes( sort_free_size )
if sort_total_size > 0:
pretty_free_size = '{} ({})'.format( pretty_free_size, HydrusData.ConvertFloatToPercentage( sort_free_size / sort_total_size ) )
sort_last_vacuumed = vacuum_dict[ 'last_vacuumed' ]
if sort_last_vacuumed == 0 or sort_last_vacuumed is None:
sort_last_vacuumed = 0
pretty_last_vacuumed = 'never done'
else:
pretty_last_vacuumed = HydrusTime.TimestampToPrettyTimeDelta( sort_last_vacuumed )
( result, info ) = self._CanVacuumName( name )
sort_can_vacuum = result
pretty_can_vacuum = info
( sort_vacuum_time_estimate, pretty_vacuum_time_estimate ) = self._GetVacuumTimeEstimate( sort_total_size )
display_tuple = ( pretty_name, pretty_total_size, pretty_free_size, pretty_last_vacuumed, pretty_can_vacuum, pretty_vacuum_time_estimate )
sort_tuple = ( sort_name, sort_total_size, sort_free_size, sort_last_vacuumed, sort_can_vacuum, sort_vacuum_time_estimate )
return ( display_tuple, sort_tuple )
def _GetVacuumTimeEstimate( self, db_size ):
from hydrus.core import HydrusDB
vacuum_time_estimate = HydrusDB.GetApproxVacuumDuration( db_size )
pretty_vacuum_time_estimate = '{} to {}'.format( HydrusTime.TimeDeltaToPrettyTimeDelta( vacuum_time_estimate / 40 ), HydrusTime.TimeDeltaToPrettyTimeDelta( vacuum_time_estimate ) )
return ( vacuum_time_estimate, pretty_vacuum_time_estimate )
def _Vacuum( self ):
names = list( self._vacuum_listctrl.GetData( only_selected = True ) )
if len( names ) > 0:
total_size = sum( ( vd[ 'page_size' ] * vd[ 'page_count' ] for vd in ( self._vacuum_data[ name ] for name in names ) ) )
( sort_time_estimate, pretty_time_estimate ) = self._GetVacuumTimeEstimate( total_size )
message = 'Do vacuum now? Estimated time to vacuum is {}.'.format( pretty_time_estimate )
result = ClientGUIDialogsQuick.GetYesNo( self, message, yes_label = 'do it', no_label = 'forget it' )
if result == QW.QDialog.Accepted:
self._controller.Write( 'vacuum', names )
self._OKParent()