findertools.py 29.7 KB
Newer Older
1 2
"""Utility routines depending on the finder,
a combination of code by Jack Jansen and erik@letterror.com.
3

4 5 6 7 8 9 10 11 12 13 14 15 16
Most events have been captured from
Lasso Capture AE and than translated to python code.

IMPORTANT
Note that the processes() function returns different values
depending on the OS version it is running on. On MacOS 9
the Finder returns the process *names* which can then be
used to find out more about them. On MacOS 8.6 and earlier
the Finder returns a code which does not seem to work.
So bottom line: the processes() stuff does not work on < MacOS9

Mostly written by erik@letterror.com
"""
17 18

from warnings import warnpy3k
Benjamin Peterson's avatar
Benjamin Peterson committed
19
warnpy3k("In 3.x, the findertools module is removed.", stacklevel=2)
20

21
import Finder
22
from Carbon import AppleEvents
23 24 25
import aetools
import MacOS
import sys
Jack Jansen's avatar
Jack Jansen committed
26 27
import Carbon.File
import Carbon.Folder
28 29 30 31 32
import aetypes
from types import *

__version__ = '1.1'
Error = 'findertools.Error'
33 34 35 36

_finder_talker = None

def _getfinder():
Jack Jansen's avatar
Jack Jansen committed
37 38 39 40
    """returns basic (recyclable) Finder AE interface object"""
    global _finder_talker
    if not _finder_talker:
        _finder_talker = Finder.Finder()
41
    _finder_talker.send_flags = ( _finder_talker.send_flags |
Jack Jansen's avatar
Jack Jansen committed
42 43
        AppleEvents.kAECanInteract | AppleEvents.kAECanSwitchLayer)
    return _finder_talker
44

45
def launch(file):
Jack Jansen's avatar
Jack Jansen committed
46 47 48 49
    """Open a file thru the finder. Specify file by name or fsspec"""
    finder = _getfinder()
    fss = Carbon.File.FSSpec(file)
    return finder.open(fss)
50

51
def Print(file):
Jack Jansen's avatar
Jack Jansen committed
52 53 54 55
    """Print a file thru the finder. Specify file by name or fsspec"""
    finder = _getfinder()
    fss = Carbon.File.FSSpec(file)
    return finder._print(fss)
56

57
def copy(src, dstdir):
Jack Jansen's avatar
Jack Jansen committed
58 59 60 61 62 63 64 65 66 67
    """Copy a file to a folder"""
    finder = _getfinder()
    if type(src) == type([]):
        src_fss = []
        for s in src:
            src_fss.append(Carbon.File.FSSpec(s))
    else:
        src_fss = Carbon.File.FSSpec(src)
    dst_fss = Carbon.File.FSSpec(dstdir)
    return finder.duplicate(src_fss, to=dst_fss)
68 69

def move(src, dstdir):
Jack Jansen's avatar
Jack Jansen committed
70 71 72 73 74 75 76 77 78 79
    """Move a file to a folder"""
    finder = _getfinder()
    if type(src) == type([]):
        src_fss = []
        for s in src:
            src_fss.append(Carbon.File.FSSpec(s))
    else:
        src_fss = Carbon.File.FSSpec(src)
    dst_fss = Carbon.File.FSSpec(dstdir)
    return finder.move(src_fss, to=dst_fss)
80

81
def sleep():
Jack Jansen's avatar
Jack Jansen committed
82 83 84
    """Put the mac to sleep"""
    finder = _getfinder()
    finder.sleep()
85

86
def shutdown():
Jack Jansen's avatar
Jack Jansen committed
87 88 89
    """Shut the mac down"""
    finder = _getfinder()
    finder.shut_down()
90

91
def restart():
Jack Jansen's avatar
Jack Jansen committed
92 93 94
    """Restart the mac"""
    finder = _getfinder()
    finder.restart()
95 96 97


#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
98
#   Additional findertools
99 100 101
#

def reveal(file):
Jack Jansen's avatar
Jack Jansen committed
102 103 104 105 106
    """Reveal a file in the finder. Specify file by name, fsref or fsspec."""
    finder = _getfinder()
    fsr = Carbon.File.FSRef(file)
    file_alias = fsr.FSNewAliasMinimal()
    return finder.reveal(file_alias)
107

108
def select(file):
Jack Jansen's avatar
Jack Jansen committed
109 110 111 112 113
    """select a file in the finder. Specify file by name, fsref or fsspec."""
    finder = _getfinder()
    fsr = Carbon.File.FSRef(file)
    file_alias = fsr.FSNewAliasMinimal()
    return finder.select(file_alias)
114

115
def update(file):
116
    """Update the display of the specified object(s) to match
Jack Jansen's avatar
Jack Jansen committed
117 118 119 120 121
    their on-disk representation. Specify file by name, fsref or fsspec."""
    finder = _getfinder()
    fsr = Carbon.File.FSRef(file)
    file_alias = fsr.FSNewAliasMinimal()
    return finder.update(file_alias)
122 123 124


#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
125
#   More findertools
126 127 128
#

def comment(object, comment=None):
Jack Jansen's avatar
Jack Jansen committed
129 130 131
    """comment: get or set the Finder-comment of the item, displayed in the 'Get Info' window."""
    object = Carbon.File.FSRef(object)
    object_alias = object.FSNewAliasMonimal()
132
    if comment is None:
Jack Jansen's avatar
Jack Jansen committed
133 134 135
        return _getcomment(object_alias)
    else:
        return _setcomment(object_alias, comment)
136

137
def _setcomment(object_alias, comment):
Jack Jansen's avatar
Jack Jansen committed
138 139 140 141 142 143 144 145 146 147 148 149
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cobj'), form="alis", seld=object_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('comt'), fr=aeobj_00)
    args['----'] = aeobj_01
    args["data"] = comment
    _reply, args, attrs = finder.send("core", "setd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
150 151

def _getcomment(object_alias):
Jack Jansen's avatar
Jack Jansen committed
152 153 154 155 156 157 158 159 160 161 162
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cobj'), form="alis", seld=object_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('comt'), fr=aeobj_00)
    args['----'] = aeobj_01
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
163 164 165


#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
166
#   Get information about current processes in the Finder.
167 168

def processes():
Jack Jansen's avatar
Jack Jansen committed
169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213
    """processes returns a list of all active processes running on this computer and their creators."""
    finder = _getfinder()
    args = {}
    attrs = {}
    processnames = []
    processnumbers = []
    creators = []
    partitions = []
    used = []
    ## get the processnames or else the processnumbers
    args['----'] = aetypes.ObjectSpecifier(want=aetypes.Type('prcs'), form="indx", seld=aetypes.Unknown('abso', "all "), fr=None)
    _reply, args, attrs = finder.send('core', 'getd', args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    p = []
    if args.has_key('----'):
        p =  args['----']
        for proc in p:
            if hasattr(proc, 'seld'):
                # it has a real name
                processnames.append(proc.seld)
            elif hasattr(proc, 'type'):
                if proc.type == "psn ":
                    # it has a process number
                    processnumbers.append(proc.data)
    ## get the creators
    args = {}
    attrs = {}
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('prcs'), form="indx", seld=aetypes.Unknown('abso', "all "), fr=None)
    args['----'] =  aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('fcrt'), fr=aeobj_0)
    _reply, args, attrs = finder.send('core', 'getd', args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(_arg)
    if args.has_key('----'):
        p =  args['----']
        creators = p[:]
    ## concatenate in one dict
    result = []
    if len(processnames) > len(processnumbers):
        data = processnames
    else:
        data = processnumbers
    for i in range(len(creators)):
        result.append((data[i], creators[i]))
    return result
214 215

class _process:
Jack Jansen's avatar
Jack Jansen committed
216
    pass
217 218

def isactiveprocess(processname):
Jack Jansen's avatar
Jack Jansen committed
219 220 221 222 223 224 225
    """Check of processname is active. MacOS9"""
    all = processes()
    ok = 0
    for n, c in all:
        if n == processname:
            return 1
    return 0
226

227
def processinfo(processname):
Jack Jansen's avatar
Jack Jansen committed
228 229
    """Return an object with all process properties as attributes for processname. MacOS9"""
    p = _process()
230

Jack Jansen's avatar
Jack Jansen committed
231 232 233 234 235 236 237 238 239 240 241 242 243 244
    if processname == "Finder":
        p.partition = None
        p.used = None
    else:
        p.partition = _processproperty(processname, 'appt')
        p.used = _processproperty(processname, 'pusd')
    p.visible = _processproperty(processname, 'pvis')       #Is the process' layer visible?
    p.frontmost = _processproperty(processname, 'pisf') #Is the process the frontmost process?
    p.file = _processproperty(processname, 'file')          #the file from which the process was launched
    p.filetype  = _processproperty(processname, 'asty')     #the OSType of the file type of the process
    p.creatortype = _processproperty(processname, 'fcrt')   #the OSType of the creator of the process (the signature)
    p.accepthighlevel = _processproperty(processname, 'revt')   #Is the process high-level event aware (accepts open application, open document, print document, and quit)?
    p.hasscripting = _processproperty(processname, 'hscr')      #Does the process have a scripting terminology, i.e., can it be scripted?
    return p
245

246
def _processproperty(processname, property):
Jack Jansen's avatar
Jack Jansen committed
247 248 249 250 251 252 253 254 255 256 257 258
    """return the partition size and memory used for processname"""
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('prcs'), form="name", seld=processname, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type(property), fr=aeobj_00)
    args['----'] = aeobj_01
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
259 260 261


#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
262
#   Mess around with Finder windows.
263

264
def openwindow(object):
Jack Jansen's avatar
Jack Jansen committed
265 266 267 268 269 270 271 272 273 274 275 276 277
    """Open a Finder window for object, Specify object by name or fsspec."""
    finder = _getfinder()
    object = Carbon.File.FSRef(object)
    object_alias = object.FSNewAliasMinimal()
    args = {}
    attrs = {}
    _code = 'aevt'
    _subcode = 'odoc'
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'), form="alis", seld=object_alias, fr=None)
    args['----'] = aeobj_0
    _reply, args, attrs = finder.send(_code, _subcode, args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
278

279
def closewindow(object):
Jack Jansen's avatar
Jack Jansen committed
280 281 282 283 284 285 286 287 288 289 290 291 292
    """Close a Finder window for folder, Specify by path."""
    finder = _getfinder()
    object = Carbon.File.FSRef(object)
    object_alias = object.FSNewAliasMinimal()
    args = {}
    attrs = {}
    _code = 'core'
    _subcode = 'clos'
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'), form="alis", seld=object_alias, fr=None)
    args['----'] = aeobj_0
    _reply, args, attrs = finder.send(_code, _subcode, args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
293 294

def location(object, pos=None):
Jack Jansen's avatar
Jack Jansen committed
295 296 297 298 299 300 301
    """Set the position of a Finder window for folder to pos=(w, h). Specify file by name or fsspec.
    If pos=None, location will return the current position of the object."""
    object = Carbon.File.FSRef(object)
    object_alias = object.FSNewAliasMinimal()
    if not pos:
        return _getlocation(object_alias)
    return _setlocation(object_alias, pos)
302

303
def _setlocation(object_alias, (x, y)):
Jack Jansen's avatar
Jack Jansen committed
304 305 306 307 308 309 310 311 312 313 314 315
    """_setlocation: Set the location of the icon for the object."""
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'), form="alis", seld=object_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('posn'), fr=aeobj_00)
    args['----'] = aeobj_01
    args["data"] = [x, y]
    _reply, args, attrs = finder.send("core", "setd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    return (x,y)
316

317
def _getlocation(object_alias):
Jack Jansen's avatar
Jack Jansen committed
318 319 320 321 322 323 324 325 326 327 328 329 330
    """_getlocation: get the location of the icon for the object."""
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'), form="alis", seld=object_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('posn'), fr=aeobj_00)
    args['----'] = aeobj_01
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        pos = args['----']
        return pos.h, pos.v
331 332

def label(object, index=None):
Jack Jansen's avatar
Jack Jansen committed
333 334 335
    """label: set or get the label of the item. Specify file by name or fsspec."""
    object = Carbon.File.FSRef(object)
    object_alias = object.FSNewAliasMinimal()
336
    if index is None:
Jack Jansen's avatar
Jack Jansen committed
337 338 339 340
        return _getlabel(object_alias)
    if index < 0 or index > 7:
        index = 0
    return _setlabel(object_alias, index)
341

342
def _getlabel(object_alias):
Jack Jansen's avatar
Jack Jansen committed
343 344 345 346 347 348 349 350 351 352 353 354
    """label: Get the label for the object."""
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cobj'), form="alis", seld=object_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('labi'), fr=aeobj_00)
    args['----'] = aeobj_01
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
355 356

def _setlabel(object_alias, index):
Jack Jansen's avatar
Jack Jansen committed
357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372
    """label: Set the label for the object."""
    finder = _getfinder()
    args = {}
    attrs = {}
    _code = 'core'
    _subcode = 'setd'
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
            form="alis", seld=object_alias, fr=None)
    aeobj_1 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
            form="prop", seld=aetypes.Type('labi'), fr=aeobj_0)
    args['----'] = aeobj_1
    args["data"] = index
    _reply, args, attrs = finder.send(_code, _subcode, args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    return index
373 374

def windowview(folder, view=None):
Jack Jansen's avatar
Jack Jansen committed
375 376 377 378 379 380 381
    """windowview: Set the view of the window for the folder. Specify file by name or fsspec.
    0 = by icon (default)
    1 = by name
    2 = by button
    """
    fsr = Carbon.File.FSRef(folder)
    folder_alias = fsr.FSNewAliasMinimal()
382
    if view is None:
Jack Jansen's avatar
Jack Jansen committed
383 384
        return _getwindowview(folder_alias)
    return _setwindowview(folder_alias, view)
385

386
def _setwindowview(folder_alias, view=0):
Jack Jansen's avatar
Jack Jansen committed
387 388 389 390 391 392 393 394 395 396
    """set the windowview"""
    attrs = {}
    args = {}
    if view == 1:
        _v = aetypes.Type('pnam')
    elif view == 2:
        _v = aetypes.Type('lgbu')
    else:
        _v = aetypes.Type('iimg')
    finder = _getfinder()
397
    aeobj_0 = aetypes.ObjectSpecifier(want = aetypes.Type('cfol'),
Jack Jansen's avatar
Jack Jansen committed
398
            form = 'alis', seld = folder_alias, fr=None)
399
    aeobj_1 = aetypes.ObjectSpecifier(want = aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
400
            form = 'prop', seld = aetypes.Type('cwnd'), fr=aeobj_0)
401
    aeobj_2 = aetypes.ObjectSpecifier(want = aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
402
            form = 'prop', seld = aetypes.Type('pvew'), fr=aeobj_1)
403
    aeobj_3 = aetypes.ObjectSpecifier(want = aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
404 405 406 407 408 409 410 411 412 413
            form = 'prop', seld = _v, fr=None)
    _code = 'core'
    _subcode = 'setd'
    args['----'] = aeobj_2
    args['data'] = aeobj_3
    _reply, args, attrs = finder.send(_code, _subcode, args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
414 415

def _getwindowview(folder_alias):
Jack Jansen's avatar
Jack Jansen committed
416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431
    """get the windowview"""
    attrs = {}
    args = {}
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'), form="alis", seld=folder_alias, fr=None)
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('cwnd'), fr=aeobj_00)
    aeobj_02 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('pvew'), fr=aeobj_01)
    args['----'] = aeobj_02
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    views = {'iimg':0, 'pnam':1, 'lgbu':2}
    if args.has_key('----'):
        return views[args['----'].enum]
432 433

def windowsize(folder, size=None):
Jack Jansen's avatar
Jack Jansen committed
434 435 436 437 438 439 440 441 442 443
    """Set the size of a Finder window for folder to size=(w, h), Specify by path.
    If size=None, windowsize will return the current size of the window.
    Specify file by name or fsspec.
    """
    fsr = Carbon.File.FSRef(folder)
    folder_alias = fsr.FSNewAliasMinimal()
    openwindow(fsr)
    if not size:
        return _getwindowsize(folder_alias)
    return _setwindowsize(folder_alias, size)
444

445
def _setwindowsize(folder_alias, (w, h)):
Jack Jansen's avatar
Jack Jansen committed
446 447 448 449 450 451 452 453 454
    """Set the size of a Finder window for folder to (w, h)"""
    finder = _getfinder()
    args = {}
    attrs = {}
    _code = 'core'
    _subcode = 'setd'
    aevar00 = [w, h]
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'),
            form="alis", seld=folder_alias, fr=None)
455
    aeobj_1 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
456
            form="prop", seld=aetypes.Type('cwnd'), fr=aeobj_0)
457
    aeobj_2 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
458 459 460 461 462 463 464
            form="prop", seld=aetypes.Type('ptsz'), fr=aeobj_1)
    args['----'] = aeobj_2
    args["data"] = aevar00
    _reply, args, attrs = finder.send(_code, _subcode, args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    return (w, h)
465

466
def _getwindowsize(folder_alias):
Jack Jansen's avatar
Jack Jansen committed
467 468 469 470
    """Set the size of a Finder window for folder to (w, h)"""
    finder = _getfinder()
    args = {}
    attrs = {}
471
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'),
Jack Jansen's avatar
Jack Jansen committed
472
            form="alis", seld=folder_alias, fr=None)
473
    aeobj_1 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
474
            form="prop", seld=aetypes.Type('cwnd'), fr=aeobj_0)
475
    aeobj_2 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
476 477 478 479 480 481 482
            form="prop", seld=aetypes.Type('posn'), fr=aeobj_1)
    args['----'] = aeobj_2
    _reply, args, attrs = finder.send('core', 'getd', args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
483 484

def windowposition(folder, pos=None):
Jack Jansen's avatar
Jack Jansen committed
485 486 487 488 489 490 491 492 493 494
    """Set the position of a Finder window for folder to pos=(w, h)."""
    fsr = Carbon.File.FSRef(folder)
    folder_alias = fsr.FSNewAliasMinimal()
    openwindow(fsr)
    if not pos:
        return _getwindowposition(folder_alias)
    if type(pos) == InstanceType:
        # pos might be a QDPoint object as returned by _getwindowposition
        pos = (pos.h, pos.v)
    return _setwindowposition(folder_alias, pos)
495

496
def _setwindowposition(folder_alias, (x, y)):
Jack Jansen's avatar
Jack Jansen committed
497 498 499 500
    """Set the size of a Finder window for folder to (w, h)."""
    finder = _getfinder()
    args = {}
    attrs = {}
501
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'),
Jack Jansen's avatar
Jack Jansen committed
502
            form="alis", seld=folder_alias, fr=None)
503
    aeobj_1 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
504
            form="prop", seld=aetypes.Type('cwnd'), fr=aeobj_0)
505
    aeobj_2 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
506 507 508 509 510 511 512 513
            form="prop", seld=aetypes.Type('posn'), fr=aeobj_1)
    args['----'] = aeobj_2
    args["data"] = [x, y]
    _reply, args, attrs = finder.send('core', 'setd', args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
514 515

def _getwindowposition(folder_alias):
Jack Jansen's avatar
Jack Jansen committed
516 517 518 519
    """Get the size of a Finder window for folder, Specify by path."""
    finder = _getfinder()
    args = {}
    attrs = {}
520
    aeobj_0 = aetypes.ObjectSpecifier(want=aetypes.Type('cfol'),
Jack Jansen's avatar
Jack Jansen committed
521
            form="alis", seld=folder_alias, fr=None)
522
    aeobj_1 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
523
            form="prop", seld=aetypes.Type('cwnd'), fr=aeobj_0)
524
    aeobj_2 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
525 526 527 528 529 530 531
            form="prop", seld=aetypes.Type('ptsz'), fr=aeobj_1)
    args['----'] = aeobj_2
    _reply, args, attrs = finder.send('core', 'getd', args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
532 533

def icon(object, icondata=None):
Jack Jansen's avatar
Jack Jansen committed
534 535 536 537 538 539
    """icon sets the icon of object, if no icondata is given,
    icon will return an AE object with binary data for the current icon.
    If left untouched, this data can be used to paste the icon on another file.
    Development opportunity: get and set the data as PICT."""
    fsr = Carbon.File.FSRef(object)
    object_alias = fsr.FSNewAliasMinimal()
540
    if icondata is None:
Jack Jansen's avatar
Jack Jansen committed
541 542
        return _geticon(object_alias)
    return _seticon(object_alias, icondata)
543

544
def _geticon(object_alias):
Jack Jansen's avatar
Jack Jansen committed
545 546 547 548
    """get the icondata for object. Binary data of some sort."""
    finder = _getfinder()
    args = {}
    attrs = {}
549
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cobj'),
Jack Jansen's avatar
Jack Jansen committed
550
            form="alis", seld=object_alias, fr=None)
551
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
552 553 554 555 556 557 558
            form="prop", seld=aetypes.Type('iimg'), fr=aeobj_00)
    args['----'] = aeobj_01
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
559 560

def _seticon(object_alias, icondata):
Jack Jansen's avatar
Jack Jansen committed
561 562 563 564
    """set the icondata for object, formatted as produced by _geticon()"""
    finder = _getfinder()
    args = {}
    attrs = {}
565
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('cobj'),
Jack Jansen's avatar
Jack Jansen committed
566
            form="alis", seld=object_alias, fr=None)
567
    aeobj_01 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'),
Jack Jansen's avatar
Jack Jansen committed
568 569 570 571 572 573 574 575
            form="prop", seld=aetypes.Type('iimg'), fr=aeobj_00)
    args['----'] = aeobj_01
    args["data"] = icondata
    _reply, args, attrs = finder.send("core", "setd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----'].data
576 577 578


#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
579
#   Volumes and servers.
580

581
def mountvolume(volume, server=None, username=None, password=None):
Jack Jansen's avatar
Jack Jansen committed
582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
    """mount a volume, local or on a server on AppleTalk.
    Note: mounting a ASIP server requires a different operation.
    server is the name of the server where the volume belongs
    username, password belong to a registered user of the volume."""
    finder = _getfinder()
    args = {}
    attrs = {}
    if password:
        args["PASS"] = password
    if username:
        args["USER"] = username
    if server:
        args["SRVR"] = server
    args['----'] = volume
    _reply, args, attrs = finder.send("aevt", "mvol", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
601 602

def unmountvolume(volume):
Jack Jansen's avatar
Jack Jansen committed
603 604
    """unmount a volume that's on the desktop"""
    putaway(volume)
605

606
def putaway(object):
Jack Jansen's avatar
Jack Jansen committed
607 608 609 610 611 612 613 614 615 616
    """puth the object away, whereever it came from."""
    finder = _getfinder()
    args = {}
    attrs = {}
    args['----'] = aetypes.ObjectSpecifier(want=aetypes.Type('cdis'), form="name", seld=object, fr=None)
    _reply, args, attrs = talker.send("fndr", "ptwy", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
617

618

619
#---------------------------------------------------
Jack Jansen's avatar
Jack Jansen committed
620
#   Miscellaneous functions
621 622 623
#

def volumelevel(level):
Jack Jansen's avatar
Jack Jansen committed
624 625 626 627 628 629 630 631 632 633 634 635 636 637
    """set the audio output level, parameter between 0 (silent) and 7 (full blast)"""
    finder = _getfinder()
    args = {}
    attrs = {}
    if level < 0:
        level = 0
    elif level > 7:
        level = 7
    args['----'] = level
    _reply, args, attrs = finder.send("aevt", "stvl", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
638 639

def OSversion():
Jack Jansen's avatar
Jack Jansen committed
640 641 642 643 644 645 646 647 648 649 650
    """return the version of the system software"""
    finder = _getfinder()
    args = {}
    attrs = {}
    aeobj_00 = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('ver2'), fr=None)
    args['----'] = aeobj_00
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        return args['----']
651 652

def filesharing():
Jack Jansen's avatar
Jack Jansen committed
653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681
    """return the current status of filesharing and whether it is starting up or not:
        -1  file sharing is off and not starting up
        0   file sharing is off and starting up
        1   file sharing is on"""
    status = -1
    finder = _getfinder()
    # see if it is on
    args = {}
    attrs = {}
    args['----'] = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('fshr'), fr=None)
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        if args['----'] == 0:
            status = -1
        else:
            status = 1
    # is it starting up perchance?
    args = {}
    attrs = {}
    args['----'] = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('fsup'), fr=None)
    _reply, args, attrs = finder.send("core", "getd", args, attrs)
    if args.has_key('errn'):
        raise Error, aetools.decodeerror(args)
    if args.has_key('----'):
        if args['----'] == 1:
            status = 0
    return status
682

683
def movetotrash(path):
Jack Jansen's avatar
Jack Jansen committed
684 685 686 687
    """move the object to the trash"""
    fss = Carbon.File.FSSpec(path)
    trashfolder = Carbon.Folder.FSFindFolder(fss.as_tuple()[0], 'trsh', 0)
    move(path, trashfolder)
688 689

def emptytrash():
Jack Jansen's avatar
Jack Jansen committed
690 691 692 693 694 695 696 697
    """empty the trash"""
    finder = _getfinder()
    args = {}
    attrs = {}
    args['----'] = aetypes.ObjectSpecifier(want=aetypes.Type('prop'), form="prop", seld=aetypes.Type('trsh'), fr=None)
    _reply, args, attrs = finder.send("fndr", "empt", args, attrs)
    if args.has_key('errn'):
        raise aetools.Error, aetools.decodeerror(args)
698 699 700


def _test():
Jack Jansen's avatar
Jack Jansen committed
701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759
    import EasyDialogs
    print 'Original findertools functionality test...'
    print 'Testing launch...'
    pathname = EasyDialogs.AskFileForOpen('File to launch:')
    if pathname:
        result = launch(pathname)
        if result:
            print 'Result: ', result
        print 'Press return-',
        sys.stdin.readline()
    print 'Testing print...'
    pathname = EasyDialogs.AskFileForOpen('File to print:')
    if pathname:
        result = Print(pathname)
        if result:
            print 'Result: ', result
        print 'Press return-',
        sys.stdin.readline()
    print 'Testing copy...'
    pathname = EasyDialogs.AskFileForOpen('File to copy:')
    if pathname:
        destdir = EasyDialogs.AskFolder('Destination:')
        if destdir:
            result = copy(pathname, destdir)
            if result:
                print 'Result:', result
            print 'Press return-',
            sys.stdin.readline()
    print 'Testing move...'
    pathname = EasyDialogs.AskFileForOpen('File to move:')
    if pathname:
        destdir = EasyDialogs.AskFolder('Destination:')
        if destdir:
            result = move(pathname, destdir)
            if result:
                print 'Result:', result
            print 'Press return-',
            sys.stdin.readline()
    print 'Testing sleep...'
    if EasyDialogs.AskYesNoCancel('Sleep?') > 0:
        result = sleep()
        if result:
            print 'Result:', result
        print 'Press return-',
        sys.stdin.readline()
    print 'Testing shutdown...'
    if EasyDialogs.AskYesNoCancel('Shut down?') > 0:
        result = shutdown()
        if result:
            print 'Result:', result
        print 'Press return-',
        sys.stdin.readline()
    print 'Testing restart...'
    if EasyDialogs.AskYesNoCancel('Restart?') > 0:
        result = restart()
        if result:
            print 'Result:', result
        print 'Press return-',
        sys.stdin.readline()
760 761

def _test2():
Jack Jansen's avatar
Jack Jansen committed
762 763 764 765 766 767 768 769 770 771 772
    print '\nmorefindertools version %s\nTests coming up...' %__version__
    import os
    import random

    # miscellaneous
    print '\tfilesharing on?',  filesharing()       # is file sharing on, off, starting up?
    print '\tOS version',       OSversion()     # the version of the system software

    # set the soundvolume in a simple way
    print '\tSystem beep volume'
    for i in range(0, 7):
773
        volumelevel(i)
Jack Jansen's avatar
Jack Jansen committed
774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811
        MacOS.SysBeep()

    # Finder's windows, file location, file attributes
    open("@findertoolstest", "w")
    f = "@findertoolstest"
    reveal(f)               # reveal this file in a Finder window
    select(f)               # select this file

    base, file = os.path.split(f)
    closewindow(base)   # close the window this file is in  (opened by reveal)
    openwindow(base)        # open it again
    windowview(base, 1) # set the view by list

    label(f, 2)             # set the label of this file to something orange
    print '\tlabel', label(f)   # get the label of this file

    # the file location only works in a window with icon view!
    print 'Random locations for an icon'
    windowview(base, 0)     # set the view by icon
    windowsize(base, (600, 600))
    for i in range(50):
        location(f, (random.randint(10, 590), random.randint(10, 590)))

    windowsize(base, (200, 400))
    windowview(base, 1)     # set the view by icon

    orgpos = windowposition(base)
    print 'Animated window location'
    for i in range(10):
        pos = (100+i*10, 100+i*10)
        windowposition(base, pos)
        print '\twindow position', pos
    windowposition(base, orgpos)    # park it where it was before

    print 'Put a comment in file', f, ':'
    print '\t', comment(f)      # print the Finder comment this file has
    s = 'This is a comment no one reads!'
    comment(f, s)           # set the Finder comment
812

813
def _test3():
Jack Jansen's avatar
Jack Jansen committed
814 815 816 817 818 819
    print 'MacOS9 or better specific functions'
    # processes
    pr = processes()        # return a list of tuples with (active_processname, creatorcode)
    print 'Return a list of current active processes:'
    for p in pr:
        print '\t', p
820

Jack Jansen's avatar
Jack Jansen committed
821 822 823 824 825 826 827 828 829 830
    # get attributes of the first process in the list
    print 'Attributes of the first process in the list:'
    pinfo = processinfo(pr[0][0])
    print '\t', pr[0][0]
    print '\t\tmemory partition', pinfo.partition       # the memory allocated to this process
    print '\t\tmemory used', pinfo.used         # the memory actuall used by this process
    print '\t\tis visible', pinfo.visible           # is the process visible to the user
    print '\t\tis frontmost', pinfo.frontmost       # is the process the front most one?
    print '\t\thas scripting', pinfo.hasscripting       # is the process scriptable?
    print '\t\taccepts high level events',  pinfo.accepthighlevel   # does the process accept high level appleevents?
831

832
if __name__ == '__main__':
Jack Jansen's avatar
Jack Jansen committed
833 834 835
    _test()
    _test2()
    _test3()