2009-08-05 08:59:46 +00:00
|
|
|
# Created By: Virgil Dupras
|
|
|
|
# Created On: 2006/11/11
|
|
|
|
# $Id$
|
|
|
|
# Copyright 2009 Hardcoded Software (http://www.hardcoded.net)
|
|
|
|
#
|
|
|
|
# This software is licensed under the "HS" License as described in the "LICENSE" file,
|
|
|
|
# which should be included with this package. The terms are also available at
|
|
|
|
# http://www.hardcoded.net/licenses/hs_license
|
|
|
|
|
2009-09-29 14:07:50 +00:00
|
|
|
from Foundation import *
|
2009-06-01 09:55:11 +00:00
|
|
|
from AppKit import *
|
|
|
|
import logging
|
|
|
|
import os.path as op
|
|
|
|
|
|
|
|
from hsutil import io, cocoa, job
|
2009-06-16 09:04:41 +00:00
|
|
|
from hsutil.cocoa import install_exception_hook
|
|
|
|
from hsutil.misc import stripnone
|
2009-06-01 09:55:11 +00:00
|
|
|
from hsutil.reg import RegistrationRequired
|
|
|
|
|
2009-10-23 12:56:52 +00:00
|
|
|
from . import app, fs
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
JOBID2TITLE = {
|
|
|
|
app.JOB_SCAN: "Scanning for duplicates",
|
|
|
|
app.JOB_LOAD: "Loading",
|
|
|
|
app.JOB_MOVE: "Moving",
|
|
|
|
app.JOB_COPY: "Copying",
|
|
|
|
app.JOB_DELETE: "Sending to Trash",
|
|
|
|
}
|
|
|
|
|
|
|
|
def demo_method(method):
|
|
|
|
def wrapper(self, *args, **kwargs):
|
|
|
|
try:
|
|
|
|
return method(self, *args, **kwargs)
|
|
|
|
except RegistrationRequired:
|
|
|
|
NSNotificationCenter.defaultCenter().postNotificationName_object_('RegistrationRequired', self)
|
|
|
|
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
class DupeGuru(app.DupeGuru):
|
|
|
|
def __init__(self, data_module, appdata_subdir, appid):
|
|
|
|
LOGGING_LEVEL = logging.DEBUG if NSUserDefaults.standardUserDefaults().boolForKey_('debug') else logging.WARNING
|
|
|
|
logging.basicConfig(level=LOGGING_LEVEL, format='%(levelname)s %(message)s')
|
|
|
|
logging.debug('started in debug mode')
|
|
|
|
install_exception_hook()
|
2009-09-29 14:07:50 +00:00
|
|
|
appsupport = NSSearchPathForDirectoriesInDomains(NSApplicationSupportDirectory, NSUserDomainMask, True)[0]
|
|
|
|
appdata = op.join(appsupport, appdata_subdir)
|
2009-06-01 09:55:11 +00:00
|
|
|
app.DupeGuru.__init__(self, data_module, appdata, appid)
|
|
|
|
self.progress = cocoa.ThreadedJobPerformer()
|
|
|
|
self.display_delta_values = False
|
|
|
|
self.selected_dupes = []
|
|
|
|
self.RefreshDetailsTable(None,None)
|
|
|
|
|
|
|
|
#--- Override
|
|
|
|
@staticmethod
|
|
|
|
def _recycle_dupe(dupe):
|
|
|
|
if not io.exists(dupe.path):
|
|
|
|
dupe.parent = None
|
|
|
|
return True
|
|
|
|
directory = unicode(dupe.parent.path)
|
|
|
|
filename = dupe.name
|
|
|
|
result, tag = NSWorkspace.sharedWorkspace().performFileOperation_source_destination_files_tag_(
|
|
|
|
NSWorkspaceRecycleOperation, directory, '', [filename])
|
|
|
|
if not io.exists(dupe.path):
|
|
|
|
dupe.parent = None
|
|
|
|
return True
|
|
|
|
logging.warning('Could not send %s to trash. tag: %d' % (unicode(dupe.path), tag))
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _start_job(self, jobid, func):
|
|
|
|
try:
|
|
|
|
j = self.progress.create_job()
|
|
|
|
self.progress.run_threaded(func, args=(j, ))
|
|
|
|
except job.JobInProgressError:
|
|
|
|
NSNotificationCenter.defaultCenter().postNotificationName_object_('JobInProgress', self)
|
|
|
|
else:
|
|
|
|
ud = {'desc': JOBID2TITLE[jobid], 'jobid':jobid}
|
|
|
|
NSNotificationCenter.defaultCenter().postNotificationName_object_userInfo_('JobStarted', self, ud)
|
|
|
|
|
|
|
|
#---Helpers
|
|
|
|
def GetObjects(self,node_path):
|
|
|
|
#returns a tuple g,d
|
|
|
|
try:
|
|
|
|
g = self.results.groups[node_path[0]]
|
|
|
|
if len(node_path) == 2:
|
|
|
|
return (g,self.results.groups[node_path[0]].dupes[node_path[1]])
|
|
|
|
else:
|
|
|
|
return (g,None)
|
|
|
|
except IndexError:
|
|
|
|
return (None,None)
|
|
|
|
|
2009-10-23 12:56:52 +00:00
|
|
|
def get_folder_path(self, node_path, curr_path=None):
|
2009-06-01 09:55:11 +00:00
|
|
|
if not node_path:
|
2009-10-23 12:56:52 +00:00
|
|
|
return curr_path
|
|
|
|
current_index = node_path[0]
|
|
|
|
if curr_path is None:
|
|
|
|
curr_path = self.directories[current_index]
|
2009-06-01 09:55:11 +00:00
|
|
|
else:
|
2009-10-23 12:56:52 +00:00
|
|
|
curr_path = self.directories.get_subfolders(curr_path)[current_index]
|
|
|
|
return self.get_folder_path(node_path[1:], curr_path)
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
def RefreshDetailsTable(self,dupe,group):
|
2009-09-02 10:21:11 +00:00
|
|
|
l1 = self._get_display_info(dupe, group, False)
|
2009-09-02 11:08:36 +00:00
|
|
|
# we don't want the two sides of the table to display the stats for the same file
|
|
|
|
ref = group.ref if group is not None and group.ref is not dupe else None
|
|
|
|
l2 = self._get_display_info(ref, group, False)
|
2009-06-01 09:55:11 +00:00
|
|
|
names = [c['display'] for c in self.data.COLUMNS]
|
|
|
|
self.details_table = zip(names,l1,l2)
|
|
|
|
|
|
|
|
#---Public
|
|
|
|
def AddSelectedToIgnoreList(self):
|
|
|
|
for dupe in self.selected_dupes:
|
2009-06-07 07:13:07 +00:00
|
|
|
self.add_to_ignore_list(dupe)
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
copy_or_move_marked = demo_method(app.DupeGuru.copy_or_move_marked)
|
|
|
|
delete_marked = demo_method(app.DupeGuru.delete_marked)
|
|
|
|
|
|
|
|
def MakeSelectedReference(self):
|
|
|
|
self.make_reference(self.selected_dupes)
|
|
|
|
|
|
|
|
def OpenSelected(self):
|
|
|
|
if self.selected_dupes:
|
|
|
|
path = unicode(self.selected_dupes[0].path)
|
|
|
|
NSWorkspace.sharedWorkspace().openFile_(path)
|
|
|
|
|
|
|
|
def PurgeIgnoreList(self):
|
|
|
|
self.scanner.ignore_list.Filter(lambda f,s:op.exists(f) and op.exists(s))
|
|
|
|
|
|
|
|
def RefreshDetailsWithSelected(self):
|
|
|
|
if self.selected_dupes:
|
|
|
|
self.RefreshDetailsTable(
|
|
|
|
self.selected_dupes[0],
|
|
|
|
self.results.get_group_of_duplicate(self.selected_dupes[0])
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.RefreshDetailsTable(None,None)
|
|
|
|
|
|
|
|
def RemoveDirectory(self,index):
|
|
|
|
try:
|
|
|
|
del self.directories[index]
|
|
|
|
except IndexError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def RemoveSelected(self):
|
|
|
|
self.results.remove_duplicates(self.selected_dupes)
|
|
|
|
|
2009-10-23 12:56:52 +00:00
|
|
|
def RenameSelected(self, newname):
|
2009-06-01 09:55:11 +00:00
|
|
|
try:
|
|
|
|
d = self.selected_dupes[0]
|
2009-10-23 12:56:52 +00:00
|
|
|
d.rename(newname)
|
2009-06-01 09:55:11 +00:00
|
|
|
return True
|
2009-10-23 12:56:52 +00:00
|
|
|
except (IndexError, fs.FSError) as e:
|
|
|
|
logging.warning("dupeGuru Warning: %s" % unicode(e))
|
2009-06-01 09:55:11 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
def RevealSelected(self):
|
|
|
|
if self.selected_dupes:
|
|
|
|
path = unicode(self.selected_dupes[0].path)
|
|
|
|
NSWorkspace.sharedWorkspace().selectFile_inFileViewerRootedAtPath_(path,'')
|
|
|
|
|
|
|
|
def start_scanning(self):
|
|
|
|
self.RefreshDetailsTable(None, None)
|
|
|
|
try:
|
|
|
|
app.DupeGuru.start_scanning(self)
|
|
|
|
return 0
|
|
|
|
except app.NoScannableFileError:
|
|
|
|
return 3
|
|
|
|
except app.AllFilesAreRefError:
|
|
|
|
return 1
|
|
|
|
|
2009-06-16 09:04:41 +00:00
|
|
|
def selected_result_node_paths(self):
|
|
|
|
def get_path(dupe):
|
|
|
|
try:
|
|
|
|
group = self.results.get_group_of_duplicate(dupe)
|
|
|
|
groupindex = self.results.groups.index(group)
|
|
|
|
if dupe is group.ref:
|
|
|
|
return [groupindex]
|
|
|
|
dupeindex = group.dupes.index(dupe)
|
|
|
|
return [groupindex, dupeindex]
|
|
|
|
except ValueError: # dupe not in there
|
|
|
|
return None
|
|
|
|
|
|
|
|
dupes = self.selected_dupes
|
|
|
|
return stripnone(get_path(dupe) for dupe in dupes)
|
|
|
|
|
|
|
|
def selected_powermarker_node_paths(self):
|
|
|
|
def get_path(dupe):
|
|
|
|
try:
|
|
|
|
dupeindex = self.results.dupes.index(dupe)
|
|
|
|
return [dupeindex]
|
|
|
|
except ValueError: # dupe not in there
|
|
|
|
return None
|
|
|
|
|
|
|
|
dupes = self.selected_dupes
|
|
|
|
return stripnone(get_path(dupe) for dupe in dupes)
|
|
|
|
|
2009-06-01 09:55:11 +00:00
|
|
|
def SelectResultNodePaths(self,node_paths):
|
|
|
|
def extract_dupe(t):
|
|
|
|
g,d = t
|
|
|
|
if d is not None:
|
|
|
|
return d
|
|
|
|
else:
|
|
|
|
if g is not None:
|
|
|
|
return g.ref
|
|
|
|
|
|
|
|
selected = [extract_dupe(self.GetObjects(p)) for p in node_paths]
|
|
|
|
self.selected_dupes = [dupe for dupe in selected if dupe is not None]
|
|
|
|
|
|
|
|
def SelectPowerMarkerNodePaths(self,node_paths):
|
|
|
|
rows = [p[0] for p in node_paths]
|
|
|
|
self.selected_dupes = [
|
|
|
|
self.results.dupes[row] for row in rows if row in xrange(len(self.results.dupes))
|
|
|
|
]
|
|
|
|
|
2009-10-23 12:56:52 +00:00
|
|
|
def SetDirectoryState(self, node_path, state):
|
|
|
|
p = self.get_folder_path(node_path)
|
|
|
|
self.directories.set_state(p, state)
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
def sort_dupes(self,key,asc):
|
|
|
|
self.results.sort_dupes(key,asc,self.display_delta_values)
|
|
|
|
|
|
|
|
def sort_groups(self,key,asc):
|
|
|
|
self.results.sort_groups(key,asc)
|
|
|
|
|
|
|
|
def ToggleSelectedMarkState(self):
|
|
|
|
for dupe in self.selected_dupes:
|
|
|
|
self.results.mark_toggle(dupe)
|
|
|
|
|
|
|
|
#---Data
|
|
|
|
def GetOutlineViewMaxLevel(self, tag):
|
|
|
|
if tag == 0:
|
|
|
|
return 2
|
|
|
|
elif tag == 1:
|
|
|
|
return 0
|
|
|
|
elif tag == 2:
|
|
|
|
return 1
|
|
|
|
|
|
|
|
def GetOutlineViewChildCounts(self, tag, node_path):
|
|
|
|
if self.progress._job_running:
|
|
|
|
return []
|
|
|
|
if tag == 0: #Normal results
|
|
|
|
assert not node_path # no other value is possible
|
|
|
|
return [len(g.dupes) for g in self.results.groups]
|
|
|
|
elif tag == 1: #Directories
|
2009-09-01 06:53:58 +00:00
|
|
|
try:
|
2009-10-23 12:56:52 +00:00
|
|
|
path = self.get_folder_path(node_path)
|
|
|
|
subfolders = self.directories.get_subfolders(path)
|
|
|
|
return [len(self.directories.get_subfolders(path)) for path in subfolders]
|
2009-09-01 06:53:58 +00:00
|
|
|
except IndexError: # node_path out of range
|
|
|
|
return []
|
2009-06-01 09:55:11 +00:00
|
|
|
else: #Power Marker
|
|
|
|
assert not node_path # no other value is possible
|
|
|
|
return [0 for d in self.results.dupes]
|
|
|
|
|
|
|
|
def GetOutlineViewValues(self, tag, node_path):
|
|
|
|
if self.progress._job_running:
|
|
|
|
return
|
|
|
|
if not node_path:
|
|
|
|
return
|
|
|
|
if tag in (0,2): #Normal results / Power Marker
|
|
|
|
if tag == 0:
|
|
|
|
g, d = self.GetObjects(node_path)
|
|
|
|
if d is None:
|
|
|
|
d = g.ref
|
|
|
|
else:
|
|
|
|
d = self.results.dupes[node_path[0]]
|
|
|
|
g = self.results.get_group_of_duplicate(d)
|
2009-09-02 10:21:11 +00:00
|
|
|
result = self._get_display_info(d, g, self.display_delta_values)
|
2009-06-01 09:55:11 +00:00
|
|
|
return result
|
|
|
|
elif tag == 1: #Directories
|
2009-09-01 06:53:58 +00:00
|
|
|
try:
|
2009-10-23 12:56:52 +00:00
|
|
|
path = self.get_folder_path(node_path)
|
|
|
|
return [path[-1], self.directories.get_state(path)]
|
2009-09-01 06:53:58 +00:00
|
|
|
except IndexError: # node_path out of range
|
|
|
|
return []
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
def GetOutlineViewMarked(self, tag, node_path):
|
|
|
|
# 0=unmarked 1=marked 2=unmarkable
|
|
|
|
if self.progress._job_running:
|
|
|
|
return
|
|
|
|
if not node_path:
|
|
|
|
return 2
|
|
|
|
if tag == 1: #Directories
|
|
|
|
return 2
|
|
|
|
if tag == 0: #Normal results
|
|
|
|
g, d = self.GetObjects(node_path)
|
|
|
|
else: #Power Marker
|
|
|
|
d = self.results.dupes[node_path[0]]
|
|
|
|
if (d is None) or (not self.results.is_markable(d)):
|
|
|
|
return 2
|
|
|
|
elif self.results.is_marked(d):
|
|
|
|
return 1
|
|
|
|
else:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def GetTableViewCount(self, tag):
|
|
|
|
if self.progress._job_running:
|
|
|
|
return 0
|
|
|
|
return len(self.details_table)
|
|
|
|
|
|
|
|
def GetTableViewMarkedIndexes(self,tag):
|
|
|
|
return []
|
|
|
|
|
|
|
|
def GetTableViewValues(self,tag,row):
|
|
|
|
return self.details_table[row]
|
|
|
|
|
|
|
|
|