import W
import Wkeys
import struct
import string
import types
import re
from Carbon import Qd, Icn, Fm, QuickDraw
from Carbon.QuickDraw import hilitetransfermode


nullid = '\0\0'
closedid = struct.pack('h', 468)
openid = struct.pack('h', 469)
closedsolidid = struct.pack('h', 470)
opensolidid = struct.pack('h', 471)

arrows = (nullid, closedid, openid, closedsolidid, opensolidid)

has_ctlcharsRE = re.compile(r'[\000-\037\177-\377]')
def ctlcharsREsearch(str):
	if has_ctlcharsRE.search(str) is None:
		return -1
	return 1
	
def double_repr(key, value, truncvalue = 0, 
			type = type, StringType = types.StringType,
			has_ctlchars = ctlcharsREsearch, _repr = repr, str = str):
	if type(key) == StringType and has_ctlchars(key) < 0:
		key = str(key)
	else:
		key = _repr(key)
	if key == '__builtins__':
		value = "<" + type(value).__name__ + " '__builtin__'>"
	elif key == '__return__':
		# bleh, when returning from a class codeblock we get infinite recursion in repr. 
		# Use safe repr instead.
		import repr
		value = repr.repr(value)
	else:
		try:
			value = _repr(value)
			'' + value	# test to see if it is a string, in case a __repr__ method is buggy
		except:
			value = '\xa5\xa5\xa5 exception in repr()'
	if truncvalue:
		return key + '\t' + value[:255]
	return key + '\t' + value


def truncString(s, maxwid):
	if maxwid < 1:
		return 1, ""
	strlen = len(s)
	strwid = Qd.TextWidth(s, 0, strlen);
	if strwid <= maxwid:
		return 0, s
	
	Qd.TextFace(QuickDraw.condense)
	strwid = Qd.TextWidth(s, 0, strlen)
	ellipsis = Qd.StringWidth('\xc9')
	
	if strwid <= maxwid:
		Qd.TextFace(0)
		return 1, s
	if strwid < 1:
		Qd.TextFace(0)
		return 1, ""
	
	mid = int(strlen * maxwid / strwid)
	while 1:
		if mid <= 0:
			mid = 0
			break
		strwid = Qd.TextWidth(s, 0, mid) + ellipsis
		strwid2 = Qd.TextWidth(s, 0, mid + 1) + ellipsis
		if strwid <= maxwid and maxwid <= strwid2:
			if maxwid == strwid2:
				mid += 1
			break
		if strwid > maxwid:
			mid -= 1
			if mid <= 0:
				mid = 0
				break
		elif strwid2 < maxwid:
			mid += 1
	Qd.TextFace(0)
	return 1, s[:mid] + '\xc9'


def drawTextCell(text, cellRect, ascent, theList):
	l, t, r, b = cellRect
	cellwidth = r - l
	Qd.MoveTo(l + 2, t + ascent)
	condense, text = truncString(text, cellwidth - 3)
	if condense:
		Qd.TextFace(QuickDraw.condense)
	Qd.DrawText(text, 0, len(text))
	Qd.TextFace(0)


PICTWIDTH = 16


class BrowserWidget(W.CustomList):
	
	def __init__(self, possize, object = None, col = 100, closechildren = 0):
		W.List.__init__(self, possize, callback = self.listhit)
		self.object = (None,)
		self.indent = 16
		self.lastmaxindent = 0
		self.closechildren = closechildren
		self.children = []
		self.mincol = 64
		self.setcolumn(col)
		self.bind('return', self.openselection)
		self.bind('enter', self.openselection)
		if object is not None:
			self.set(object)
	
	def set(self, object):
		if self.object[0] is not object:
			self.object = object,
			self[:] = self.unpack(object, 0)
		elif self._parentwindow is not None and self._parentwindow.wid:
			self.update()
	
	def unpack(self, object, indent):
		return unpack_object(object, indent)
	
	def update(self):
		# for now...
		W.SetCursor('watch')
		self.setdrawingmode(0)
		sel = self.getselectedobjects()
		fold = self.getunfoldedobjects()
		topcell = self.gettopcell()
		self[:] = self.unpack(self.object[0], 0)
		self.unfoldobjects(fold)
		self.setselectedobjects(sel)
		self.settopcell(topcell)
		self.setdrawingmode(1)
	
	def setcolumn(self, col):
		self.col = col
		self.colstr = struct.pack('h', col)
		if self._list:
			sel = self.getselection()
			self.setitems(self.items)
			self.setselection(sel)
	
	def key(self, char, event):
		if char in (Wkeys.leftarrowkey, Wkeys.rightarrowkey):
			sel = self.getselection()
			sel.reverse()
			self.setdrawingmode(0)
			for index in sel:
				self.fold(index, char == Wkeys.rightarrowkey)
			self.setdrawingmode(1)
		else:
			W.List.key(self, char, event)
	
	def rollover(self, (x, y), onoff):
		if onoff:
			if self.incolumn((x, y)):
				W.SetCursor('hmover')
			else:
				W.SetCursor('arrow')
	
	def inarrow(self, (x, y)):
		cl, ct, cr, cb = self._list.LRect((0, 0))
		l, t, r, b = self._bounds
		if (x - cl) < 16:
			cellheight = cb - ct
			index = (y - ct) / cellheight
			if index < len(self.items):
				return 1, index
		return None, None
	
	def incolumn(self, (x, y)):
		l, t, r, b = self._list.LRect((0, 0))
		abscol = l + self.col
		return abs(abscol - x) < 3
	
	def trackcolumn(self, (x, y)):
		from Carbon import Qd, QuickDraw, Evt
		self.SetPort()
		l, t, r, b = self._bounds
		bounds = l, t, r, b = l + 1, t + 1, r - 16, b - 1
		abscol = l + self.col
		mincol = l + self.mincol
		maxcol = r - 10
		diff = abscol - x
		Qd.PenPat('\000\377\000\377\000\377\000\377')
		Qd.PenMode(QuickDraw.srcXor)
		rect = abscol - 1, t, abscol, b
		Qd.PaintRect(rect)
		lastpoint = (x, y)
		newcol = -1
		#W.SetCursor('fist')
		while Evt.Button():
			Evt.WaitNextEvent(0, 1, None)  # needed for OSX
			(x, y) = Evt.GetMouse()
			if (x, y) <> lastpoint:
				newcol = x + diff
				newcol = max(newcol, mincol)
				newcol = min(newcol, maxcol)
				Qd.PaintRect(rect)
				rect = newcol - 1, t, newcol, b
				Qd.PaintRect(rect)
				lastpoint = (x, y)
		Qd.PaintRect(rect)
		Qd.PenPat(Qd.GetQDGlobalsBlack())
		Qd.PenNormal()
		if newcol > 0 and newcol <> abscol:
			self.setcolumn(newcol - l)
	
	def click(self, point, modifiers):
		if point == (-1, -1):	# gross.
			W.List.click(self, point ,modifiers)
			return
		hit, index = self.inarrow(point)
		if hit:
			(key, value, arrow, indent) = self.items[index]
			self.fold(index, arrow == 1)
		elif self.incolumn(point):
			self.trackcolumn(point)
		else:
			W.List.click(self, point, modifiers)
	
	# for W.List.key
	def findmatch(self, tag):
		lower = string.lower
		items = self.items
		taglen = len(tag)
		match = '\377' * 100
		match_i = -1
		for i in range(len(items)):
			item = lower(str(items[i][0]))
			if tag <= item < match:
				match = item
				match_i = i
		if match_i >= 0:
			return match_i
		else:
			return len(items) - 1
	
	def close(self):
		if self.closechildren:
			for window in self.children:
				window.close()
		self.children = []
		W.List.close(self)
	
	def fold(self, index, onoff):
		(key, value, arrow, indent) = self.items[index]
		if arrow == 0 or (onoff and arrow == 2) or (not onoff and arrow == 1):
			return
		W.SetCursor('watch')
		topcell = self.gettopcell()
		if onoff:
			self[index] = (key, value, 4, indent)
			self.setdrawingmode(0)
			self[index+1:index+1] = self.unpack(value, indent + 1)
			self[index] = (key, value, 2, indent)
		else:
			self[index] = (key, value, 3, indent)
			self.setdrawingmode(0)
			count = 0
			for i in range(index + 1, len(self.items)):
				(dummy, dummy, dummy, subindent) = self.items[i]
				if subindent <= indent:
					break
				count = count + 1
			self[index+1:index+1+count] = []
			self[index] = (key, value, 1, indent)
		maxindent = self.getmaxindent()
		if maxindent <> self.lastmaxindent:
			newabsindent = self.col + (maxindent - self.lastmaxindent) * self.indent
			if newabsindent >= self.mincol:
				self.setcolumn(newabsindent)
			self.lastmaxindent = maxindent
		self.settopcell(topcell)
		self.setdrawingmode(1)
	
	def unfoldobjects(self, objects):
		for obj in objects:
			try:
				index = self.items.index(obj)
			except ValueError:
				pass
			else:
				self.fold(index, 1)
	
	def getunfoldedobjects(self):
		curindent = 0
		objects = []
		for index in range(len(self.items)):
			(key, value, arrow, indent) = self.items[index]
			if indent > curindent:
				(k, v, a, i) = self.items[index - 1]
				objects.append((k, v, 1, i))
				curindent = indent
			elif indent < curindent:
				curindent = indent
		return objects
	
	def listhit(self, isdbl):
		if isdbl:
			self.openselection()
	
	def openselection(self):
		import os
		sel = self.getselection()
		for index in sel:
			(key, value, arrow, indent) = self[index]
			if arrow:
				self.children.append(Browser(value))
			elif type(value) == types.StringType and '\0' not in value:
				editor = self._parentwindow.parent.getscript(value)
				if editor:
					editor.select()
					return
				elif os.path.exists(value) and os.path.isfile(value):
					if MacOS.GetCreatorAndType(value)[1] in ('TEXT', '\0\0\0\0'):
						W.getapplication().openscript(value)
	
	def itemrepr(self, (key, value, arrow, indent), str = str, double_repr = double_repr, 
			arrows = arrows, pack = struct.pack):
		arrow = arrows[arrow]
		return arrow + pack('h', self.indent * indent) + self.colstr + \
				double_repr(key, value, 1)
	
	def getmaxindent(self, max = max):
		maxindent = 0
		for item in self.items:
			maxindent = max(maxindent, item[3])
		return maxindent
	
	def domenu_copy(self, *args):
		sel = self.getselectedobjects()
		selitems = []
		for key, value, dummy, dummy in sel:
			selitems.append(double_repr(key, value))
		text = string.join(selitems, '\r')
		if text:
			from Carbon import Scrap
			if hasattr(Scrap, 'PutScrap'):
				Scrap.ZeroScrap()
				Scrap.PutScrap('TEXT', text)
			else:
				Scrap.ClearCurrentScrap()
				sc = Scrap.GetCurrentScrap()
				sc.PutScrapFlavor('TEXT', 0, text)

	def listDefDraw(self, selected, cellRect, theCell, 
			dataOffset, dataLen, theList):
		self.myDrawCell(0, selected, cellRect, theCell, 
			dataOffset, dataLen, theList)
	
	def listDefHighlight(self, selected, cellRect, theCell, 
			dataOffset, dataLen, theList):
		self.myDrawCell(1, selected, cellRect, theCell, 
			dataOffset, dataLen, theList)
	
	def myDrawCell(self, onlyHilite, selected, cellRect, theCell, 
			dataOffset, dataLen, theList):
		savedPort = Qd.GetPort()
		Qd.SetPort(theList.GetListPort())
		savedClip = Qd.NewRgn()
		Qd.GetClip(savedClip)
		Qd.ClipRect(cellRect)
		savedPenState = Qd.GetPenState()
		Qd.PenNormal()
		
		l, t, r, b = cellRect
		
		if not onlyHilite:
			Qd.EraseRect(cellRect)
			
			ascent, descent, leading, size, hm = Fm.FontMetrics()
			linefeed = ascent + descent + leading
			
			if dataLen >= 6:
				data = theList.LGetCell(dataLen, theCell)
				iconId, indent, tab = struct.unpack("hhh", data[:6])
				try:
					key, value = data[6:].split("\t", 1)
				except ValueError:
					# bogus data, at least don't crash.
					indent = 0
					tab = 0
					iconId = 0
					key = ""
					value = data[6:]
				
				if iconId:
					try:
						theIcon = Icn.GetCIcon(iconId)
					except Icn.Error:
						pass
					else:
						rect = (0, 0, 16, 16)
						rect = Qd.OffsetRect(rect, l, t)
						rect = Qd.OffsetRect(rect, 0, (theList.cellSize[1] - (rect[3] - rect[1])) / 2)
						Icn.PlotCIcon(rect, theIcon)
				
				if len(key) >= 0:
					cl, ct, cr, cb = cellRect
					vl, vt, vr, vb = self._viewbounds
					cl = vl + PICTWIDTH + indent
					cr = vl + tab
					if cr > vr:
						cr = vr
					if cl < cr:
						drawTextCell(key, (cl, ct, cr, cb), ascent, theList)
					cl = vl + tab
					cr = vr
					if cl < cr:
						drawTextCell(value, (cl, ct, cr, cb), ascent, theList)
			#elif dataLen != 0:
			#	drawTextCell("???", 3, cellRect, ascent, theList)
			else:
				return  # we have bogus data
			
			# draw nice dotted line
			l, t, r, b = cellRect
			l = self._viewbounds[0] + tab
			r = l + 1;
			if not (theList.cellSize[1] & 0x01) or (t & 0x01):
				myPat = "\xff\x00\xff\x00\xff\x00\xff\x00"
			else:
				myPat = "\x00\xff\x00\xff\x00\xff\x00\xff"
			Qd.PenPat(myPat)
			Qd.PenMode(QuickDraw.srcCopy)
			Qd.PaintRect((l, t, r, b))
			Qd.PenNormal()
		
		if selected or onlyHilite:
			l, t, r, b = cellRect
			l = self._viewbounds[0] + PICTWIDTH
			r = self._viewbounds[2]
			Qd.PenMode(hilitetransfermode)
			Qd.PaintRect((l, t, r, b))
		
		# restore graphics environment
		Qd.SetPort(savedPort)
		Qd.SetClip(savedClip)
		Qd.DisposeRgn(savedClip)
		Qd.SetPenState(savedPenState)



class Browser:
	
	def __init__(self, object = None, title = None, closechildren = 0):
		if hasattr(object, '__name__'):
			name = object.__name__
		else:
			name = ''
		if title is None:
			title = 'Object browser'
			if name:
				title = title + ': ' + name
		self.w = w = W.Window((300, 400), title, minsize = (100, 100))
		w.info = W.TextBox((18, 8, -70, 15))
		w.updatebutton = W.BevelButton((-64, 4, 50, 16), 'Update', self.update)
		w.browser = BrowserWidget((-1, 24, 1, -14), None)
		w.bind('cmdu', w.updatebutton.push)
		w.open()
		self.set(object, name)
	
	def close(self):
		if self.w.wid:
			self.w.close()
	
	def set(self, object, name = ''):
		W.SetCursor('watch')
		tp = type(object).__name__
		try:
			length = len(object)
		except:
			length = -1
		if not name and hasattr(object, '__name__'):
			name = object.__name__
		if name:
			info = name + ': ' + tp
		else:
			info = tp
		if length >= 0:
			if length == 1:
				info = info + ' (%d element)' % length
			else:
				info = info + ' (%d elements)' % length
		self.w.info.set(info)
		self.w.browser.set(object)
	
	def update(self):
		self.w.browser.update()


SIMPLE_TYPES = (
	type(None),
	int,
	long,
	float,
	complex,
	str,
	unicode,
)

def get_ivars(obj):
	"""Return a list the names of all (potential) instance variables."""
	# __mro__ recipe from Guido
	slots = {}
	# old-style C objects
	if hasattr(obj, "__members__"):
		for name in obj.__members__:
			slots[name] = None
	if hasattr(obj, "__methods__"):
		for name in obj.__methods__:
			slots[name] = None
	# generic type
	if hasattr(obj, "__dict__"):
		slots.update(obj.__dict__)
	cls = type(obj)
	if hasattr(cls, "__mro__"):
		# new-style class, use descriptors
		for base in cls.__mro__:
			for name, value in base.__dict__.items():
				# XXX using callable() is a heuristic which isn't 100%
				# foolproof.
				if hasattr(value, "__get__") and not callable(value):
					slots[name] = None
	if "__dict__" in slots:
		del slots["__dict__"]
	slots = slots.keys()
	slots.sort()
	return slots

def unpack_object(object, indent = 0):
	tp = type(object)
	if isinstance(object, SIMPLE_TYPES) and object is not None:
		raise TypeError, "can't browse simple type: %s" % tp.__name__
	elif isinstance(object, dict):
		return unpack_dict(object, indent)
	elif isinstance(object, (tuple, list)):
		return unpack_sequence(object, indent)
	elif isinstance(object, types.ModuleType):
		return unpack_dict(object.__dict__, indent)
	else:
		return unpack_other(object, indent)

def unpack_sequence(seq, indent = 0):
	return [(i, v, not isinstance(v, SIMPLE_TYPES), indent)
	         for i, v in enumerate(seq)]

def unpack_dict(dict, indent = 0):
	items = dict.items()
	return pack_items(items, indent)

def unpack_instance(inst, indent = 0):
	if hasattr(inst, '__pybrowse_unpack__'):
		return unpack_object(inst.__pybrowse_unpack__(), indent)
	else:
		items = [('__class__', inst.__class__)] + inst.__dict__.items()
		return pack_items(items, indent)

def unpack_class(clss, indent = 0):
	items = [('__bases__', clss.__bases__), ('__name__', clss.__name__)] + clss.__dict__.items()
	return pack_items(items, indent)

def unpack_other(object, indent = 0):
	attrs = get_ivars(object)
	items = []
	for attr in attrs:
		try:
			value = getattr(object, attr)
		except:
			pass
		else:
			items.append((attr, value))
	return pack_items(items, indent)

def pack_items(items, indent = 0):
	items = [(k, v, not isinstance(v, SIMPLE_TYPES), indent)
	         for k, v in items]
	return tuple_caselesssort(items)

def caselesssort(alist):
	"""Return a sorted copy of a list. If there are only strings in the list, 
	it will not consider case"""
	
	try:
		# turn ['FOO',  'aaBc', 'ABcD'] into [('foo', 'FOO'), ('aabc', 'aaBc'), ('abcd', 'ABcD')], if possible
		tupledlist = map(lambda item, lower = string.lower: (lower(item), item), alist)
	except TypeError:
		# at least one element in alist is not a string, proceed the normal way...
		alist = alist[:]
		alist.sort()
		return alist
	else:
		tupledlist.sort()
		# turn [('aabc', 'aaBc'), ('abcd', 'ABcD'), ('foo', 'FOO')] into ['aaBc', 'ABcD', 'FOO']
		return map(lambda x: x[1], tupledlist)

def tuple_caselesssort(items):
	try:
		tupledlist = map(lambda tuple, lower = string.lower: (lower(tuple[0]), tuple), items)
	except (AttributeError, TypeError):
		items = items[:]
		items.sort()
		return items
	else:
		tupledlist.sort()
		return map(lambda (low, tuple): tuple, tupledlist)