/** NSColorWell Control for selecting and display a single color value. Copyright (C) 1996 Free Software Foundation, Inc. Author: Scott Christley Date: 1996 Author: Felipe A. Rodriguez Date: May 1998 This file is part of the GNUstep GUI Library. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; see the file COPYING.LIB. If not, see or write to the Free Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #import "AppKit/NSActionCell.h" #import "AppKit/NSApplication.h" #import "AppKit/NSBezierPath.h" #import "AppKit/NSColorPanel.h" #import "AppKit/NSColorWell.h" #import "AppKit/NSColor.h" #import "AppKit/NSDragging.h" #import "AppKit/NSEvent.h" #import "AppKit/NSGraphics.h" #import "AppKit/NSPasteboard.h" #import "AppKit/NSWindow.h" #import "GNUstepGUI/GSTheme.h" #import #import #include static NSString *GSColorWellDidBecomeExclusiveNotification = @"GSColorWellDidBecomeExclusiveNotification"; @implementation NSColorWell /* * Class methods */ + (void) initialize { if (self == [NSColorWell class]) { [self setVersion: 1]; } } /* * Instance methods */ - (BOOL) acceptsFirstMouse: (NSEvent *)event { return YES; } - (SEL) action { return _action; } /**

Activates the NSColorWell and displays the NSColorPanel with the current NSColorWell's color. The NSColorWell can take color from the NSColorPanel. If exclusive is YES other NSColorWells are desacivated (through notifications).

See Also: -deactivate

*/ - (void) activate: (BOOL)exclusive { NSNotificationCenter *nc = [NSNotificationCenter defaultCenter]; NSColorPanel *colorPanel = [NSColorPanel sharedColorPanel]; if (exclusive == YES) { [nc postNotificationName: GSColorWellDidBecomeExclusiveNotification object: self]; } [nc addObserver: self selector: @selector(deactivate) name: GSColorWellDidBecomeExclusiveNotification object: nil]; [nc addObserver: self selector: @selector(_takeColorFromPanel:) name: NSColorPanelColorDidChangeNotification object: nil]; [nc addObserver: self selector: @selector(deactivate) name: NSWindowWillCloseNotification object: colorPanel]; _is_active = YES; [colorPanel setColor: _the_color]; [colorPanel orderFront: self]; [self setNeedsDisplay: YES]; } /**

Returns the current NSColor of the NSColorWell.

See Also: -setColor:

*/ - (NSColor *) color { return _the_color; } /**

Deactivates the NSColorWell and marks self for display. It is usally call from an observer, when another NSColorWell is activate.

See Also: -activate:

*/ - (void) deactivate { _is_active = NO; [[NSNotificationCenter defaultCenter] removeObserver: self]; [self setNeedsDisplay: YES]; } - (void) dealloc { if (_is_active == YES) { [self deactivate]; } TEST_RELEASE(_the_color); [self unregisterDraggedTypes]; [super dealloc]; } - (NSDragOperation) draggingEntered: (id )sender { NSPasteboard *pb; NSDragOperation sourceDragMask; NSDebugLLog(@"NSColorWell", @"%@: draggingEntered", self); if ([self isEnabled] == NO) return NSDragOperationNone; sourceDragMask = [sender draggingSourceOperationMask]; pb = [sender draggingPasteboard]; if ([[pb types] indexOfObject: NSColorPboardType] != NSNotFound) { if (sourceDragMask & NSDragOperationCopy) { return NSDragOperationCopy; } } return NSDragOperationNone; } - (NSDragOperation) draggingSourceOperationMaskForLocal: (BOOL)flag { return NSDragOperationCopy; } - (void) drawRect: (NSRect)clipRect { if (NSIntersectsRect(_bounds, clipRect) == NO) { return; } _wellRect = [[GSTheme theme] drawColorWellBorder: self withBounds: _bounds withClip: clipRect]; [self drawWellInside: _wellRect]; } /**

Draws the NSColorWell inside the rectangle insideRect.

See Also: [NSColor-drawSwatchInRect:]

*/ - (void) drawWellInside: (NSRect)insideRect { if (NSIsEmptyRect(insideRect)) { return; } [_the_color drawSwatchInRect: insideRect]; } - (void) encodeWithCoder: (NSCoder*)aCoder { [super encodeWithCoder: aCoder]; if ([aCoder allowsKeyedCoding]) { [aCoder encodeObject: _the_color forKey: @"NSColor"]; // [aCoder encodeBool: _is_active forKey: @"NSEnabled"]; [aCoder encodeBool: _is_bordered forKey: @"NSIsBordered"]; [aCoder encodeConditionalObject: _target forKey: @"NSTarget"]; [aCoder encodeConditionalObject: NSStringFromSelector(_action) forKey: @"NSAction"]; } else { [aCoder encodeObject: _the_color]; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &_is_active]; [aCoder encodeValueOfObjCType: @encode(BOOL) at: &_is_bordered]; [aCoder encodeConditionalObject: _target]; [aCoder encodeValueOfObjCType: @encode(SEL) at: &_action]; } } - (id) initWithCoder: (NSCoder*)aDecoder { self = [super initWithCoder: aDecoder]; if (self != nil) { if ([aDecoder allowsKeyedCoding]) { NSString *action; ASSIGN(_the_color, [aDecoder decodeObjectForKey: @"NSColor"]); // _is_active = [aDecoder decodeBoolForKey: @"NSEnabled"]; _is_bordered = [aDecoder decodeBoolForKey: @"NSIsBordered"]; _target = [aDecoder decodeObjectForKey: @"NSTarget"]; action = [aDecoder decodeObjectForKey: @"NSAction"]; _action = NSSelectorFromString(action); [self registerForDraggedTypes: [NSArray arrayWithObjects: NSColorPboardType, nil]]; } else { [aDecoder decodeValueOfObjCType: @encode(id) at: &_the_color]; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &_is_active]; [aDecoder decodeValueOfObjCType: @encode(BOOL) at: &_is_bordered]; [aDecoder decodeValueOfObjCType: @encode(id) at: &_target]; // Undo RETAIN by decoder TEST_RELEASE(_target); [aDecoder decodeValueOfObjCType: @encode(SEL) at: &_action]; [self registerForDraggedTypes: [NSArray arrayWithObjects: NSColorPboardType, nil]]; } } return self; } - (id) initWithFrame: (NSRect)frameRect { self = [super initWithFrame: frameRect]; if (self != nil) { _is_bordered = YES; _is_active = NO; _the_color = RETAIN([NSColor blackColor]); [self registerForDraggedTypes: [NSArray arrayWithObjects: NSColorPboardType, nil]]; } return self; } /**

Returns whether the NSColorWell is active. By default a NSColorWell is not active.

See Also: -activate: -deactivate

*/ - (BOOL) isActive { return _is_active; } /**

Returns whether the NSColorWell has border. By default a NSColorWell has border.

See Also: -setBordered:

*/ - (BOOL) isBordered { return _is_bordered; } - (BOOL) isOpaque { // May not be opaque, due to themes return NO; } - (void) mouseDown: (NSEvent *)theEvent { // // OPENSTEP 4.2 and OSX behavior indicates that the colorwell doesn't // work when the widget is marked as disabled. // if ([self isEnabled] == NO) return; // Unbordered color wells start a drag immediately upon mouse down if ([self isBordered] == NO) { [NSColorPanel dragColor: _the_color withEvent: theEvent fromView: self]; return; } _mouseDownPoint = [self convertPoint: [theEvent locationInWindow] fromView: nil]; [[self cell] setHighlighted: YES]; [self setNeedsDisplay: YES]; } - (void) mouseDragged: (NSEvent *)theEvent { NSPoint point = [self convertPoint: [theEvent locationInWindow] fromView: nil]; BOOL inside = [self mouse: point inRect: [self bounds]]; BOOL startedInWell = [self mouse: _mouseDownPoint inRect: _wellRect]; NSSize delta = NSMakeSize(_mouseDownPoint.x - point.x, _mouseDownPoint.y - point.y); double distance = sqrt(delta.width*delta.width + delta.height*delta.height); // FIXME: Make the dragging threshold a user default if (distance < 4) return; if ([self isEnabled] == NO) return; if (startedInWell) { [[self cell] setHighlighted: NO]; [self setNeedsDisplay: YES]; [NSColorPanel dragColor: _the_color withEvent: theEvent fromView: self]; return; } else { [[self cell] setHighlighted: inside]; [self setNeedsDisplay: YES]; } } - (void) mouseUp: (NSEvent *)theEvent { NSPoint point = [self convertPoint: [theEvent locationInWindow] fromView: nil]; BOOL inside = [self mouse: point inRect: [self bounds]]; if ([self isEnabled] == NO) return; [[self cell] setHighlighted: NO]; [self setNeedsDisplay: YES]; if (inside) { [self performClick: self]; } } - (id) objectValue { return [self color]; } - (void) performClick: (id)sender { if ([self isActive]) { [self deactivate]; } else { [self activate: YES]; } } - (BOOL) performDragOperation: (id )sender { NSPasteboard *pb = [sender draggingPasteboard]; NSDebugLLog(@"NSColorWell", @"%@: performDragOperation", self); [self setColor: [NSColor colorFromPasteboard: pb]]; /* When our color is changed by having a new color dropped on us, * we send our action. */ [self sendAction: _action to: _target]; return YES; } - (BOOL) prepareForDragOperation: (id )sender { return YES; } - (void) setAction: (SEL)action { _action = action; } /**

Sets whether the NSColorWell has border and marks self for display. By default a NSColorWell has border.

See Also: -isBordered

*/ - (void) setBordered: (BOOL)bordered { _is_bordered = bordered; [self setNeedsDisplay: YES]; } /**

Sets the NSColorWell to color and marks self for display.
* Sets the NSColorPanel if active.
* Does NOT notify target of color change. *

*

See Also: -color

*/ - (void) setColor: (NSColor *)color { ASSIGN(_the_color, color); [self setNeedsDisplay: YES]; /* * Experimentation with NeXTstep shows that when the color of an active * colorwell is set, the color of the shared color panel is set too, * though this does not raise the color panel, only the event of * activation does that. */ if ([self isActive]) { NSColorPanel *colorPanel = [NSColorPanel sharedColorPanel]; [colorPanel setColor: _the_color]; } } - (void) setObjectValue: (id)anObject { [self setColor: anObject]; } - (void) setTarget: (id)target { _target = target; } /**

Sets the NSColorWell's color to the sender color.

*

See Also: -setColor:

*/ - (void) takeColorFrom: (id)sender { if ([sender respondsToSelector: @selector(color)]) { [self setColor: [sender color]]; } } - (void) _takeColorFromPanel: (NSNotification *) notification { id sender = [notification object]; if ([sender respondsToSelector: @selector(color)]) { NSColor *c = [(id)sender color]; /* Don't use -setColor: as that would send a message back to the * panel telling it to se its color again. * Instead we assign the color and mark for redisplay directly. * NB. For MacOS-X compatibility, we only send the action if the * coor has actually changed. */ if (c != nil && [c isEqual: _the_color] == NO) { ASSIGN(_the_color, [(id)sender color]); [self setNeedsDisplay: YES]; /* When our color is changed from the color panel, we should * send our action. */ [self sendAction: _action to: _target]; } } } - (void) takeObjectValueFrom: (id)sender { [self takeColorFrom: sender]; } - (id) target { return _target; } @end