netsurf/cocoa/PSMTabBarControl/PSMTabBarControl.m

1991 lines
67 KiB

//
// PSMTabBarControl.m
// PSMTabBarControl
//
// Created by John Pannell on 10/13/05.
// Copyright 2005 Positive Spin Media. All rights reserved.
//
#import "PSMTabBarControl.h"
#import "PSMTabBarCell.h"
#import "PSMOverflowPopUpButton.h"
#import "PSMRolloverButton.h"
#import "PSMTabStyle.h"
#import "PSMUnifiedTabStyle.h"
#import "PSMTabDragAssistant.h"
#import "PSMTabBarController.h"
@interface PSMTabBarControl (Private)
// constructor/destructor
- (void)initAddedProperties;
// accessors
- (NSEvent *)lastMouseDownEvent;
- (void)setLastMouseDownEvent:(NSEvent *)event;
// contents
- (void)addTabViewItem:(NSTabViewItem *)item;
- (void)removeTabForCell:(PSMTabBarCell *)cell;
// draw
- (void)update;
- (void)update:(BOOL)animate;
- (void)_setupTrackingRectsForCell:(PSMTabBarCell *)cell;
- (void)_positionOverflowMenu;
- (void)_checkWindowFrame;
// actions
- (void)overflowMenuAction:(id)sender;
- (void)closeTabClick:(id)sender;
- (void)tabClick:(id)sender;
- (void)tabNothing:(id)sender;
// notification handlers
- (void)frameDidChange:(NSNotification *)notification;
- (void)windowDidMove:(NSNotification *)aNotification;
- (void)windowDidUpdate:(NSNotification *)notification;
// NSTabView delegate
- (void)tabView:(NSTabView *)tabView didSelectTabViewItem:(NSTabViewItem *)tabViewItem;
- (BOOL)tabView:(NSTabView *)tabView shouldSelectTabViewItem:(NSTabViewItem *)tabViewItem;
- (void)tabView:(NSTabView *)tabView willSelectTabViewItem:(NSTabViewItem *)tabViewItem;
- (void)tabViewDidChangeNumberOfTabViewItems:(NSTabView *)tabView;
// archiving
- (void)encodeWithCoder:(NSCoder *)aCoder;
- (id)initWithCoder:(NSCoder *)aDecoder;
// convenience
- (void)_bindPropertiesForCell:(PSMTabBarCell *)cell andTabViewItem:(NSTabViewItem *)item;
- (id)cellForPoint:(NSPoint)point cellFrame:(NSRectPointer)outFrame;
- (void)_animateCells:(NSTimer *)timer;
@end
@implementation PSMTabBarControl
#pragma mark -
#pragma mark Characteristics
+ (NSBundle *)bundle;
{
static NSBundle *bundle = nil;
if(!bundle) {
bundle = [NSBundle bundleForClass:[PSMTabBarControl class]];
}
return bundle;
}
/*!
@method availableCellWidth
@abstract The number of pixels available for cells
@discussion Calculates the number of pixels available for cells based on margins and the window resize badge.
@returns Returns the amount of space for cells.
*/
- (CGFloat)availableCellWidth {
return [self frame].size.width - [style leftMarginForTabBarControl] - [style rightMarginForTabBarControl] - _resizeAreaCompensation;
}
/*!
@method genericCellRect
@abstract The basic rect for a tab cell.
@discussion Creates a generic frame for a tab cell based on the current control state.
@returns Returns a basic rect for a tab cell.
*/
- (NSRect)genericCellRect {
NSRect aRect = [self frame];
aRect.origin.x = [style leftMarginForTabBarControl];
aRect.origin.y = 0.0;
aRect.size.width = [self availableCellWidth];
aRect.size.height = [style tabCellHeight];
return aRect;
}
#pragma mark -
#pragma mark Constructor/destructor
- (void)initAddedProperties {
_cells = [[NSMutableArray alloc] initWithCapacity:10];
_controller = [[PSMTabBarController alloc] initWithTabBarControl:self];
_animationTimer = nil;
// default config
_currentStep = kPSMIsNotBeingResized;
_orientation = PSMTabBarHorizontalOrientation;
_canCloseOnlyTab = NO;
_disableTabClose = NO;
_showAddTabButton = NO;
_hideForSingleTab = NO;
_sizeCellsToFit = NO;
_isHidden = NO;
_awakenedFromNib = NO;
_automaticallyAnimates = NO;
_useOverflowMenu = YES;
_allowsBackgroundTabClosing = YES;
_allowsResizing = NO;
_selectsTabsOnMouseDown = NO;
_alwaysShowActiveTab = NO;
_allowsScrubbing = NO;
_cellMinWidth = 100;
_cellMaxWidth = 280;
_cellOptimumWidth = 130;
_tearOffStyle = PSMTabBarTearOffAlphaWindow;
style = [[[[self class] defaultStyleClass] alloc] init];
// the overflow button/menu
NSRect overflowButtonRect = NSMakeRect([self frame].size.width - [style rightMarginForTabBarControl] + 1, 0, [style rightMarginForTabBarControl] - 1, [self frame].size.height);
_overflowPopUpButton = [[PSMOverflowPopUpButton alloc] initWithFrame:overflowButtonRect pullsDown:YES];
[_overflowPopUpButton setAutoresizingMask:NSViewNotSizable | NSViewMinXMargin];
[_overflowPopUpButton setHidden:YES];
[self addSubview:_overflowPopUpButton];
[self _positionOverflowMenu];
// new tab button
NSRect addTabButtonRect = NSMakeRect([self frame].size.width - [style rightMarginForTabBarControl] + 1, 3.0, 16.0, 16.0);
_addTabButton = [[PSMRolloverButton alloc] initWithFrame:addTabButtonRect];
if(_addTabButton) {
NSImage *newButtonImage = [style addTabButtonImage];
if(newButtonImage) {
[_addTabButton setUsualImage:newButtonImage];
}
newButtonImage = [style addTabButtonPressedImage];
if(newButtonImage) {
[_addTabButton setAlternateImage:newButtonImage];
}
newButtonImage = [style addTabButtonRolloverImage];
if(newButtonImage) {
[_addTabButton setRolloverImage:newButtonImage];
}
[_addTabButton setTitle:@""];
[_addTabButton setImagePosition:NSImageOnly];
[_addTabButton setButtonType:NSMomentaryChangeButton];
[_addTabButton setBordered:NO];
[_addTabButton setBezelStyle:NSShadowlessSquareBezelStyle];
[self addSubview:_addTabButton];
if(_showAddTabButton) {
[_addTabButton setHidden:NO];
} else {
[_addTabButton setHidden:YES];
}
[_addTabButton setNeedsDisplay:YES];
}
}
+ (Class) defaultStyleClass;
{
return [PSMUnifiedTabStyle class];
}
- (id)initWithFrame:(NSRect)frame {
self = [super initWithFrame:frame];
if(self) {
// Initialization
[self initAddedProperties];
[self registerForDraggedTypes:[NSArray arrayWithObjects:@"PSMTabBarControlItemPBType", nil]];
// resize
[self setPostsFrameChangedNotifications:YES];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(frameDidChange:) name:NSViewFrameDidChangeNotification object:self];
}
[self setTarget:self];
return self;
}
- (void)dealloc {
[[NSNotificationCenter defaultCenter] removeObserver:self];
//stop any animations that may be running
[_animationTimer invalidate];
[_animationTimer release]; _animationTimer = nil;
[_showHideAnimationTimer invalidate];
[_showHideAnimationTimer release]; _showHideAnimationTimer = nil;
//Also unwind the spring, if it's wound.
[_springTimer invalidate];
[_springTimer release]; _springTimer = nil;
//unbind all the items to prevent crashing
//not sure if this is necessary or not
// http://code.google.com/p/maccode/issues/detail?id=35
NSEnumerator *enumerator = [[[_cells copy] autorelease] objectEnumerator];
PSMTabBarCell *nextCell;
while((nextCell = [enumerator nextObject])) {
[self removeTabForCell:nextCell];
}
[_overflowPopUpButton release];
[_cells release];
[_controller release];
[tabView release];
[_addTabButton release];
[partnerView release];
[_lastMouseDownEvent release];
[style release];
[self unregisterDraggedTypes];
[super dealloc];
}
- (void)awakeFromNib {
// build cells from existing tab view items
NSArray *existingItems = [tabView tabViewItems];
NSEnumerator *e = [existingItems objectEnumerator];
NSTabViewItem *item;
while((item = [e nextObject])) {
if(![[self representedTabViewItems] containsObject:item]) {
[self addTabViewItem:item];
}
}
}
- (void)viewWillMoveToWindow:(NSWindow *)aWindow {
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
[center removeObserver:self name:NSWindowDidBecomeKeyNotification object:nil];
[center removeObserver:self name:NSWindowDidResignKeyNotification object:nil];
[center removeObserver:self name:NSWindowDidUpdateNotification object:nil];
[center removeObserver:self name:NSWindowDidMoveNotification object:nil];
if(_showHideAnimationTimer) {
[_showHideAnimationTimer invalidate];
[_showHideAnimationTimer release]; _showHideAnimationTimer = nil;
}
if(aWindow) {
[center addObserver:self selector:@selector(windowStatusDidChange:) name:NSWindowDidBecomeKeyNotification object:aWindow];
[center addObserver:self selector:@selector(windowStatusDidChange:) name:NSWindowDidResignKeyNotification object:aWindow];
[center addObserver:self selector:@selector(windowDidUpdate:) name:NSWindowDidUpdateNotification object:aWindow];
[center addObserver:self selector:@selector(windowDidMove:) name:NSWindowDidMoveNotification object:aWindow];
}
}
- (void)windowStatusDidChange:(NSNotification *)notification {
[self setNeedsDisplay:YES];
}
#pragma mark -
#pragma mark Accessors
- (NSMutableArray *)cells {
return _cells;
}
- (NSEvent *)lastMouseDownEvent {
return _lastMouseDownEvent;
}
- (void)setLastMouseDownEvent:(NSEvent *)event {
[event retain];
[_lastMouseDownEvent release];
_lastMouseDownEvent = event;
}
- (id)delegate {
return delegate;
}
- (void)setDelegate:(id)object {
delegate = object;
NSMutableArray *types = [NSMutableArray arrayWithObject:@"PSMTabBarControlItemPBType"];
//Update the allowed drag types
if([self delegate] && [[self delegate] respondsToSelector:@selector(allowedDraggedTypesForTabView:)]) {
[types addObjectsFromArray:[[self delegate] allowedDraggedTypesForTabView:tabView]];
}
[self unregisterDraggedTypes];
[self registerForDraggedTypes:types];
}
- (NSTabView *)tabView {
return tabView;
}
- (void)setTabView:(NSTabView *)view {
[view retain];
[tabView release];
tabView = view;
}
- (id<PSMTabStyle>)style {
return style;
}
- (NSString *)styleName {
return [style name];
}
- (void)setStyle:(id <PSMTabStyle>)newStyle {
if(style != newStyle) {
[style autorelease];
style = [newStyle retain];
// restyle add tab button
if(_addTabButton) {
NSImage *newButtonImage = [style addTabButtonImage];
if(newButtonImage) {
[_addTabButton setUsualImage:newButtonImage];
}
newButtonImage = [style addTabButtonPressedImage];
if(newButtonImage) {
[_addTabButton setAlternateImage:newButtonImage];
}
newButtonImage = [style addTabButtonRolloverImage];
if(newButtonImage) {
[_addTabButton setRolloverImage:newButtonImage];
}
}
[self update];
}
}
- (void)setStyleNamed:(NSString *)name {
Class styleClass = NSClassFromString( [NSString stringWithFormat: @"PSM%@TabStyle", [name capitalizedString]] );
if (styleClass == Nil) styleClass = [isa defaultStyleClass];
id <PSMTabStyle> newStyle = [[styleClass alloc] init];
[self setStyle:newStyle];
[newStyle release];
}
- (PSMTabBarOrientation)orientation {
return _orientation;
}
- (void)setOrientation:(PSMTabBarOrientation)value {
PSMTabBarOrientation lastOrientation = _orientation;
_orientation = value;
if(_tabBarWidth < 10) {
_tabBarWidth = 120;
}
if(lastOrientation != _orientation) {
[[self style] setOrientation:_orientation];
[self _positionOverflowMenu]; //move the overflow popup button to the right place
[self update:NO];
}
}
- (BOOL)canCloseOnlyTab {
return _canCloseOnlyTab;
}
- (void)setCanCloseOnlyTab:(BOOL)value {
_canCloseOnlyTab = value;
if([_cells count] == 1) {
[self update];
}
}
- (BOOL)disableTabClose {
return _disableTabClose;
}
- (void)setDisableTabClose:(BOOL)value {
_disableTabClose = value;
[self update];
}
- (BOOL)hideForSingleTab {
return _hideForSingleTab;
}
- (void)setHideForSingleTab:(BOOL)value {
_hideForSingleTab = value;
[self update];
}
- (BOOL)showAddTabButton {
return _showAddTabButton;
}
- (void)setShowAddTabButton:(BOOL)value {
_showAddTabButton = value;
if(!NSIsEmptyRect([_controller addButtonRect])) {
[_addTabButton setFrame:[_controller addButtonRect]];
}
[_addTabButton setHidden:!_showAddTabButton];
[_addTabButton setNeedsDisplay:YES];
[self update];
}
- (NSInteger)cellMinWidth {
return _cellMinWidth;
}
- (void)setCellMinWidth:(NSInteger)value {
_cellMinWidth = value;
[self update];
}
- (NSInteger)cellMaxWidth {
return _cellMaxWidth;
}
- (void)setCellMaxWidth:(NSInteger)value {
_cellMaxWidth = value;
[self update];
}
- (NSInteger)cellOptimumWidth {
return _cellOptimumWidth;
}
- (void)setCellOptimumWidth:(NSInteger)value {
_cellOptimumWidth = value;
[self update];
}
- (BOOL)sizeCellsToFit {
return _sizeCellsToFit;
}
- (void)setSizeCellsToFit:(BOOL)value {
_sizeCellsToFit = value;
[self update];
}
- (BOOL)useOverflowMenu {
return _useOverflowMenu;
}
- (void)setUseOverflowMenu:(BOOL)value {
_useOverflowMenu = value;
[self update];
}
- (PSMRolloverButton *)addTabButton {
return _addTabButton;
}
- (PSMOverflowPopUpButton *)overflowPopUpButton {
return _overflowPopUpButton;
}
- (BOOL)allowsBackgroundTabClosing {
return _allowsBackgroundTabClosing;
}
- (void)setAllowsBackgroundTabClosing:(BOOL)value {
_allowsBackgroundTabClosing = value;
}
- (BOOL)allowsResizing {
return _allowsResizing;
}
- (void)setAllowsResizing:(BOOL)value {
_allowsResizing = value;
}
- (BOOL)selectsTabsOnMouseDown {
return _selectsTabsOnMouseDown;
}
- (void)setSelectsTabsOnMouseDown:(BOOL)value {
_selectsTabsOnMouseDown = value;
}
- (BOOL)automaticallyAnimates {
return _automaticallyAnimates;
}
- (void)setAutomaticallyAnimates:(BOOL)value {
_automaticallyAnimates = value;
}
- (BOOL)alwaysShowActiveTab {
return _alwaysShowActiveTab;
}
- (void)setAlwaysShowActiveTab:(BOOL)value {
_alwaysShowActiveTab = value;
}
- (BOOL)allowsScrubbing {
return _allowsScrubbing;
}
- (void)setAllowsScrubbing:(BOOL)value {
_allowsScrubbing = value;
}
- (PSMTabBarTearOffStyle)tearOffStyle {
return _tearOffStyle;
}
- (void)setTearOffStyle:(PSMTabBarTearOffStyle)tearOffStyle {
_tearOffStyle = tearOffStyle;
}
#pragma mark -
#pragma mark Functionality
- (void)addTabViewItem:(NSTabViewItem *)item {
// create cell
PSMTabBarCell *cell = [[PSMTabBarCell alloc] initWithControlView:self];
NSRect cellRect, lastCellFrame;
if([_cells lastObject] != nil) {
cellRect = lastCellFrame = [[_cells lastObject] frame];
} else {
cellRect = lastCellFrame = NSZeroRect;
}
if([self orientation] == PSMTabBarHorizontalOrientation) {
cellRect = [self genericCellRect];
cellRect.size.width = 30;
cellRect.origin.x = lastCellFrame.origin.x + lastCellFrame.size.width;
} else {
cellRect = /*lastCellFrame*/ [self genericCellRect];
cellRect.size.width = lastCellFrame.size.width;
cellRect.size.height = 0;
cellRect.origin.y = lastCellFrame.origin.y + lastCellFrame.size.height;
}
[cell setRepresentedObject:item];
[cell setFrame:cellRect];
// bind it up
[self bindPropertiesForCell:cell andTabViewItem:item];
// add to collection
[_cells addObject:cell];
[cell release];
if([_cells count] == (NSUInteger)[tabView numberOfTabViewItems]) {
[self update]; // don't update unless all are accounted for!
}
}
- (void)removeTabForCell:(PSMTabBarCell *)cell {
NSTabViewItem *item = [cell representedObject];
// unbind
[[cell indicator] unbind:@"animate"];
[[cell indicator] unbind:@"hidden"];
[cell unbind:@"hasIcon"];
[cell unbind:@"hasLargeImage"];
[cell unbind:@"title"];
[cell unbind:@"count"];
[cell unbind:@"countColor"];
[cell unbind:@"isEdited"];
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(isProcessing)]) {
[[item identifier] removeObserver:cell forKeyPath:@"isProcessing"];
}
}
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(icon)]) {
[[item identifier] removeObserver:cell forKeyPath:@"icon"];
}
}
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(objectCount)]) {
[[item identifier] removeObserver:cell forKeyPath:@"objectCount"];
}
}
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(countColor)]) {
[[item identifier] removeObserver:cell forKeyPath:@"countColor"];
}
}
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(largeImage)]) {
[[item identifier] removeObserver:cell forKeyPath:@"largeImage"];
}
}
if([item identifier] != nil) {
if([[item identifier] respondsToSelector:@selector(isEdited)]) {
[[item identifier] removeObserver:cell forKeyPath:@"isEdited"];
}
}
// stop watching identifier
[item removeObserver:self forKeyPath:@"identifier"];
// remove indicator
if([[self subviews] containsObject:[cell indicator]]) {
[[cell indicator] removeFromSuperview];
}
// remove tracking
[[NSNotificationCenter defaultCenter] removeObserver:cell];
if([cell closeButtonTrackingTag] != 0) {
[self removeTrackingRect:[cell closeButtonTrackingTag]];
[cell setCloseButtonTrackingTag:0];
}
if([cell cellTrackingTag] != 0) {
[self removeTrackingRect:[cell cellTrackingTag]];
[cell setCellTrackingTag:0];
}
// pull from collection
[_cells removeObject:cell];
[self update];
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
// did the tab's identifier change?
if([keyPath isEqualToString:@"identifier"]) {
NSEnumerator *e = [_cells objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
if([cell representedObject] == object) {
[self _bindPropertiesForCell:cell andTabViewItem:object];
}
}
}
}
#pragma mark -
#pragma mark Hide/Show
- (void)hideTabBar:(BOOL)hide animate:(BOOL)animate {
if(!_awakenedFromNib || (_isHidden && hide) || (!_isHidden && !hide) || (_currentStep != kPSMIsNotBeingResized)) {
return;
}
[[self subviews] makeObjectsPerformSelector:@selector(removeFromSuperview)];
_isHidden = hide;
_currentStep = 0;
if(!animate) {
_currentStep = (NSInteger)kPSMHideAnimationSteps;
}
if(hide) {
[_overflowPopUpButton removeFromSuperview];
[_addTabButton removeFromSuperview];
} else if(!animate) {
[self addSubview:_overflowPopUpButton];
[self addSubview:_addTabButton];
}
CGFloat partnerOriginalSize, partnerOriginalOrigin, myOriginalSize, myOriginalOrigin, partnerTargetSize, partnerTargetOrigin, myTargetSize, myTargetOrigin;
// target values for partner
if([self orientation] == PSMTabBarHorizontalOrientation) {
// current (original) values
myOriginalSize = [self frame].size.height;
myOriginalOrigin = [self frame].origin.y;
if(partnerView) {
partnerOriginalSize = [partnerView frame].size.height;
partnerOriginalOrigin = [partnerView frame].origin.y;
} else {
partnerOriginalSize = [[self window] frame].size.height;
partnerOriginalOrigin = [[self window] frame].origin.y;
}
if(partnerView) {
// above or below me?
if((myOriginalOrigin - 22) > partnerOriginalOrigin) {
// partner is below me
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin + 21;
myTargetSize = myOriginalSize - 21;
partnerTargetOrigin = partnerOriginalOrigin;
partnerTargetSize = partnerOriginalSize + 21;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin - 21;
myTargetSize = myOriginalSize + 21;
partnerTargetOrigin = partnerOriginalOrigin;
partnerTargetSize = partnerOriginalSize - 21;
}
} else {
// partner is above me
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin;
myTargetSize = myOriginalSize - 21;
partnerTargetOrigin = partnerOriginalOrigin - 21;
partnerTargetSize = partnerOriginalSize + 21;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin;
myTargetSize = myOriginalSize + 21;
partnerTargetOrigin = partnerOriginalOrigin + 21;
partnerTargetSize = partnerOriginalSize - 21;
}
}
} else {
// for window movement
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin;
myTargetSize = myOriginalSize - 21;
partnerTargetOrigin = partnerOriginalOrigin + 21;
partnerTargetSize = partnerOriginalSize - 21;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin;
myTargetSize = myOriginalSize + 21;
partnerTargetOrigin = partnerOriginalOrigin - 21;
partnerTargetSize = partnerOriginalSize + 21;
}
}
} else { /* vertical */
// current (original) values
myOriginalSize = [self frame].size.width;
myOriginalOrigin = [self frame].origin.x;
if(partnerView) {
partnerOriginalSize = [partnerView frame].size.width;
partnerOriginalOrigin = [partnerView frame].origin.x;
} else {
partnerOriginalSize = [[self window] frame].size.width;
partnerOriginalOrigin = [[self window] frame].origin.x;
}
if(partnerView) {
//to the left or right?
if(myOriginalOrigin < partnerOriginalOrigin + partnerOriginalSize) {
// partner is to the left
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin;
myTargetSize = 1;
partnerTargetOrigin = partnerOriginalOrigin - myOriginalSize + 1;
partnerTargetSize = partnerOriginalSize + myOriginalSize - 1;
_tabBarWidth = myOriginalSize;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin;
myTargetSize = myOriginalSize + _tabBarWidth;
partnerTargetOrigin = partnerOriginalOrigin + _tabBarWidth;
partnerTargetSize = partnerOriginalSize - _tabBarWidth;
}
} else {
// partner is to the right
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin + myOriginalSize;
myTargetSize = 1;
partnerTargetOrigin = partnerOriginalOrigin;
partnerTargetSize = partnerOriginalSize + myOriginalSize;
_tabBarWidth = myOriginalSize;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin - _tabBarWidth;
myTargetSize = myOriginalSize + _tabBarWidth;
partnerTargetOrigin = partnerOriginalOrigin;
partnerTargetSize = partnerOriginalSize - _tabBarWidth;
}
}
} else {
// for window movement
if(_isHidden) {
// I'm shrinking
myTargetOrigin = myOriginalOrigin;
myTargetSize = 1;
partnerTargetOrigin = partnerOriginalOrigin + myOriginalSize - 1;
partnerTargetSize = partnerOriginalSize - myOriginalSize + 1;
_tabBarWidth = myOriginalSize;
} else {
// I'm growing
myTargetOrigin = myOriginalOrigin;
myTargetSize = _tabBarWidth;
partnerTargetOrigin = partnerOriginalOrigin - _tabBarWidth + 1;
partnerTargetSize = partnerOriginalSize + _tabBarWidth - 1;
}
}
if(!_isHidden && [[self delegate] respondsToSelector:@selector(desiredWidthForVerticalTabBar:)]) {
myTargetSize = [[self delegate] desiredWidthForVerticalTabBar:self];
}
}
NSDictionary *userInfo = [NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithDouble:myOriginalOrigin], @"myOriginalOrigin", [NSNumber numberWithDouble:partnerOriginalOrigin], @"partnerOriginalOrigin", [NSNumber numberWithDouble:myOriginalSize], @"myOriginalSize", [NSNumber numberWithDouble:partnerOriginalSize], @"partnerOriginalSize", [NSNumber numberWithDouble:myTargetOrigin], @"myTargetOrigin", [NSNumber numberWithDouble:partnerTargetOrigin], @"partnerTargetOrigin", [NSNumber numberWithDouble:myTargetSize], @"myTargetSize", [NSNumber numberWithDouble:partnerTargetSize], @"partnerTargetSize", nil];
if(_showHideAnimationTimer) {
[_showHideAnimationTimer invalidate];
[_showHideAnimationTimer release];
}
_showHideAnimationTimer = [[NSTimer scheduledTimerWithTimeInterval:(1.0 / 30.0) target:self selector:@selector(animateShowHide:) userInfo:userInfo repeats:YES] retain];
}
- (void)animateShowHide:(NSTimer *)timer {
// moves the frame of the tab bar and window (or partner view) linearly to hide or show the tab bar
NSRect myFrame = [self frame];
NSDictionary *userInfo = [timer userInfo];
CGFloat myCurrentOrigin = ([[userInfo objectForKey:@"myOriginalOrigin"] doubleValue] + (([[userInfo objectForKey:@"myTargetOrigin"] doubleValue] - [[userInfo objectForKey:@"myOriginalOrigin"] doubleValue]) * (_currentStep / kPSMHideAnimationSteps)));
CGFloat myCurrentSize = ([[userInfo objectForKey:@"myOriginalSize"] doubleValue] + (([[userInfo objectForKey:@"myTargetSize"] doubleValue] - [[userInfo objectForKey:@"myOriginalSize"] doubleValue]) * (_currentStep / kPSMHideAnimationSteps)));
CGFloat partnerCurrentOrigin = ([[userInfo objectForKey:@"partnerOriginalOrigin"] doubleValue] + (([[userInfo objectForKey:@"partnerTargetOrigin"] doubleValue] - [[userInfo objectForKey:@"partnerOriginalOrigin"] doubleValue]) * (_currentStep / kPSMHideAnimationSteps)));
CGFloat partnerCurrentSize = ([[userInfo objectForKey:@"partnerOriginalSize"] doubleValue] + (([[userInfo objectForKey:@"partnerTargetSize"] doubleValue] - [[userInfo objectForKey:@"partnerOriginalSize"] doubleValue]) * (_currentStep / kPSMHideAnimationSteps)));
NSRect myNewFrame;
if([self orientation] == PSMTabBarHorizontalOrientation) {
myNewFrame = NSMakeRect(myFrame.origin.x, myCurrentOrigin, myFrame.size.width, myCurrentSize);
} else {
myNewFrame = NSMakeRect(myCurrentOrigin, myFrame.origin.y, myCurrentSize, myFrame.size.height);
}
if(partnerView) {
// resize self and view
NSRect resizeRect;
if([self orientation] == PSMTabBarHorizontalOrientation) {
resizeRect = NSMakeRect([partnerView frame].origin.x, partnerCurrentOrigin, [partnerView frame].size.width, partnerCurrentSize);
} else {
resizeRect = NSMakeRect(partnerCurrentOrigin, [partnerView frame].origin.y, partnerCurrentSize, [partnerView frame].size.height);
}
[partnerView setFrame:resizeRect];
[partnerView setNeedsDisplay:YES];
[self setFrame:myNewFrame];
} else {
// resize self and window
NSRect resizeRect;
if([self orientation] == PSMTabBarHorizontalOrientation) {
resizeRect = NSMakeRect([[self window] frame].origin.x, partnerCurrentOrigin, [[self window] frame].size.width, partnerCurrentSize);
} else {
resizeRect = NSMakeRect(partnerCurrentOrigin, [[self window] frame].origin.y, partnerCurrentSize, [[self window] frame].size.height);
}
[[self window] setFrame:resizeRect display:YES];
[self setFrame:myNewFrame];
}
// next
_currentStep++;
if(_currentStep == kPSMHideAnimationSteps + 1) {
_currentStep = kPSMIsNotBeingResized;
[self viewDidEndLiveResize];
[self update:NO];
//send the delegate messages
if(_isHidden) {
if([[self delegate] respondsToSelector:@selector(tabView:tabBarDidHide:)]) {
[[self delegate] tabView:[self tabView] tabBarDidHide:self];
}
} else {
[self addSubview:_overflowPopUpButton];
[self addSubview:_addTabButton];
if([[self delegate] respondsToSelector:@selector(tabView:tabBarDidUnhide:)]) {
[[self delegate] tabView:[self tabView] tabBarDidUnhide:self];
}
}
[_showHideAnimationTimer invalidate];
[_showHideAnimationTimer release]; _showHideAnimationTimer = nil;
}
[[self window] display];
}
- (BOOL)isTabBarHidden {
return _isHidden;
}
- (BOOL)isAnimating {
return _animationTimer != nil;
}
- (id)partnerView {
return partnerView;
}
- (void)setPartnerView:(id)view {
[partnerView release];
[view retain];
partnerView = view;
}
#pragma mark -
#pragma mark Drawing
- (BOOL)isFlipped {
return YES;
}
- (void)drawRect:(NSRect)rect {
[style drawTabBar:self inRect:rect];
}
- (void)update {
[self update:_automaticallyAnimates];
}
- (void)update:(BOOL)animate {
// make sure all of our tabs are accounted for before updating
if((NSUInteger)[[self tabView] numberOfTabViewItems] != [_cells count]) {
return;
}
// hide/show? (these return if already in desired state)
if((_hideForSingleTab) && ([_cells count] <= 1)) {
[self hideTabBar:YES animate:YES];
return;
} else {
[self hideTabBar:NO animate:YES];
}
[self removeAllToolTips];
[_controller layoutCells]; //eventually we should only have to call this when we know something has changed
PSMTabBarCell *currentCell;
NSMenu *overflowMenu = [_controller overflowMenu];
[_overflowPopUpButton setHidden:(overflowMenu == nil)];
[_overflowPopUpButton setMenu:overflowMenu];
if(_animationTimer) {
[_animationTimer invalidate];
[_animationTimer release]; _animationTimer = nil;
}
if(animate) {
NSMutableArray *targetFrames = [NSMutableArray arrayWithCapacity:[_cells count]];
for(NSUInteger i = 0; i < [_cells count]; i++) {
currentCell = [_cells objectAtIndex:i];
//we're going from NSRect -> NSValue -> NSRect -> NSValue here - oh well
[targetFrames addObject:[NSValue valueWithRect:[_controller cellFrameAtIndex:i]]];
}
[_addTabButton setHidden:!_showAddTabButton];
NSAnimation *animation = [[NSAnimation alloc] initWithDuration:0.50 animationCurve:NSAnimationEaseInOut];
[animation setAnimationBlockingMode:NSAnimationNonblocking];
[animation startAnimation];
_animationTimer = [[NSTimer scheduledTimerWithTimeInterval:1.0 / 30.0
target:self
selector:@selector(_animateCells:)
userInfo:[NSArray arrayWithObjects:targetFrames, animation, nil]
repeats:YES] retain];
[animation release];
[[NSRunLoop currentRunLoop] addTimer:_animationTimer forMode:NSEventTrackingRunLoopMode];
[self _animateCells:_animationTimer];
} else {
for(NSUInteger i = 0; i < [_cells count]; i++) {
currentCell = [_cells objectAtIndex:i];
[currentCell setFrame:[_controller cellFrameAtIndex:i]];
if(![currentCell isInOverflowMenu]) {
[self _setupTrackingRectsForCell:currentCell];
}
}
[_addTabButton setFrame:[_controller addButtonRect]];
[_addTabButton setHidden:!_showAddTabButton];
[self setNeedsDisplay:YES];
}
}
- (void)_animateCells:(NSTimer *)timer {
NSAnimation *animation = [[timer userInfo] objectAtIndex:1];
NSArray *targetFrames = [[timer userInfo] objectAtIndex:0];
PSMTabBarCell *currentCell;
NSUInteger cellCount = [_cells count];
if((cellCount > 0) && [animation isAnimating]) {
//compare our target position with the current position and move towards the target
for(NSUInteger i = 0; i < [targetFrames count] && i < cellCount; i++) {
currentCell = [_cells objectAtIndex:i];
NSRect cellFrame = [currentCell frame], targetFrame = [[targetFrames objectAtIndex:i] rectValue];
CGFloat sizeChange;
CGFloat originChange;
if([self orientation] == PSMTabBarHorizontalOrientation) {
sizeChange = (targetFrame.size.width - cellFrame.size.width) * [animation currentProgress];
originChange = (targetFrame.origin.x - cellFrame.origin.x) * [animation currentProgress];
cellFrame.size.width += sizeChange;
cellFrame.origin.x += originChange;
} else {
sizeChange = (targetFrame.size.height - cellFrame.size.height) * [animation currentProgress];
originChange = (targetFrame.origin.y - cellFrame.origin.y) * [animation currentProgress];
cellFrame.size.height += sizeChange;
cellFrame.origin.y += originChange;
}
[currentCell setFrame:cellFrame];
//highlight the cell if the mouse is over it
NSPoint mousePoint = [self convertPoint:[[self window] mouseLocationOutsideOfEventStream] fromView:nil];
NSRect closeRect = [currentCell closeButtonRectForFrame:cellFrame];
[currentCell setHighlighted:NSMouseInRect(mousePoint, cellFrame, [self isFlipped])];
[currentCell setCloseButtonOver:NSMouseInRect(mousePoint, closeRect, [self isFlipped])];
}
if(_showAddTabButton) {
//animate the add tab button
NSRect target = [_controller addButtonRect], frame = [_addTabButton frame];
frame.origin.x += (target.origin.x - frame.origin.x) * [animation currentProgress];
[_addTabButton setFrame:frame];
}
} else {
//put all the cells where they should be in their final position
if(cellCount > 0) {
for(NSUInteger i = 0; i < [targetFrames count] && i < cellCount; i++) {
PSMTabBarCell *currentCell = [_cells objectAtIndex:i];
NSRect cellFrame = [currentCell frame], targetFrame = [[targetFrames objectAtIndex:i] rectValue];
if([self orientation] == PSMTabBarHorizontalOrientation) {
cellFrame.size.width = targetFrame.size.width;
cellFrame.origin.x = targetFrame.origin.x;
} else {
cellFrame.size.height = targetFrame.size.height;
cellFrame.origin.y = targetFrame.origin.y;
}
[currentCell setFrame:cellFrame];
//highlight the cell if the mouse is over it
NSPoint mousePoint = [self convertPoint:[[self window] mouseLocationOutsideOfEventStream] fromView:nil];
NSRect closeRect = [currentCell closeButtonRectForFrame:cellFrame];
[currentCell setHighlighted:NSMouseInRect(mousePoint, cellFrame, [self isFlipped])];
[currentCell setCloseButtonOver:NSMouseInRect(mousePoint, closeRect, [self isFlipped])];
}
}
//set the frame for the add tab button
if(_showAddTabButton) {
NSRect frame = [_addTabButton frame];
frame.origin.x = [_controller addButtonRect].origin.x;
[_addTabButton setFrame:frame];
}
[_animationTimer invalidate];
[_animationTimer release]; _animationTimer = nil;
for(NSUInteger i = 0; i < cellCount; i++) {
currentCell = [_cells objectAtIndex:i];
//we've hit the cells that are in overflow, stop setting up tracking rects
if([currentCell isInOverflowMenu]) {
break;
}
[self _setupTrackingRectsForCell:currentCell];
}
}
[self setNeedsDisplay:YES];
}
- (void)_setupTrackingRectsForCell:(PSMTabBarCell *)cell {
NSInteger tag, index = [_cells indexOfObject:cell];
NSRect cellTrackingRect = [_controller cellTrackingRectAtIndex:index];
NSPoint mousePoint = [self convertPoint:[[self window] mouseLocationOutsideOfEventStream] fromView:nil];
BOOL mouseInCell = NSMouseInRect(mousePoint, cellTrackingRect, [self isFlipped]);
//set the cell tracking rect
[self removeTrackingRect:[cell cellTrackingTag]];
tag = [self addTrackingRect:cellTrackingRect owner:cell userData:nil assumeInside:mouseInCell];
[cell setCellTrackingTag:tag];
[cell setHighlighted:mouseInCell];
if([cell hasCloseButton] && ![cell isCloseButtonSuppressed]) {
NSRect closeRect = [_controller closeButtonTrackingRectAtIndex:index];
BOOL mouseInCloseRect = NSMouseInRect(mousePoint, closeRect, [self isFlipped]);
//set the close button tracking rect
[self removeTrackingRect:[cell closeButtonTrackingTag]];
tag = [self addTrackingRect:closeRect owner:cell userData:nil assumeInside:mouseInCloseRect];
[cell setCloseButtonTrackingTag:tag];
[cell setCloseButtonOver:mouseInCloseRect];
}
//set the tooltip tracking rect
[self addToolTipRect:[cell frame] owner:self userData:nil];
}
- (void)_positionOverflowMenu {
NSRect cellRect, frame = [self frame];
cellRect.size.height = [style tabCellHeight];
cellRect.size.width = [style rightMarginForTabBarControl];
if([self orientation] == PSMTabBarHorizontalOrientation) {
cellRect.origin.y = 0;
cellRect.origin.x = frame.size.width - [style rightMarginForTabBarControl] + (_resizeAreaCompensation ? -(_resizeAreaCompensation - 1) : 1);
[_overflowPopUpButton setAutoresizingMask:NSViewNotSizable | NSViewMinXMargin];
} else {
cellRect.origin.x = 0;
cellRect.origin.y = frame.size.height - [style tabCellHeight];
cellRect.size.width = frame.size.width;
[_overflowPopUpButton setAutoresizingMask:NSViewNotSizable | NSViewMinXMargin | NSViewMinYMargin];
}
[_overflowPopUpButton setFrame:cellRect];
}
- (void)_checkWindowFrame {
//figure out if the new frame puts the control in the way of the resize widget
NSWindow *window = [self window];
if(window) {
NSRect resizeWidgetFrame = [[window contentView] frame];
resizeWidgetFrame.origin.x += resizeWidgetFrame.size.width - 22;
resizeWidgetFrame.size.width = 22;
resizeWidgetFrame.size.height = 22;
if([window showsResizeIndicator] && NSIntersectsRect([self frame], resizeWidgetFrame)) {
//the resize widgets are larger on metal windows
_resizeAreaCompensation = [window styleMask] & NSTexturedBackgroundWindowMask ? 20 : 8;
} else {
_resizeAreaCompensation = 0;
}
[self _positionOverflowMenu];
}
}
#pragma mark -
#pragma mark Mouse Tracking
- (BOOL)mouseDownCanMoveWindow {
return NO;
}
- (BOOL)acceptsFirstMouse:(NSEvent *)theEvent {
return YES;
}
- (void)mouseDown:(NSEvent *)theEvent {
_didDrag = NO;
// keep for dragging
[self setLastMouseDownEvent:theEvent];
// what cell?
NSPoint mousePt = [self convertPoint:[theEvent locationInWindow] fromView:nil];
NSRect frame = [self frame];
if([self orientation] == PSMTabBarVerticalOrientation && [self allowsResizing] && partnerView && (mousePt.x > frame.size.width - 3)) {
_resizing = YES;
}
NSRect cellFrame;
PSMTabBarCell *cell = [self cellForPoint:mousePt cellFrame:&cellFrame];
if(cell) {
BOOL overClose = NSMouseInRect(mousePt, [cell closeButtonRectForFrame:cellFrame], [self isFlipped]);
if(overClose &&
![self disableTabClose] &&
![cell isCloseButtonSuppressed] &&
([self allowsBackgroundTabClosing] || [[cell representedObject] isEqualTo:[tabView selectedTabViewItem]] || [theEvent modifierFlags] & NSCommandKeyMask)) {
[cell setCloseButtonOver:NO];
[cell setCloseButtonPressed:YES];
_closeClicked = YES;
} else {
[cell setCloseButtonPressed:NO];
if(_selectsTabsOnMouseDown) {
[self performSelector:@selector(tabClick:) withObject:cell];
}
}
[self setNeedsDisplay:YES];
}
}
- (void)mouseDragged:(NSEvent *)theEvent {
if([self lastMouseDownEvent] == nil) {
return;
}
NSPoint currentPoint = [self convertPoint:[theEvent locationInWindow] fromView:nil];
if(_resizing) {
NSRect frame = [self frame];
CGFloat resizeAmount = [theEvent deltaX];
if((currentPoint.x > frame.size.width && resizeAmount > 0) || (currentPoint.x < frame.size.width && resizeAmount < 0)) {
[[NSCursor resizeLeftRightCursor] push];
NSRect partnerFrame = [partnerView frame];
//do some bounds checking
if((frame.size.width + resizeAmount > [self cellMinWidth]) && (frame.size.width + resizeAmount < [self cellMaxWidth])) {
frame.size.width += resizeAmount;
partnerFrame.size.width -= resizeAmount;
partnerFrame.origin.x += resizeAmount;
[self setFrame:frame];
[partnerView setFrame:partnerFrame];
[[self superview] setNeedsDisplay:YES];
}
}
return;
}
NSRect cellFrame;
NSPoint trackingStartPoint = [self convertPoint:[[self lastMouseDownEvent] locationInWindow] fromView:nil];
PSMTabBarCell *cell = [self cellForPoint:trackingStartPoint cellFrame:&cellFrame];
if(cell) {
//check to see if the close button was the target in the clicked cell
//highlight/unhighlight the close button as necessary
NSRect iconRect = [cell closeButtonRectForFrame:cellFrame];
if(_closeClicked && NSMouseInRect(trackingStartPoint, iconRect, [self isFlipped]) &&
([self allowsBackgroundTabClosing] || [[cell representedObject] isEqualTo:[tabView selectedTabViewItem]])) {
[cell setCloseButtonPressed:NSMouseInRect(currentPoint, iconRect, [self isFlipped])];
[self setNeedsDisplay:YES];
return;
}
CGFloat dx = fabs(currentPoint.x - trackingStartPoint.x);
CGFloat dy = fabs(currentPoint.y - trackingStartPoint.y);
CGFloat distance = sqrt(dx * dx + dy * dy);
if(distance >= 10 && !_didDrag && ![[PSMTabDragAssistant sharedDragAssistant] isDragging] &&
[self delegate] && [[self delegate] respondsToSelector:@selector(tabView:shouldDragTabViewItem:fromTabBar:)] &&
[[self delegate] tabView:tabView shouldDragTabViewItem:[cell representedObject] fromTabBar:self]) {
_didDrag = YES;
[[PSMTabDragAssistant sharedDragAssistant] startDraggingCell:cell fromTabBar:self withMouseDownEvent:[self lastMouseDownEvent]];
}
}
}
- (void)mouseUp:(NSEvent *)theEvent {
if(_resizing) {
_resizing = NO;
[[NSCursor arrowCursor] set];
} else {
// what cell?
NSPoint mousePt = [self convertPoint:[theEvent locationInWindow] fromView:nil];
NSRect cellFrame, mouseDownCellFrame;
PSMTabBarCell *cell = [self cellForPoint:mousePt cellFrame:&cellFrame];
PSMTabBarCell *mouseDownCell = [self cellForPoint:[self convertPoint:[[self lastMouseDownEvent] locationInWindow] fromView:nil] cellFrame:&mouseDownCellFrame];
if(cell) {
NSPoint trackingStartPoint = [self convertPoint:[[self lastMouseDownEvent] locationInWindow] fromView:nil];
NSRect iconRect = [mouseDownCell closeButtonRectForFrame:mouseDownCellFrame];
if((NSMouseInRect(mousePt, iconRect, [self isFlipped])) && ![self disableTabClose] && ![cell isCloseButtonSuppressed] && [mouseDownCell closeButtonPressed]) {
if(([[NSApp currentEvent] modifierFlags] & NSAlternateKeyMask) != 0) {
//If the user is holding Option, close all other tabs
NSEnumerator *enumerator = [[[[self cells] copy] autorelease] objectEnumerator];
PSMTabBarCell *otherCell;
while((otherCell = [enumerator nextObject])) {
if(otherCell != cell) {
[self performSelector:@selector(closeTabClick:) withObject:otherCell];
}
}
//Fix the close button for the clicked tab not to be pressed
[cell setCloseButtonPressed:NO];
} else {
//Otherwise, close this tab
[self performSelector:@selector(closeTabClick:) withObject:cell];
}
} else if(NSMouseInRect(mousePt, mouseDownCellFrame, [self isFlipped]) &&
(!NSMouseInRect(trackingStartPoint, [cell closeButtonRectForFrame:cellFrame], [self isFlipped]) || ![self allowsBackgroundTabClosing] || [self disableTabClose])) {
[mouseDownCell setCloseButtonPressed:NO];
// If -[self selectsTabsOnMouseDown] is TRUE, we already performed tabClick: on mouseDown.
if(![self selectsTabsOnMouseDown]) {
[self performSelector:@selector(tabClick:) withObject:cell];
}
} else {
[mouseDownCell setCloseButtonPressed:NO];
[self performSelector:@selector(tabNothing:) withObject:cell];
}
}
_closeClicked = NO;
}
}
- (NSMenu *)menuForEvent:(NSEvent *)event {
NSMenu *menu = nil;
NSTabViewItem *item = [[self cellForPoint:[self convertPoint:[event locationInWindow] fromView:nil] cellFrame:nil] representedObject];
if(item && [[self delegate] respondsToSelector:@selector(tabView:menuForTabViewItem:)]) {
menu = [[self delegate] tabView:tabView menuForTabViewItem:item];
}
return menu;
}
#pragma mark -
#pragma mark Drag and Drop
- (BOOL)shouldDelayWindowOrderingForEvent:(NSEvent *)theEvent {
return YES;
}
// NSDraggingSource
- (NSDragOperation)draggingSourceOperationMaskForLocal:(BOOL)isLocal {
return(isLocal ? NSDragOperationMove : NSDragOperationNone);
}
- (BOOL)ignoreModifierKeysWhileDragging {
return YES;
}
- (void)draggedImage:(NSImage *)anImage beganAt:(NSPoint)screenPoint {
[[PSMTabDragAssistant sharedDragAssistant] draggingBeganAt:screenPoint];
}
- (void)draggedImage:(NSImage *)image movedTo:(NSPoint)screenPoint {
[[PSMTabDragAssistant sharedDragAssistant] draggingMovedTo:screenPoint];
}
// NSDraggingDestination
- (NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender {
if([[[sender draggingPasteboard] types] indexOfObject:@"PSMTabBarControlItemPBType"] != NSNotFound) {
if([self delegate] && [[self delegate] respondsToSelector:@selector(tabView:shouldDropTabViewItem:inTabBar:)] &&
![[self delegate] tabView:[[sender draggingSource] tabView] shouldDropTabViewItem:[[[PSMTabDragAssistant sharedDragAssistant] draggedCell] representedObject] inTabBar:self]) {
return NSDragOperationNone;
}
[[PSMTabDragAssistant sharedDragAssistant] draggingEnteredTabBar:self atPoint:[self convertPoint:[sender draggingLocation] fromView:nil]];
return NSDragOperationMove;
}
return NSDragOperationNone;
}
- (NSDragOperation)draggingUpdated:(id <NSDraggingInfo>)sender {
PSMTabBarCell *cell = [self cellForPoint:[self convertPoint:[sender draggingLocation] fromView:nil] cellFrame:nil];
if([[[sender draggingPasteboard] types] indexOfObject:@"PSMTabBarControlItemPBType"] != NSNotFound) {
if([self delegate] && [[self delegate] respondsToSelector:@selector(tabView:shouldDropTabViewItem:inTabBar:)] &&
![[self delegate] tabView:[[sender draggingSource] tabView] shouldDropTabViewItem:[[[PSMTabDragAssistant sharedDragAssistant] draggedCell] representedObject] inTabBar:self]) {
return NSDragOperationNone;
}
[[PSMTabDragAssistant sharedDragAssistant] draggingUpdatedInTabBar:self atPoint:[self convertPoint:[sender draggingLocation] fromView:nil]];
return NSDragOperationMove;
} else if(cell) {
//something that was accepted by the delegate was dragged on
//Test for the space bar (the skip-the-delay key).
/*enum { virtualKeycodeForSpace = 49 }; //Source: IM:Tx (Fig. C-2)
union {
KeyMap keymap;
char bits[16];
} keymap;
GetKeys(keymap.keymap);
if ((GetCurrentEventKeyModifiers() == 0) && bit_test(keymap.bits, virtualKeycodeForSpace)) {
//The user pressed the space bar. This skips the delay; the user wants to pop the spring on this tab *now*.
//For some reason, it crashes if I call -fire here. I don't know why. It doesn't crash if I simply set the fire date to now.
[_springTimer setFireDate:[NSDate date]];
} else {*/
//Wind the spring for a spring-loaded drop.
//The delay time comes from Finder's defaults, which specifies it in milliseconds.
//If the delegate can't handle our spring-loaded drop, we'll abort it when the timer fires. See fireSpring:. This is simpler than constantly (checking for spring-loaded awareness and tearing down/rebuilding the timer) at every delegate change.
//If the user has dragged to a different tab, reset the timer.
if(_tabViewItemWithSpring != [cell representedObject]) {
[_springTimer invalidate];
[_springTimer release]; _springTimer = nil;
_tabViewItemWithSpring = [cell representedObject];
}
if(!_springTimer) {
//Finder's default delay time, as of Tiger, is 668 ms. If the user has never changed it, there's no setting in its defaults, so we default to that amount.
NSNumber *delayNumber = [(NSNumber *)CFPreferencesCopyAppValue((CFStringRef)@"SpringingDelayMilliseconds", (CFStringRef)@"com.apple.finder") autorelease];
NSTimeInterval delaySeconds = delayNumber ?[delayNumber doubleValue] / 1000.0 : 0.668;
_springTimer = [[NSTimer scheduledTimerWithTimeInterval:delaySeconds
target:self
selector:@selector(fireSpring:)
userInfo:sender
repeats:NO] retain];
}
return NSDragOperationCopy;
}
return NSDragOperationNone;
}
- (void)draggingExited:(id <NSDraggingInfo>)sender {
[_springTimer invalidate];
[_springTimer release]; _springTimer = nil;
[[PSMTabDragAssistant sharedDragAssistant] draggingExitedTabBar:self];
}
- (BOOL)prepareForDragOperation:(id <NSDraggingInfo>)sender {
//validate the drag operation only if there's a valid tab bar to drop into
return [[[sender draggingPasteboard] types] indexOfObject:@"PSMTabBarControlItemPBType"] == NSNotFound ||
[[PSMTabDragAssistant sharedDragAssistant] destinationTabBar] != nil;
}
- (BOOL)performDragOperation:(id <NSDraggingInfo>)sender {
if([[[sender draggingPasteboard] types] indexOfObject:@"PSMTabBarControlItemPBType"] != NSNotFound) {
[[PSMTabDragAssistant sharedDragAssistant] performDragOperation];
} else if([self delegate] && [[self delegate] respondsToSelector:@selector(tabView:acceptedDraggingInfo:onTabViewItem:)]) {
//forward the drop to the delegate
[[self delegate] tabView:tabView acceptedDraggingInfo:sender onTabViewItem:[[self cellForPoint:[self convertPoint:[sender draggingLocation] fromView:nil] cellFrame:nil] representedObject]];
}
return YES;
}
- (void)draggedImage:(NSImage *)anImage endedAt:(NSPoint)aPoint operation:(NSDragOperation)operation {
[[PSMTabDragAssistant sharedDragAssistant] draggedImageEndedAt:aPoint operation:operation];
}
- (void)concludeDragOperation:(id <NSDraggingInfo>)sender {
}
#pragma mark -
#pragma mark Spring-loading
- (void)fireSpring:(NSTimer *)timer {
NSAssert1(timer == _springTimer, @"Spring fired by unrecognized timer %@", timer);
id <NSDraggingInfo> sender = [timer userInfo];
PSMTabBarCell *cell = [self cellForPoint:[self convertPoint:[sender draggingLocation] fromView:nil] cellFrame:nil];
[tabView selectTabViewItem:[cell representedObject]];
_tabViewItemWithSpring = nil;
[_springTimer invalidate];
[_springTimer release]; _springTimer = nil;
}
#pragma mark -
#pragma mark Actions
- (void)overflowMenuAction:(id)sender {
NSTabViewItem *tabViewItem = (NSTabViewItem *)[sender representedObject];
[tabView selectTabViewItem:tabViewItem];
}
- (void)closeTabClick:(id)sender {
NSTabViewItem *item = [sender representedObject];
[sender retain];
if(([_cells count] == 1) && (![self canCloseOnlyTab])) {
return;
}
if([[self delegate] respondsToSelector:@selector(tabView:shouldCloseTabViewItem:)]) {
if(![[self delegate] tabView:tabView shouldCloseTabViewItem:item]) {
// fix mouse downed close button
[sender setCloseButtonPressed:NO];
return;
}
}
[item retain];
[tabView removeTabViewItem:item];
[item release];
[sender release];
}
- (void)tabClick:(id)sender {
[tabView selectTabViewItem:[sender representedObject]];
}
- (void)tabNothing:(id)sender {
//[self update]; // takes care of highlighting based on state
}
- (void)frameDidChange:(NSNotification *)notification {
[self _checkWindowFrame];
// trying to address the drawing artifacts for the progress indicators - hackery follows
// this one fixes the "blanking" effect when the control hides and shows itself
NSEnumerator *e = [_cells objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
[[cell indicator] stopAnimation:self];
[[cell indicator] performSelector:@selector(startAnimation:)
withObject:nil
afterDelay:0];
}
[self update:NO];
}
- (void)viewDidMoveToWindow {
[self _checkWindowFrame];
}
- (void)viewWillStartLiveResize {
NSEnumerator *e = [_cells objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
[[cell indicator] stopAnimation:self];
}
[self setNeedsDisplay:YES];
}
-(void)viewDidEndLiveResize {
NSEnumerator *e = [_cells objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
[[cell indicator] startAnimation:self];
}
[self _checkWindowFrame];
[self update:NO];
}
- (void)resetCursorRects {
[super resetCursorRects];
if([self orientation] == PSMTabBarVerticalOrientation) {
NSRect frame = [self frame];
[self addCursorRect:NSMakeRect(frame.size.width - 2, 0, 2, frame.size.height) cursor:[NSCursor resizeLeftRightCursor]];
}
}
- (void)windowDidMove:(NSNotification *)aNotification {
[self setNeedsDisplay:YES];
}
- (void)windowDidUpdate:(NSNotification *)notification {
// hide? must readjust things if I'm not supposed to be showing
// this block of code only runs when the app launches
if([self hideForSingleTab] && ([_cells count] <= 1) && !_awakenedFromNib) {
// must adjust frames now before display
NSRect myFrame = [self frame];
if([self orientation] == PSMTabBarHorizontalOrientation) {
if(partnerView) {
NSRect partnerFrame = [partnerView frame];
// above or below me?
if(myFrame.origin.y - 22 > [partnerView frame].origin.y) {
// partner is below me
[self setFrame:NSMakeRect(myFrame.origin.x, myFrame.origin.y + 21, myFrame.size.width, myFrame.size.height - 21)];
[partnerView setFrame:NSMakeRect(partnerFrame.origin.x, partnerFrame.origin.y, partnerFrame.size.width, partnerFrame.size.height + 21)];
} else {
// partner is above me
[self setFrame:NSMakeRect(myFrame.origin.x, myFrame.origin.y, myFrame.size.width, myFrame.size.height - 21)];
[partnerView setFrame:NSMakeRect(partnerFrame.origin.x, partnerFrame.origin.y - 21, partnerFrame.size.width, partnerFrame.size.height + 21)];
}
[partnerView setNeedsDisplay:YES];
[self setNeedsDisplay:YES];
} else {
// for window movement
NSRect windowFrame = [[self window] frame];
[[self window] setFrame:NSMakeRect(windowFrame.origin.x, windowFrame.origin.y + 21, windowFrame.size.width, windowFrame.size.height - 21) display:YES];
[self setFrame:NSMakeRect(myFrame.origin.x, myFrame.origin.y, myFrame.size.width, myFrame.size.height - 21)];
}
} else {
if(partnerView) {
NSRect partnerFrame = [partnerView frame];
//to the left or right?
if(myFrame.origin.x < [partnerView frame].origin.x) {
// partner is to the left
[self setFrame:NSMakeRect(myFrame.origin.x, myFrame.origin.y, 1, myFrame.size.height)];
[partnerView setFrame:NSMakeRect(partnerFrame.origin.x - myFrame.size.width + 1, partnerFrame.origin.y, partnerFrame.size.width + myFrame.size.width - 1, partnerFrame.size.height)];
} else {
// partner to the right
[self setFrame:NSMakeRect(myFrame.origin.x + myFrame.size.width, myFrame.origin.y, 1, myFrame.size.height)];
[partnerView setFrame:NSMakeRect(partnerFrame.origin.x, partnerFrame.origin.y, partnerFrame.size.width + myFrame.size.width, partnerFrame.size.height)];
}
_tabBarWidth = myFrame.size.width;
[partnerView setNeedsDisplay:YES];
[self setNeedsDisplay:YES];
} else {
// for window movement
NSRect windowFrame = [[self window] frame];
[[self window] setFrame:NSMakeRect(windowFrame.origin.x + myFrame.size.width - 1, windowFrame.origin.y, windowFrame.size.width - myFrame.size.width + 1, windowFrame.size.height) display:YES];
[self setFrame:NSMakeRect(myFrame.origin.x, myFrame.origin.y, 1, myFrame.size.height)];
}
}
_isHidden = YES;
if([[self delegate] respondsToSelector:@selector(tabView:tabBarDidHide:)]) {
[[self delegate] tabView:[self tabView] tabBarDidHide:self];
}
}
_awakenedFromNib = YES;
[self setNeedsDisplay:YES];
//we only need to do this once
[[NSNotificationCenter defaultCenter] removeObserver:self name:NSWindowDidUpdateNotification object:nil];
}
#pragma mark -
#pragma mark Menu Validation
- (BOOL)validateMenuItem:(NSMenuItem *)sender {
[sender setState:([[sender representedObject] isEqualTo:[tabView selectedTabViewItem]]) ? NSOnState : NSOffState];
return [[self delegate] respondsToSelector:@selector(tabView:validateOverflowMenuItem:forTabViewItem:)] ?
[[self delegate] tabView:[self tabView] validateOverflowMenuItem:sender forTabViewItem:[sender representedObject]] : YES;
}
#pragma mark -
#pragma mark NSTabView Delegate
- (void)tabView:(NSTabView *)aTabView didSelectTabViewItem:(NSTabViewItem *)tabViewItem {
// here's a weird one - this message is sent before the "tabViewDidChangeNumberOfTabViewItems"
// message, thus I can end up updating when there are no cells, if no tabs were (yet) present
NSUInteger tabIndex = [aTabView indexOfTabViewItem:tabViewItem];
if([_cells count] > 0 && tabIndex < [_cells count]) {
PSMTabBarCell *thisCell = [_cells objectAtIndex:tabIndex];
if(_alwaysShowActiveTab && [thisCell isInOverflowMenu]) {
//temporarily disable the delegate in order to move the tab to a different index
id tempDelegate = [aTabView delegate];
[aTabView setDelegate:nil];
// move it all around first
[tabViewItem retain];
[thisCell retain];
[aTabView removeTabViewItem:tabViewItem];
[aTabView insertTabViewItem:tabViewItem atIndex:0];
[_cells removeObjectAtIndex:tabIndex];
[_cells insertObject:thisCell atIndex:0];
[thisCell setIsInOverflowMenu:NO]; //very important else we get a fun recursive loop going
[[_cells objectAtIndex:[_cells count] - 1] setIsInOverflowMenu:YES]; //these 2 lines are pretty uncool and this logic needs to be updated
[thisCell release];
[tabViewItem release];
[aTabView setDelegate:tempDelegate];
//reset the selection since removing it changed the selection
[aTabView selectTabViewItem:tabViewItem];
[self update];
} else {
[_controller setSelectedCell:thisCell];
[self setNeedsDisplay:YES];
}
}
if([[self delegate] respondsToSelector:@selector(tabView:didSelectTabViewItem:)]) {
[[self delegate] performSelector:@selector(tabView:didSelectTabViewItem:) withObject:aTabView withObject:tabViewItem];
}
}
- (BOOL)tabView:(NSTabView *)aTabView shouldSelectTabViewItem:(NSTabViewItem *)tabViewItem {
if([[self delegate] respondsToSelector:@selector(tabView:shouldSelectTabViewItem:)]) {
return [[self delegate] tabView:aTabView shouldSelectTabViewItem:tabViewItem];
} else {
return YES;
}
}
- (void)tabView:(NSTabView *)aTabView willSelectTabViewItem:(NSTabViewItem *)tabViewItem {
if([[self delegate] respondsToSelector:@selector(tabView:willSelectTabViewItem:)]) {
[[self delegate] performSelector:@selector(tabView:willSelectTabViewItem:) withObject:aTabView withObject:tabViewItem];
}
}
- (void)tabViewDidChangeNumberOfTabViewItems:(NSTabView *)aTabView {
NSArray *tabItems = [tabView tabViewItems];
// go through cells, remove any whose representedObjects are not in [tabView tabViewItems]
NSEnumerator *e = [[[_cells copy] autorelease] objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
//remove the observer binding
if([cell representedObject] && ![tabItems containsObject:[cell representedObject]]) {
if([[self delegate] respondsToSelector:@selector(tabView:didCloseTabViewItem:)]) {
[[self delegate] tabView:aTabView didCloseTabViewItem:[cell representedObject]];
}
[self removeTabForCell:cell];
}
}
// go through tab view items, add cell for any not present
NSMutableArray *cellItems = [self representedTabViewItems];
NSEnumerator *ex = [tabItems objectEnumerator];
NSTabViewItem *item;
while((item = [ex nextObject])) {
if(![cellItems containsObject:item]) {
[self addTabViewItem:item];
}
}
// pass along for other delegate responses
if([[self delegate] respondsToSelector:@selector(tabViewDidChangeNumberOfTabViewItems:)]) {
[[self delegate] performSelector:@selector(tabViewDidChangeNumberOfTabViewItems:) withObject:aTabView];
}
// reset cursor tracking for the add tab button if one exists
if([self addTabButton]) {
[[self addTabButton] resetCursorRects];
}
}
#pragma mark -
#pragma mark Tooltips
- (NSString *)view:(NSView *)view stringForToolTip:(NSToolTipTag)tag point:(NSPoint)point userData:(void *)userData {
if([[self delegate] respondsToSelector:@selector(tabView:toolTipForTabViewItem:)]) {
return [[self delegate] tabView:[self tabView] toolTipForTabViewItem:[[self cellForPoint:point cellFrame:nil] representedObject]];
}
return nil;
}
#pragma mark -
#pragma mark Archiving
- (void)encodeWithCoder:(NSCoder *)aCoder
{
[super encodeWithCoder:aCoder];
if ([aCoder allowsKeyedCoding]) {
[aCoder encodeObject:_cells forKey:@"PSMcells"];
[aCoder encodeObject:tabView forKey:@"PSMtabView"];
[aCoder encodeObject:_overflowPopUpButton forKey:@"PSMoverflowPopUpButton"];
[aCoder encodeObject:_addTabButton forKey:@"PSMaddTabButton"];
[aCoder encodeObject:style forKey:@"PSMstyle"];
[aCoder encodeInteger:_orientation forKey:@"PSMorientation"];
[aCoder encodeBool:_canCloseOnlyTab forKey:@"PSMcanCloseOnlyTab"];
[aCoder encodeBool:_disableTabClose forKey:@"PSMdisableTabClose"];
[aCoder encodeBool:_hideForSingleTab forKey:@"PSMhideForSingleTab"];
[aCoder encodeBool:_allowsBackgroundTabClosing forKey:@"PSMallowsBackgroundTabClosing"];
[aCoder encodeBool:_allowsResizing forKey:@"PSMallowsResizing"];
[aCoder encodeBool:_selectsTabsOnMouseDown forKey:@"PSMselectsTabsOnMouseDown"];
[aCoder encodeBool:_showAddTabButton forKey:@"PSMshowAddTabButton"];
[aCoder encodeBool:_sizeCellsToFit forKey:@"PSMsizeCellsToFit"];
[aCoder encodeInteger:_cellMinWidth forKey:@"PSMcellMinWidth"];
[aCoder encodeInteger:_cellMaxWidth forKey:@"PSMcellMaxWidth"];
[aCoder encodeInteger:_cellOptimumWidth forKey:@"PSMcellOptimumWidth"];
[aCoder encodeInteger:_currentStep forKey:@"PSMcurrentStep"];
[aCoder encodeBool:_isHidden forKey:@"PSMisHidden"];
[aCoder encodeObject:partnerView forKey:@"PSMpartnerView"];
[aCoder encodeBool:_awakenedFromNib forKey:@"PSMawakenedFromNib"];
[aCoder encodeObject:_lastMouseDownEvent forKey:@"PSMlastMouseDownEvent"];
[aCoder encodeObject:delegate forKey:@"PSMdelegate"];
[aCoder encodeBool:_useOverflowMenu forKey:@"PSMuseOverflowMenu"];
[aCoder encodeBool:_automaticallyAnimates forKey:@"PSMautomaticallyAnimates"];
[aCoder encodeBool:_alwaysShowActiveTab forKey:@"PSMalwaysShowActiveTab"];
}
}
- (id)initWithCoder:(NSCoder *)aDecoder
{
self = [super initWithCoder:aDecoder];
if (self) {
// Initialization
[self initAddedProperties];
[self registerForDraggedTypes:[NSArray arrayWithObjects:@"PSMTabBarControlItemPBType", nil]];
// resize
[self setPostsFrameChangedNotifications:YES];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(frameDidChange:) name:NSViewFrameDidChangeNotification object:self];
if ([aDecoder allowsKeyedCoding]) {
_cells = [[aDecoder decodeObjectForKey:@"PSMcells"] retain];
tabView = [[aDecoder decodeObjectForKey:@"PSMtabView"] retain];
_overflowPopUpButton = [[aDecoder decodeObjectForKey:@"PSMoverflowPopUpButton"] retain];
_addTabButton = [[aDecoder decodeObjectForKey:@"PSMaddTabButton"] retain];
style = [[aDecoder decodeObjectForKey:@"PSMstyle"] retain];
_orientation = (PSMTabBarOrientation)[aDecoder decodeIntegerForKey:@"PSMorientation"];
_canCloseOnlyTab = [aDecoder decodeBoolForKey:@"PSMcanCloseOnlyTab"];
_disableTabClose = [aDecoder decodeBoolForKey:@"PSMdisableTabClose"];
_hideForSingleTab = [aDecoder decodeBoolForKey:@"PSMhideForSingleTab"];
_allowsBackgroundTabClosing = [aDecoder decodeBoolForKey:@"PSMallowsBackgroundTabClosing"];
_allowsResizing = [aDecoder decodeBoolForKey:@"PSMallowsResizing"];
_selectsTabsOnMouseDown = [aDecoder decodeBoolForKey:@"PSMselectsTabsOnMouseDown"];
_showAddTabButton = [aDecoder decodeBoolForKey:@"PSMshowAddTabButton"];
_sizeCellsToFit = [aDecoder decodeBoolForKey:@"PSMsizeCellsToFit"];
_cellMinWidth = [aDecoder decodeIntegerForKey:@"PSMcellMinWidth"];
_cellMaxWidth = [aDecoder decodeIntegerForKey:@"PSMcellMaxWidth"];
_cellOptimumWidth = [aDecoder decodeIntegerForKey:@"PSMcellOptimumWidth"];
_currentStep = [aDecoder decodeIntegerForKey:@"PSMcurrentStep"];
_isHidden = [aDecoder decodeBoolForKey:@"PSMisHidden"];
partnerView = [[aDecoder decodeObjectForKey:@"PSMpartnerView"] retain];
_awakenedFromNib = [aDecoder decodeBoolForKey:@"PSMawakenedFromNib"];
_lastMouseDownEvent = [[aDecoder decodeObjectForKey:@"PSMlastMouseDownEvent"] retain];
_useOverflowMenu = [aDecoder decodeBoolForKey:@"PSMuseOverflowMenu"];
_automaticallyAnimates = [aDecoder decodeBoolForKey:@"PSMautomaticallyAnimates"];
_alwaysShowActiveTab = [aDecoder decodeBoolForKey:@"PSMalwaysShowActiveTab"];
delegate = [[aDecoder decodeObjectForKey:@"PSMdelegate"] retain];
}
}
[self setTarget:self];
return self;
}
#pragma mark -
#pragma mark IB Palette
- (NSSize)minimumFrameSizeFromKnobPosition:(NSInteger)position {
return NSMakeSize(100.0, 22.0);
}
- (NSSize)maximumFrameSizeFromKnobPosition:(NSInteger)knobPosition {
return NSMakeSize(10000.0, 22.0);
}
- (void)placeView:(NSRect)newFrame {
// this is called any time the view is resized in IB
[self setFrame:newFrame];
[self update:NO];
}
#pragma mark -
#pragma mark Convenience
- (void)bindPropertiesForCell:(PSMTabBarCell *)cell andTabViewItem:(NSTabViewItem *)item {
[self _bindPropertiesForCell:cell andTabViewItem:item];
// watch for changes in the identifier
[item addObserver:self forKeyPath:@"identifier" options:0 context:nil];
}
- (void)_bindPropertiesForCell:(PSMTabBarCell *)cell andTabViewItem:(NSTabViewItem *)item {
// bind the indicator to the represented object's status (if it exists)
[[cell indicator] setHidden:YES];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(isProcessing)]) {
NSMutableDictionary *bindingOptions = [NSMutableDictionary dictionary];
[bindingOptions setObject:NSNegateBooleanTransformerName forKey:@"NSValueTransformerName"];
[[cell indicator] bind:@"animate" toObject:[item identifier] withKeyPath:@"isProcessing" options:nil];
[[cell indicator] bind:@"hidden" toObject:[item identifier] withKeyPath:@"isProcessing" options:bindingOptions];
[[item identifier] addObserver:cell forKeyPath:@"isProcessing" options:0 context:nil];
}
}
// bind for the existence of an icon
[cell setHasIcon:NO];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(icon)]) {
NSMutableDictionary *bindingOptions = [NSMutableDictionary dictionary];
[bindingOptions setObject:NSIsNotNilTransformerName forKey:@"NSValueTransformerName"];
[cell bind:@"hasIcon" toObject:[item identifier] withKeyPath:@"icon" options:bindingOptions];
[[item identifier] addObserver:cell forKeyPath:@"icon" options:0 context:nil];
}
}
// bind for the existence of a counter
[cell setCount:0];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(objectCount)]) {
[cell bind:@"count" toObject:[item identifier] withKeyPath:@"objectCount" options:nil];
[[item identifier] addObserver:cell forKeyPath:@"objectCount" options:0 context:nil];
}
}
// bind for the color of a counter
[cell setCountColor:nil];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(countColor)]) {
[cell bind:@"countColor" toObject:[item identifier] withKeyPath:@"countColor" options:nil];
[[item identifier] addObserver:cell forKeyPath:@"countColor" options:0 context:nil];
}
}
// bind for a large image
[cell setHasLargeImage:NO];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(largeImage)]) {
NSMutableDictionary *bindingOptions = [NSMutableDictionary dictionary];
[bindingOptions setObject:NSIsNotNilTransformerName forKey:@"NSValueTransformerName"];
[cell bind:@"hasLargeImage" toObject:[item identifier] withKeyPath:@"largeImage" options:bindingOptions];
[[item identifier] addObserver:cell forKeyPath:@"largeImage" options:0 context:nil];
}
}
[cell setIsEdited:NO];
if([item identifier] != nil) {
if([[[cell representedObject] identifier] respondsToSelector:@selector(isEdited)]) {
[cell bind:@"isEdited" toObject:[item identifier] withKeyPath:@"isEdited" options:nil];
[[item identifier] addObserver:cell forKeyPath:@"isEdited" options:0 context:nil];
}
}
// bind my string value to the label on the represented tab
[cell bind:@"title" toObject:item withKeyPath:@"label" options:nil];
}
- (NSMutableArray *)representedTabViewItems {
NSMutableArray *temp = [NSMutableArray arrayWithCapacity:[_cells count]];
NSEnumerator *e = [_cells objectEnumerator];
PSMTabBarCell *cell;
while((cell = [e nextObject])) {
if([cell representedObject]) {
[temp addObject:[cell representedObject]];
}
}
return temp;
}
- (id)cellForPoint:(NSPoint)point cellFrame:(NSRectPointer)outFrame {
if([self orientation] == PSMTabBarHorizontalOrientation && !NSPointInRect(point, [self genericCellRect])) {
return nil;
}
NSInteger i, cnt = [_cells count];
for(i = 0; i < cnt; i++) {
PSMTabBarCell *cell = [_cells objectAtIndex:i];
if(NSPointInRect(point, [cell frame])) {
if(outFrame) {
*outFrame = [cell frame];
}
return cell;
}
}
return nil;
}
- (PSMTabBarCell *)lastVisibleTab {
NSInteger i, cellCount = [_cells count];
for(i = 0; i < cellCount; i++) {
if([[_cells objectAtIndex:i] isInOverflowMenu]) {
return [_cells objectAtIndex:(i - 1)];
}
}
return [_cells objectAtIndex:(cellCount - 1)];
}
- (NSInteger)numberOfVisibleTabs {
NSUInteger i, cellCount = 0;
PSMTabBarCell *nextCell;
for(i = 0; i < [_cells count]; i++) {
nextCell = [_cells objectAtIndex:i];
if([nextCell isInOverflowMenu]) {
break;
}
if(![nextCell isPlaceholder]) {
cellCount++;
}
}
return cellCount;
}
#pragma mark -
#pragma mark Accessibility
-(BOOL)accessibilityIsIgnored {
return NO;
}
- (id)accessibilityAttributeValue:(NSString *)attribute {
id attributeValue = nil;
if([attribute isEqualToString: NSAccessibilityRoleAttribute]) {
attributeValue = NSAccessibilityGroupRole;
} else if([attribute isEqualToString: NSAccessibilityChildrenAttribute]) {
attributeValue = NSAccessibilityUnignoredChildren(_cells);
} else {
attributeValue = [super accessibilityAttributeValue:attribute];
}
return attributeValue;
}
- (id)accessibilityHitTest:(NSPoint)point {
id hitTestResult = self;
NSEnumerator *enumerator = [_cells objectEnumerator];
PSMTabBarCell *cell = nil;
PSMTabBarCell *highlightedCell = nil;
while(!highlightedCell && (cell = [enumerator nextObject])) {
if([cell isHighlighted]) {
highlightedCell = cell;
}
}
if(highlightedCell) {
hitTestResult = [highlightedCell accessibilityHitTest:point];
}
return hitTestResult;
}
@end