libs-gui/Source/NSColorWell.m
Fred Kiefer 2d01942e4b Replace #include -> #import
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@32451 72102866-910b-0410-8b05-ffd578937521
2011-03-04 11:33:22 +00:00

499 lines
13 KiB
Objective-C

/** <title>NSColorWell</title>
<abstract>Control for selecting and display a single color value.</abstract>
Copyright (C) 1996 Free Software Foundation, Inc.
Author: Scott Christley <scottc@net-community.com>
Date: 1996
Author: Felipe A. Rodriguez <far@ix.netcom.com>
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 <http://www.gnu.org/licenses/> 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 <Foundation/NSDebug.h>
#import <Foundation/NSNotification.h>
#include <math.h>
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;
}
/**<p>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).</p><p>See Also: -deactivate</p>
*/
- (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];
}
/**<p> Returns the current NSColor of the NSColorWell.</p>
<p> See Also: -setColor:</p>
*/
- (NSColor *) color
{
return _the_color;
}
/** <p>Deactivates the NSColorWell and marks self for display.
It is usally call from an observer, when another NSColorWell is
activate.</p><p>See Also: -activate:</p>
*/
- (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 <NSDraggingInfo>)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: NSIntersectionRect(_wellRect, clipRect)];
}
/**<p>Draws the NSColorWell inside the rectangle <var>insideRect</var>.</p>
<p>See Also: [NSColor-drawSwatchInRect:]</p>
*/
- (void) drawWellInside: (NSRect)insideRect
{
if (NSIsEmptyRect(insideRect))
{
return;
}
if ([_the_color alphaComponent] < 1.0)
{
NSBezierPath *triangle = [NSBezierPath bezierPath];
[[NSColor whiteColor] set];
NSRectFill(insideRect);
[triangle moveToPoint: NSMakePoint(insideRect.origin.x,
insideRect.origin.y + insideRect.size.height)];
[triangle lineToPoint: NSMakePoint(insideRect.origin.x + insideRect.size.width,
insideRect.origin.y + insideRect.size.height)];
[triangle lineToPoint: insideRect.origin];
[triangle closePath];
[[NSColor blackColor] set];
[triangle fill];
}
[_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;
}
/** <p>Returns whether the NSColorWell is active. By default a NSColorWell
is not active.</p>
<p>See Also: -activate: -deactivate</p>
*/
- (BOOL) isActive
{
return _is_active;
}
/** <p>Returns whether the NSColorWell has border. By default a NSColorWell
has border.</p><p>See Also: -setBordered:</p>
*/
- (BOOL) isBordered
{
return _is_bordered;
}
- (BOOL) isOpaque
{
return _is_bordered;
}
- (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)
{
if (_is_active == NO)
{
[self activate: YES];
}
else
{
[self deactivate];
}
}
}
- (BOOL) performDragOperation: (id <NSDraggingInfo>)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 <NSDraggingInfo>)sender
{
return YES;
}
- (void) setAction: (SEL)action
{
_action = action;
}
/**<p>Sets whether the NSColorWell has border and marks self for display.
By default a NSColorWell has border.</p><p>See Also: -isBordered</p>
*/
- (void) setBordered: (BOOL)bordered
{
_is_bordered = bordered;
[self setNeedsDisplay: YES];
}
/** <p>Sets the NSColorWell to color and marks self for display.<br />
* Sets the NSColorPanel if active.<br />
* Does NOT notify target of color change.
* </p>
* <p>See Also: -color</p>
*/
- (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) setTarget: (id)target
{
_target = target;
}
/** <p>Sets the NSColorWell's color to the sender color.</p>
* <p>See Also: -setColor: </p>
*/
- (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];
}
}
}
- (id) target
{
return _target;
}
@end