mirror of
https://github.com/gnustep/libs-gui.git
synced 2025-04-25 09:11:03 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/gui/trunk@4497 72102866-910b-0410-8b05-ffd578937521
624 lines
12 KiB
Objective-C
624 lines
12 KiB
Objective-C
/*
|
|
NSPopUpButton.m
|
|
|
|
Popup list class
|
|
|
|
Copyright (C) 1996 Free Software Foundation, Inc.
|
|
|
|
Author: Scott Christley <scottc@net-community.com>
|
|
Date: 1996
|
|
Author: Michael Hanni <mhanni@sprintmail.com>
|
|
Date: June 1999
|
|
|
|
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 Library 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with this library; see the file COPYING.LIB.
|
|
If not, write to the Free Software Foundation,
|
|
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <gnustep/gui/config.h>
|
|
#import <Foundation/Foundation.h>
|
|
#include <AppKit/NSPopUpButton.h>
|
|
#include <AppKit/NSPopUpButtonCell.h>
|
|
#include <AppKit/NSApplication.h>
|
|
#include <AppKit/NSMenu.h>
|
|
#include <AppKit/NSFont.h>
|
|
|
|
@implementation NSPopUpButtonMatrix
|
|
|
|
// Class variables
|
|
static NSFont* menuFont = nil;
|
|
|
|
- (BOOL) acceptsFirstMouse: (NSEvent *)theEvent
|
|
{
|
|
return YES;
|
|
}
|
|
|
|
- (id) initWithFrame: (NSRect)rect
|
|
{
|
|
[super initWithFrame: rect];
|
|
cells = [NSMutableArray new];
|
|
|
|
selected_cell = 0;
|
|
|
|
cellSize = NSMakeSize (rect.size.width, rect.size.height);
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void) dealloc
|
|
{
|
|
NSDebugLog (@"NSMenuMatrix of menu '%@' dealloc", [menu title]);
|
|
|
|
[cells release];
|
|
[super dealloc];
|
|
}
|
|
|
|
- (id) copyWithZone: (NSZone*)zone
|
|
{
|
|
NSPopUpButtonMatrix* copy = [[isa allocWithZone: zone] initWithFrame: [self frame]];
|
|
int i, count;
|
|
|
|
NSDebugLog (@"copy menu matrix of menu with title '%@'", [menu title]);
|
|
for (i = 0, count = [cells count]; i < count; i++)
|
|
{
|
|
id aCell = [cells objectAtIndex: i];
|
|
id cellCopy = [[aCell copyWithZone: zone] autorelease];
|
|
|
|
[copy->cells addObject: cellCopy];
|
|
}
|
|
|
|
copy->cellSize = cellSize;
|
|
copy->menu = menu;
|
|
if (selectedCell)
|
|
{
|
|
int index = [cells indexOfObject: selectedCell];
|
|
|
|
copy->selectedCell = [[cells objectAtIndex: index] retain];
|
|
}
|
|
copy->selectedCellRect = selectedCellRect;
|
|
|
|
return copy;
|
|
}
|
|
|
|
- (void) _resizeMenuForCellSize
|
|
{
|
|
int i, count;
|
|
float titleWidth;
|
|
|
|
/* Compute the new width of the menu cells matrix */
|
|
cellSize.width = 0;
|
|
count = [cells count];
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
titleWidth = [menuFont widthOfString:
|
|
[[cells objectAtIndex: i] stringValue]];
|
|
cellSize.width = MAX(titleWidth + ADDITIONAL_WIDTH, cellSize.width);
|
|
}
|
|
cellSize.width = MAX([menuFont widthOfString: [menu title]]
|
|
+ ADDITIONAL_WIDTH,
|
|
cellSize.width);
|
|
|
|
/* Resize the frame to hold all the menu cells */
|
|
[super setFrameSize: NSMakeSize (cellSize.width,
|
|
count ? (cellSize.height + INTERCELL_SPACE)*count - INTERCELL_SPACE : 0)];
|
|
}
|
|
|
|
- (id <NSMenuItem>)insertItemWithTitle: (NSString*)aString
|
|
action: (SEL)aSelector
|
|
keyEquivalent: (NSString*)charCode
|
|
atIndex: (unsigned int)index
|
|
{
|
|
id menuCell = [[NSPopUpButtonCell new] autorelease];
|
|
|
|
[menuCell setFont:[NSFont systemFontOfSize:12]];
|
|
[menuCell setTitle: aString];
|
|
// [menuCell setAction: aSelector];
|
|
// [menuCell setKeyEquivalent: charCode];
|
|
|
|
[cells insertObject: menuCell atIndex: index];
|
|
|
|
return menuCell;
|
|
}
|
|
|
|
- (void) setIndexOfSelectedItem:(int)itemNum
|
|
{
|
|
selected_cell = itemNum;
|
|
}
|
|
|
|
- (NSString *)titleOfSelectedItem
|
|
{
|
|
return [[cells objectAtIndex:selected_cell] title];
|
|
}
|
|
|
|
- (void)setPopUpButton:(NSPopUpButton *)popb
|
|
{
|
|
ASSIGN(popup_button, popb);
|
|
}
|
|
|
|
- (void)setPullsDown:(BOOL)pull
|
|
{
|
|
pull_down = pull;
|
|
}
|
|
|
|
- (BOOL)pullsDown
|
|
{
|
|
return pull_down;
|
|
}
|
|
|
|
- (BOOL) performKeyEquivalent: (NSEvent*)theEvent
|
|
{
|
|
return [menu performKeyEquivalent: theEvent];
|
|
}
|
|
|
|
- (void) removeItem: (id <NSMenuItem>)anItem
|
|
{
|
|
int row = [cells indexOfObject: anItem];
|
|
|
|
if (row == -1)
|
|
return;
|
|
|
|
[cells removeObjectAtIndex: row];
|
|
}
|
|
|
|
- (NSArray*) itemArray
|
|
{
|
|
return cells;
|
|
}
|
|
|
|
- (id <NSMenuItem>) itemWithTitle: (NSString*)aString
|
|
{
|
|
unsigned i, count = [cells count];
|
|
id menuCell;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
menuCell = [cells objectAtIndex: i];
|
|
if ([[menuCell title] isEqual: aString])
|
|
return menuCell;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
- (id <NSMenuItem>) itemWithTag: (int)aTag
|
|
{
|
|
unsigned i, count = [cells count];
|
|
id menuCell;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
menuCell = [cells objectAtIndex: i];
|
|
if ([menuCell tag] == aTag)
|
|
return menuCell;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
- (NSRect) cellFrameAtRow: (int)index
|
|
{
|
|
unsigned count = [cells count];
|
|
NSRect rect;
|
|
|
|
NSAssert(index >= 0 && index < count+1, @"invalid row coordinate");
|
|
|
|
rect.origin.x = 0;
|
|
rect.origin.y = (count - index - 1)
|
|
* (cellSize.height + INTERCELL_SPACE);
|
|
rect.size = cellSize;
|
|
|
|
return rect;
|
|
}
|
|
|
|
- (void)drawRect: (NSRect)rect
|
|
{
|
|
unsigned i = 0, count = [cells count];
|
|
int max, howMany;
|
|
NSRect aRect = [self frame];
|
|
|
|
// If there are no cells then just return
|
|
if (count == 0) return;
|
|
|
|
aRect.origin.y = cellSize.height * (count - 1);
|
|
aRect.size = cellSize;
|
|
|
|
for (i=0;i<count;i++)
|
|
{
|
|
id aCell = [cells objectAtIndex:i];
|
|
|
|
[aCell drawWithFrame:aRect inView:self];
|
|
aRect.origin.y -= cellSize.height;
|
|
}
|
|
}
|
|
|
|
- (NSSize) cellSize
|
|
{
|
|
return cellSize;
|
|
}
|
|
|
|
- (void) setMenu: (NSMenu*)anObject
|
|
{
|
|
menu = anObject;
|
|
}
|
|
|
|
- (void) setSelectedCell: (id)aCell
|
|
{
|
|
selectedCell = aCell;
|
|
}
|
|
|
|
- (id) selectedCell
|
|
{
|
|
return selectedCell;
|
|
}
|
|
|
|
- (NSRect) selectedCellRect
|
|
{
|
|
return selectedCellRect;
|
|
}
|
|
@end
|
|
|
|
//
|
|
// NSPopUpButton implementation
|
|
//
|
|
@implementation NSPopUpButton
|
|
|
|
///////////////////////////////////////////////////////////////
|
|
//
|
|
// Class methods
|
|
//
|
|
+ (void)initialize
|
|
{
|
|
if (self == [NSPopUpButton class])
|
|
{
|
|
// Initial version
|
|
[self setVersion:1];
|
|
}
|
|
}
|
|
|
|
//
|
|
// Initializing an NSPopUpButton
|
|
//
|
|
- init
|
|
{
|
|
return [self initWithFrame:NSZeroRect pullsDown:NO];
|
|
}
|
|
|
|
- (id)initWithFrame:(NSRect)frameRect
|
|
{
|
|
return [self initWithFrame:frameRect pullsDown:NO];
|
|
}
|
|
|
|
- (id)initWithFrame:(NSRect)frameRect
|
|
pullsDown:(BOOL)flag
|
|
{
|
|
[super initWithFrame:frameRect];
|
|
list_items = [[NSPopUpButtonMatrix alloc] initWithFrame:frameRect];
|
|
[list_items setPopUpButton:self];
|
|
[list_items setPullsDown:flag];
|
|
is_up = NO;
|
|
pulls_down = flag;
|
|
selected_item = 0;
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void) dealloc
|
|
{
|
|
[list_items release];
|
|
[super dealloc];
|
|
}
|
|
|
|
//
|
|
// Target and Action
|
|
//
|
|
- (SEL)action
|
|
{
|
|
return pub_action;
|
|
}
|
|
|
|
- (void)setAction:(SEL)aSelector
|
|
{
|
|
pub_action = aSelector;
|
|
}
|
|
|
|
- (id)target
|
|
{
|
|
return pub_target;
|
|
}
|
|
|
|
- (void)setTarget:(id)anObject
|
|
{
|
|
pub_target = anObject;
|
|
}
|
|
|
|
- (void)buttonSelected:(id)sender
|
|
{
|
|
if (!pulls_down)
|
|
selected_item = [self indexOfItemWithTitle:[sender title]];
|
|
else
|
|
selected_item = 0;
|
|
|
|
[self synchronizeTitleAndSelectedItem];
|
|
|
|
[self drawRect:[self frame]];
|
|
[self setNeedsDisplay:YES];
|
|
|
|
if (pub_target && pub_action)
|
|
[pub_target performSelector:pub_action withObject:self];
|
|
}
|
|
|
|
//
|
|
// Adding Items
|
|
//
|
|
- (void)addItemWithTitle:(NSString *)title
|
|
{
|
|
[list_items insertItemWithTitle:title
|
|
action:nil
|
|
keyEquivalent:nil
|
|
atIndex:[[list_items itemArray] count]];
|
|
[self synchronizeTitleAndSelectedItem];
|
|
}
|
|
|
|
- (void)addItemsWithTitles:(NSArray *)itemTitles
|
|
{
|
|
int i, count = [itemTitles count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
[self addItemWithTitle:[itemTitles objectAtIndex:i]];
|
|
}
|
|
|
|
- (void)insertItemWithTitle:(NSString *)title
|
|
atIndex:(unsigned int)index
|
|
{
|
|
[list_items insertItemWithTitle:title
|
|
action:@selector(iveBeenHitCaptain:)
|
|
keyEquivalent:nil
|
|
atIndex:index];
|
|
[self synchronizeTitleAndSelectedItem];
|
|
}
|
|
|
|
//
|
|
// Removing Items
|
|
//
|
|
- (void)removeAllItems
|
|
{
|
|
[(NSMutableArray *)[list_items itemArray] removeAllObjects];
|
|
}
|
|
|
|
- (void)removeItemWithTitle:(NSString *)title
|
|
{
|
|
int index = [self indexOfItemWithTitle:title];
|
|
|
|
if (index != NSNotFound)
|
|
[(NSMutableArray *)[list_items itemArray] removeObjectAtIndex:index];
|
|
}
|
|
|
|
- (void)removeItemAtIndex:(int)index
|
|
{
|
|
[(NSMutableArray *)[list_items itemArray] removeObjectAtIndex:index];
|
|
}
|
|
|
|
//
|
|
// Querying the NSPopUpButton about Its Items
|
|
//
|
|
- (int)indexOfItemWithTitle:(NSString *)title
|
|
{
|
|
int i, count = [[list_items itemArray] count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
if ([[[[list_items itemArray] objectAtIndex:i] title] isEqual:title])
|
|
return i;
|
|
|
|
return NSNotFound;
|
|
}
|
|
|
|
- (int)indexOfSelectedItem
|
|
{
|
|
return selected_item;
|
|
}
|
|
|
|
- (int)numberOfItems
|
|
{
|
|
return [[list_items itemArray] count];
|
|
}
|
|
|
|
- (id <NSMenuItem>)itemAtIndex:(int)index
|
|
{
|
|
return [[list_items itemArray] objectAtIndex:index];
|
|
}
|
|
|
|
- (NSArray *)itemArray
|
|
{
|
|
return [list_items itemArray];
|
|
}
|
|
|
|
- (NSString *)itemTitleAtIndex:(int)index
|
|
{
|
|
return [[[list_items itemArray] objectAtIndex:index] title];
|
|
}
|
|
|
|
- (NSArray *)itemTitles
|
|
{
|
|
int i, count = [[list_items itemArray] count];
|
|
NSMutableArray* titles = [NSMutableArray arrayWithCapacity:count];
|
|
|
|
for (i = 0; i < count; i++)
|
|
[titles addObject:[[[list_items itemArray] objectAtIndex:i] title]];
|
|
|
|
return titles;
|
|
}
|
|
|
|
- (id <NSMenuItem>)itemWithTitle:(NSString *)title
|
|
{
|
|
int index = [self indexOfItemWithTitle:title];
|
|
|
|
if (index != NSNotFound)
|
|
return [[list_items itemArray] objectAtIndex:index];
|
|
return nil;
|
|
}
|
|
|
|
- (id <NSMenuItem>)lastItem
|
|
{
|
|
if ([[list_items itemArray] count])
|
|
return [[list_items itemArray] lastObject];
|
|
else
|
|
return nil;
|
|
}
|
|
|
|
- (id <NSMenuItem>)selectedItem
|
|
{
|
|
return [[list_items itemArray] objectAtIndex:selected_item];
|
|
}
|
|
|
|
- (NSString *)titleOfSelectedItem
|
|
{
|
|
return [[self selectedItem] title];
|
|
}
|
|
|
|
//
|
|
// Manipulating the NSPopUpButton
|
|
//
|
|
- (NSFont *)font
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
- (BOOL)pullsDown
|
|
{
|
|
return pulls_down;
|
|
}
|
|
|
|
- (void)selectItemAtIndex:(int)index
|
|
{
|
|
if ((index >= 0) && (index < [[list_items itemArray] count]))
|
|
{
|
|
selected_item = index;
|
|
[self synchronizeTitleAndSelectedItem];
|
|
}
|
|
}
|
|
|
|
- (void)selectItemWithTitle:(NSString *)title
|
|
{
|
|
int index = [self indexOfItemWithTitle:title];
|
|
|
|
if (index != NSNotFound)
|
|
[self selectItemAtIndex:index];
|
|
}
|
|
|
|
- (void)setFont:(NSFont *)fontObject
|
|
{}
|
|
|
|
- (void)setPullsDown:(BOOL)flag
|
|
{
|
|
pulls_down = flag;
|
|
}
|
|
|
|
- (void)setTitle:(NSString *)aString
|
|
{}
|
|
|
|
- (NSString *)stringValue
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
- (void)synchronizeTitleAndSelectedItem
|
|
{
|
|
if (!pulls_down)
|
|
[list_items setIndexOfSelectedItem:selected_item];
|
|
else
|
|
[list_items setIndexOfSelectedItem:0];
|
|
}
|
|
|
|
//
|
|
// Displaying the NSPopUpButton's Items
|
|
//
|
|
- (BOOL)autoenablesItems
|
|
{
|
|
return NO;
|
|
}
|
|
|
|
- (void)setAutoenablesItems:(BOOL)flag
|
|
{}
|
|
|
|
//
|
|
// Handle events
|
|
//
|
|
- (void)mouseDown:(NSEvent *)theEvent
|
|
{
|
|
}
|
|
|
|
- (void)mouseUp:(NSEvent *)theEvent
|
|
{
|
|
}
|
|
|
|
- (void)mouseMoved:(NSEvent *)theEvent
|
|
{
|
|
}
|
|
|
|
- (NSView *)hitTest:(NSPoint)aPoint
|
|
{
|
|
// First check ourselves
|
|
// if ([self mouse:aPoint inRect:bounds]) return self;
|
|
if ([self mouse:aPoint inRect:[self frame]]) return self;
|
|
|
|
return nil;
|
|
}
|
|
|
|
//
|
|
// Displaying
|
|
//
|
|
- (void)drawRect:(NSRect)rect
|
|
{
|
|
id aCell;
|
|
|
|
if (!pulls_down)
|
|
aCell = [[list_items itemArray] objectAtIndex:selected_item];
|
|
else
|
|
aCell = [[list_items itemArray] objectAtIndex:0];
|
|
|
|
[aCell drawWithFrame:rect inView:self];
|
|
}
|
|
|
|
//
|
|
// NSCoding protocol
|
|
//
|
|
- (void) encodeWithCoder: (NSCoder*)aCoder
|
|
{
|
|
[super encodeWithCoder: aCoder];
|
|
|
|
[aCoder encodeObject: list_items];
|
|
[aCoder encodeRect: list_rect];
|
|
[aCoder encodeValueOfObjCType: @encode(int) at: &selected_item];
|
|
[aCoder encodeConditionalObject: pub_target];
|
|
[aCoder encodeValueOfObjCType: @encode(SEL) at: &pub_action];
|
|
[aCoder encodeValueOfObjCType: @encode(BOOL) at: &is_up];
|
|
[aCoder encodeValueOfObjCType: @encode(BOOL) at: &pulls_down];
|
|
}
|
|
|
|
- (id) initWithCoder: (NSCoder*)aDecoder
|
|
{
|
|
[super initWithCoder: aDecoder];
|
|
|
|
[aDecoder decodeValueOfObjCType: @encode(id) at: &list_items];
|
|
list_rect = [aDecoder decodeRect];
|
|
[aDecoder decodeValueOfObjCType: @encode(int) at: &selected_item];
|
|
pub_target = [aDecoder decodeObject];
|
|
[aDecoder decodeValueOfObjCType: @encode(SEL) at: &pub_action];
|
|
[aDecoder decodeValueOfObjCType: @encode(BOOL) at: &is_up];
|
|
[aDecoder decodeValueOfObjCType: @encode(BOOL) at: &pulls_down];
|
|
|
|
return self;
|
|
}
|
|
|
|
@end
|