2009-08-05 08:59:46 +00:00
|
|
|
# Created By: Virgil Dupras
|
|
|
|
# Created On: 2006/11/11
|
2010-01-01 20:11:34 +00:00
|
|
|
# Copyright 2010 Hardcoded Software (http://www.hardcoded.net)
|
2009-08-05 08:59:46 +00:00
|
|
|
#
|
|
|
|
# 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-06-01 09:55:11 +00:00
|
|
|
import logging
|
|
|
|
import os.path as op
|
|
|
|
|
2010-01-18 07:48:44 +00:00
|
|
|
from hsutil import cocoa, job
|
2009-06-16 09:04:41 +00:00
|
|
|
from hsutil.cocoa import install_exception_hook
|
2010-02-04 16:12:58 +00:00
|
|
|
from hsutil.cocoa.objcmin import (NSNotificationCenter, NSUserDefaults,
|
|
|
|
NSSearchPathForDirectoriesInDomains, NSApplicationSupportDirectory, NSUserDomainMask,
|
|
|
|
NSWorkspace, NSWorkspaceRecycleOperation)
|
2009-06-16 09:04:41 +00:00
|
|
|
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
|
|
|
|
|
2010-02-05 20:09:04 +00:00
|
|
|
class DupeGuru(app.DupeGuru):
|
2009-06-01 09:55:11 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
#--- Override
|
2010-02-06 11:36:43 +00:00
|
|
|
@staticmethod
|
|
|
|
def _open_path(path):
|
|
|
|
NSWorkspace.sharedWorkspace().openFile_(unicode(path))
|
|
|
|
|
2009-06-01 09:55:11 +00:00
|
|
|
@staticmethod
|
|
|
|
def _recycle_dupe(dupe):
|
2010-01-18 07:48:44 +00:00
|
|
|
# local import because first appkit import takes a lot of memory. we want to avoid it.
|
2009-10-30 16:24:34 +00:00
|
|
|
directory = unicode(dupe.path[:-1])
|
2009-06-01 09:55:11 +00:00
|
|
|
filename = dupe.name
|
2010-02-04 16:12:58 +00:00
|
|
|
result, tag = NSWorkspace.sharedWorkspace().performFileOperation_source_destination_files_tag_(
|
|
|
|
NSWorkspaceRecycleOperation, directory, '', [filename], None)
|
2009-06-01 09:55:11 +00:00
|
|
|
|
2010-02-06 14:31:35 +00:00
|
|
|
@staticmethod
|
|
|
|
def _reveal_path(path):
|
|
|
|
NSWorkspace.sharedWorkspace().selectFile_inFileViewerRootedAtPath_(unicode(path), '')
|
|
|
|
|
2009-06-01 09:55:11 +00:00
|
|
|
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)
|
|
|
|
|
|
|
|
#---Public
|
|
|
|
copy_or_move_marked = demo_method(app.DupeGuru.copy_or_move_marked)
|
|
|
|
delete_marked = demo_method(app.DupeGuru.delete_marked)
|
|
|
|
|
|
|
|
def PurgeIgnoreList(self):
|
|
|
|
self.scanner.ignore_list.Filter(lambda f,s:op.exists(f) and op.exists(s))
|
|
|
|
|
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 start_scanning(self):
|
2010-02-05 19:32:57 +00:00
|
|
|
self._select_dupes([])
|
2009-06-01 09:55:11 +00:00
|
|
|
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]
|
2010-02-05 19:32:57 +00:00
|
|
|
self._select_dupes([dupe for dupe in selected if dupe is not None])
|
2009-06-01 09:55:11 +00:00
|
|
|
|
|
|
|
def SelectPowerMarkerNodePaths(self,node_paths):
|
|
|
|
rows = [p[0] for p in node_paths]
|
2010-02-05 19:32:57 +00:00
|
|
|
dupes = [self.results.dupes[row] for row in rows if row in xrange(len(self.results.dupes))]
|
|
|
|
self._select_dupes(dupes)
|
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 == 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]
|
|
|
|
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)
|
2010-02-05 17:16:05 +00:00
|
|
|
if (d is None) and (g is not None):
|
2009-06-01 09:55:11 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
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 == 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
|
|
|
|
|