Mercurial > hg > xemacs-beta
view lisp/oobr/tree-nx/TreeView.m @ 36:c53a95d3c46d r19-15b101
Import from CVS: tag r19-15b101
author | cvs |
---|---|
date | Mon, 13 Aug 2007 08:53:38 +0200 |
parents | 376386a54a3c |
children |
line wrap: on
line source
#import "TreeView.h" #import "TreeButton.h" #import "NamedTree.h" #import "Line.h" // constants to determine how the buttons are laid out #define BUTTONWIDTH 155.0 #define BUTTONHEIGHT 24.0 #define VERTSPACING 8.0 #define HORIZSPACING 40.0 @implementation TreeView - initFrame:(const NXRect *)frameRect { [super initFrame:frameRect]; [self setAutosizing:(unsigned int) (NX_WIDTHSIZABLE | NX_HEIGHTSIZABLE)]; currentButton = nil; lineList = nil; priorButton = nil; [selectedField setNextText: selectedField]; [selectedField setPreviousText: selectedField]; currScale = 1.0; [self registerForDragging]; [self setOpaque:YES]; return self; } - buildTreeFromNode:aNode bottom:(double)ybot top:(double)ytop atX:(double)xpos parent:(NXPoint *)pos { // add a button representing the node to the View // This method is recursive. NXRect butFrame = {{(xpos + HORIZSPACING), (ybot + (ytop - ybot) / 2 - BUTTONHEIGHT / 2)}, {BUTTONWIDTH, BUTTONHEIGHT}}; id newButton = [[[TreeButton alloc] initFrame:&butFrame] setTreeNode:aNode]; id kid, kids = [aNode branches]; int numBranches = [kids count]; int i, treeWidth; double diff, accum = ybot; NXPoint myCenter = {(NX_X(&butFrame)), (NX_Y(&butFrame) + BUTTONHEIGHT / 2)}; id newLine; [newButton setTitle:[aNode label]]; [self addSubview:newButton]; // line to parent: if (pos) { // NULL if root, so no line NXPoint parentRight = { pos->x + BUTTONWIDTH, pos->y }; newLine = [[Line alloc] init]; [newLine setStart:&parentRight end:&myCenter]; [lineList addObject:newLine]; } // now add any children and the lines to them. for (i=numBranches - 1; i >= 0; i--) { // loop isn't entered if no kids. kid = [kids objectAt:i]; treeWidth = [kid width]; diff = (treeWidth * (BUTTONHEIGHT + VERTSPACING)); [self buildTreeFromNode:kid bottom:accum top:(accum + diff + VERTSPACING) atX:(xpos + BUTTONWIDTH + HORIZSPACING) parent:&myCenter]; accum += diff; } return self; } - attachTree:aTree { int treeWidth = [aTree width]; int treeDepth = [aTree depth]; double height = (treeWidth * (BUTTONHEIGHT + VERTSPACING) + VERTSPACING); double width = (treeDepth * (BUTTONWIDTH + HORIZSPACING) + HORIZSPACING); treeRoot = aTree; if (lineList) [[lineList freeObjects] free]; lineList = [[List alloc] init]; // resize the View to accomodate the Buttons [self sizeTo:width :height]; [self buildTreeFromNode:aTree bottom:0.0 top:height atX:0.0 parent:NULL]; return self; } - drawSelf:(NXRect *)rects :(int)rectCount // standard rendering method { NXColor color = [[self window] backgroundColor]; if (NXEqualColor(color, NX_COLORLTGRAY)) color = NX_COLORDKGRAY; // PSsetgray(NX_DKGRAY); NXSetColor(color); NXRectFill(&bounds); // PSsetgray(NX_BLACK); NXSetColor(NX_COLORBLACK); PSsetlinewidth(2.0); [lineList makeObjectsPerform:@selector(render)]; [[self subviews] makeObjectsPerform:@selector(display)]; return self; } - scale:sender { id popUp = [sender window]; short index = [popUp indexOfItem:[popUp selectedItem]]; // 25% 50% 75% 100% 125% 150% 200% SizeToFit // 0 1 2 3 4 5 6 7 float factors[] = {0.25, 0.50, 0.75, 1.0, 1.25, 1.50, 2.0, 0.20}; NXPoint center; NXCoord scale = factors[index]; // Initialize width and height bounds when view is not scaled. if (currScale == 1.0) { origWidth = NX_WIDTH(&bounds); origHeight = NX_HEIGHT(&bounds); } // Remember the center to we can reset it after the scaling. center.x = NX_X(&bounds) + NX_WIDTH(&bounds) / 2; center.y = NX_Y(&bounds) + NX_HEIGHT(&bounds) / 2; // Scale the view to its new size if (index == 3) // 100% (Normal Size) { [self setDrawSize:origWidth :origHeight]; currScale = 1.0; } else { currScale *= scale; [self setDrawSize:NX_WIDTH(&bounds) / currScale :NX_HEIGHT(&bounds) / currScale]; } // Reset the center point [self setDrawOrigin:center.x - NX_WIDTH(&bounds) / 2 :center.y - NX_HEIGHT(&bounds) / 2]; // Ensure that selected button, if any, is visible. [self displayBut:currentButton]; [self update]; return self; } - setCurrentButton:but { if (but) { priorButton = currentButton; if (priorButton) { [priorButton setType:NX_MOMENTARYPUSH]; [priorButton setState:0]; } currentButton = but; [currentButton setType:NX_ONOFF]; [currentButton setState:1]; // [selectedField setStringValueNoCopy: [but title]]; } return but; } - setCurrButtonByName:sender { id currBut = [self getButByName:[sender stringValue]]; [self displayBut:[self setCurrentButton:currBut]]; [treeRoot act:currBut]; return currBut; } - getButByName:(const char*)name { id butList = [self subviews]; id but = nil; id currBut = nil; int i = 0; while (!currBut && (but = [butList objectAt:i++])) { if (!strcmp([but title], name)) currBut = but; } return currBut; } - displayButByName:sender { id but = [self getButByName:[sender stringValue]]; if (but) [self displayBut:but]; return but; } - displayBut:but { NXRect butRect; if (but) { [[but getBounds:&butRect] convertRectToSuperview:&butRect]; [self scrollRectToVisible:&butRect]; } return self; } @end // Color dragging support BOOL includesType(const NXAtom *types, NXAtom type) { if (types) while (*types) if (*types++ == type) return YES; return NO; } @implementation TreeView(Drag) - registerForDragging { [self registerForDraggedTypes:&NXColorPboardType count:1]; return self; } - (NXDragOperation)draggingEntered:(id <NXDraggingInfo>)sender { NXDragOperation sourceMask = [sender draggingSourceOperationMask]; Pasteboard *pboard = [sender draggingPasteboard]; return ((includesType([pboard types], NXColorPboardType)) ? NX_DragOperationGeneric : NX_DragOperationNone); } - (BOOL)prepareForDragOperation:(id <NXDraggingInfo>)sender { return YES; } - (BOOL)performDragOperation:(id <NXDraggingInfo>)sender { Pasteboard *pboard = [sender draggingPasteboard]; if (includesType([pboard types], NXColorPboardType)) { NXColor color = NXReadColorFromPasteboard(pboard); [[self window] setBackgroundColor:color]; [self display]; // reflect color change return YES; } else return NO; } - concludeDragOperation:(id <NXDraggingInfo>)sender { // Return value ignored. return nil; } @end