| # |
| # General parser/loaders for preferences files and such |
| # |
| import Res |
| import macfs |
| import struct |
| import MACFS |
| |
| READ=1 |
| READWRITE=3 |
| Error = "Preferences.Error" |
| |
| debug = 0 |
| |
| class NullLoader: |
| def __init__(self, data=None): |
| self.data = data |
| |
| def load(self): |
| if self.data is None: |
| raise Error, "No default given" |
| return self.data |
| |
| def save(self, data): |
| raise Error, "Cannot save to default value" |
| |
| def delete(self, deep=0): |
| if debug: |
| print 'Attempt to delete default value' |
| raise Error, "Cannot delete default value" |
| |
| _defaultdefault = NullLoader() |
| |
| class ResLoader: |
| def __init__(self, filename, resid, resnum=None, resname=None, default=_defaultdefault): |
| self.filename = filename |
| self.fss = macfs.FSSpec(self.filename) |
| self.resid = resid |
| self.resnum = resnum |
| self.resname = resname |
| self.default = default |
| self.data = None |
| |
| def load(self): |
| oldrh = Res.CurResFile() |
| try: |
| rh = Res.FSpOpenResFile(self.fss, READ) |
| except Res.Error: |
| self.data = self.default.load() |
| return self.data |
| try: |
| if self.resname: |
| handle = Res.Get1NamedResource(self.resid, self.resname) |
| else: |
| handle = Res.Get1Resource(self.resid, self.resnum) |
| except Res.Error: |
| self.data = self.default.load() |
| else: |
| if debug: |
| print 'Loaded', (self.resid, self.resnum, self.resname), 'from', self.fss.as_pathname() |
| self.data = handle.data |
| Res.CloseResFile(rh) |
| Res.UseResFile(oldrh) |
| return self.data |
| |
| def save(self, data): |
| if self.data is None or self.data != data: |
| oldrh = Res.CurResFile() |
| rh = Res.FSpOpenResFile(self.fss, READWRITE) |
| try: |
| handle = Res.Get1Resource(self.resid, self.resnum) |
| except Res.Error: |
| handle = Res.Resource(data) |
| handle.AddResource(self.resid, self.resnum, '') |
| if debug: |
| print 'Added', (self.resid, self.resnum), 'to', self.fss.as_pathname() |
| else: |
| handle.data = data |
| handle.ChangedResource() |
| if debug: |
| print 'Changed', (self.resid, self.resnum), 'in', self.fss.as_pathname() |
| Res.CloseResFile(rh) |
| Res.UseResFile(oldrh) |
| |
| def delete(self, deep=0): |
| if debug: |
| print 'Deleting in', self.fss.as_pathname(), `self.data`, deep |
| oldrh = Res.CurResFile() |
| rh = Res.FSpOpenResFile(self.fss, READWRITE) |
| try: |
| handle = Res.Get1Resource(self.resid, self.resnum) |
| except Res.Error: |
| if deep: |
| if debug: print 'deep in', self.default |
| self.default.delete(1) |
| else: |
| handle.RemoveResource() |
| if debug: |
| print 'Deleted', (self.resid, self.resnum), 'from', self.fss.as_pathname() |
| self.data = None |
| Res.CloseResFile(rh) |
| Res.UseResFile(oldrh) |
| |
| class AnyResLoader: |
| def __init__(self, resid, resnum=None, resname=None, default=_defaultdefault): |
| self.resid = resid |
| self.resnum = resnum |
| self.resname = resname |
| self.default = default |
| self.data = None |
| |
| def load(self): |
| try: |
| if self.resname: |
| handle = Res.GetNamedResource(self.resid, self.resname) |
| else: |
| handle = Res.GetResource(self.resid, self.resnum) |
| except Res.Error: |
| self.data = self.default.load() |
| else: |
| self.data = handle.data |
| return self.data |
| |
| def save(self, data): |
| raise Error, "Cannot save AnyResLoader preferences" |
| |
| def delete(self, deep=0): |
| raise Error, "Cannot delete AnyResLoader preferences" |
| |
| class StructLoader: |
| def __init__(self, format, loader): |
| self.format = format |
| self.loader = loader |
| |
| def load(self): |
| data = self.loader.load() |
| return struct.unpack(self.format, data) |
| |
| def save(self, data): |
| data = apply(struct.pack, (self.format,)+data) |
| self.loader.save(data) |
| |
| def delete(self, deep=0): |
| self.loader.delete(deep) |
| |
| class PstringLoader: |
| def __init__(self, loader): |
| self.loader = loader |
| |
| def load(self): |
| data = self.loader.load() |
| len = ord(data[0]) |
| return data[1:1+len] |
| |
| def save(self, data): |
| if len(data) > 255: |
| raise Error, "String too big for pascal-style" |
| self.loader.save(chr(len(data))+data) |
| |
| def delete(self, deep=0): |
| self.loader.delete(deep) |
| |
| class VersionLoader(StructLoader): |
| def load(self): |
| while 1: |
| data = self.loader.load() |
| if debug: |
| print 'Versionloader:', `data` |
| try: |
| rv = struct.unpack(self.format, data) |
| rv = self.versioncheck(rv) |
| return rv |
| except (struct.error, Error): |
| self.delete(1) |
| |
| def versioncheck(self, data): |
| return data |
| |
| class StrListLoader: |
| def __init__(self, loader): |
| self.loader = loader |
| |
| def load(self): |
| data = self.loader.load() |
| num, = struct.unpack('h', data[:2]) |
| data = data[2:] |
| rv = [] |
| for i in range(num): |
| strlen = ord(data[0]) |
| if strlen < 0: strlen = strlen + 256 |
| str = data[1:strlen+1] |
| data = data[strlen+1:] |
| rv.append(str) |
| return rv |
| |
| def save(self, list): |
| rv = struct.pack('h', len(list)) |
| for str in list: |
| rv = rv + chr(len(str)) + str |
| self.loader.save(rv) |
| |
| def delete(self, deep=0): |
| self.loader.delete(deep) |
| |
| def preferencefile(filename, creator=None, type=None): |
| create = creator != None and type != None |
| vrefnum, dirid = macfs.FindFolder(MACFS.kOnSystemDisk, 'pref', create) |
| fss = macfs.FSSpec((vrefnum, dirid, ":Python:" + filename)) |
| oldrf = Res.CurResFile() |
| if create: |
| try: |
| rh = Res.FSpOpenResFile(fss, READ) |
| except Res.Error: |
| Res.FSpCreateResFile(fss, creator, type, MACFS.smAllScripts) |
| else: |
| Res.CloseResFile(rh) |
| Res.UseResFile(oldrf) |
| return fss |
| |