1 2/* 3 * Copyright 2011 Google Inc. 4 * 5 * Use of this source code is governed by a BSD-style license that can be 6 * found in the LICENSE file. 7 */ 8 9#import "SkOptionsTableView.h" 10#import "SkTextFieldCell.h" 11@implementation SkOptionItem 12@synthesize fCell, fItem; 13- (void)dealloc { 14 [fCell release]; 15 [super dealloc]; 16} 17@end 18 19@implementation SkOptionsTableView 20@synthesize fItems; 21 22- (id)initWithCoder:(NSCoder*)coder { 23 if ((self = [super initWithCoder:coder])) { 24 self.dataSource = self; 25 self.delegate = self; 26 fMenus = NULL; 27 fShowKeys = YES; 28 [self setSelectionHighlightStyle:NSTableViewSelectionHighlightStyleNone]; 29 self.fItems = [NSMutableArray array]; 30 } 31 return self; 32} 33 34- (void)dealloc { 35 self.fItems = nil; 36 [super dealloc]; 37} 38 39- (void) view:(SkNSView*)view didAddMenu:(const SkOSMenu*)menu {} 40- (void) view:(SkNSView*)view didUpdateMenu:(const SkOSMenu*)menu { 41 [self updateMenu:menu]; 42} 43 44- (IBAction)toggleKeyEquivalents:(id)sender { 45 fShowKeys = !fShowKeys; 46 NSMenuItem* item = (NSMenuItem*)sender; 47 [item setState:fShowKeys]; 48 [self reloadData]; 49} 50 51- (void)registerMenus:(const SkTDArray<SkOSMenu*>*)menus { 52 fMenus = menus; 53 for (int i = 0; i < fMenus->count(); ++i) { 54 [self loadMenu:(*fMenus)[i]]; 55 } 56} 57 58- (void)updateMenu:(const SkOSMenu*)menu { 59 // the first menu is always assumed to be the static, the second is 60 // repopulated every time over and over again 61 62 // seems pretty weird that we have to get rid of the const'ness here, 63 // but trying to propagate the const'ness through all the way to the fMenus 64 // vector was a non-starter. 65 66 int menuIndex = fMenus->find(const_cast<SkOSMenu *>(menu)); 67 if (menuIndex >= 0 && menuIndex < fMenus->count()) { 68 NSUInteger first = 0; 69 for (int i = 0; i < menuIndex; ++i) { 70 first += (*fMenus)[i]->getCount(); 71 } 72 [fItems removeObjectsInRange:NSMakeRange(first, [fItems count] - first)]; 73 [self loadMenu:menu]; 74 } 75 [self reloadData]; 76} 77 78- (NSCellStateValue)triStateToNSState:(SkOSMenu::TriState)state { 79 if (SkOSMenu::kOnState == state) 80 return NSOnState; 81 else if (SkOSMenu::kOffState == state) 82 return NSOffState; 83 else 84 return NSMixedState; 85} 86 87- (void)loadMenu:(const SkOSMenu*)menu { 88 const SkOSMenu::Item* menuitems[menu->getCount()]; 89 menu->getItems(menuitems); 90 for (int i = 0; i < menu->getCount(); ++i) { 91 const SkOSMenu::Item* item = menuitems[i]; 92 SkOptionItem* option = [[SkOptionItem alloc] init]; 93 option.fItem = item; 94 95 if (SkOSMenu::kList_Type == item->getType()) { 96 int index = 0, count = 0; 97 SkOSMenu::FindListItemCount(*item->getEvent(), &count); 98 NSMutableArray* optionstrs = [[NSMutableArray alloc] initWithCapacity:count]; 99 std::unique_ptr<SkString[]> ada(new SkString[count]); 100 SkString* options = ada.get(); 101 SkOSMenu::FindListItems(*item->getEvent(), options); 102 for (int i = 0; i < count; ++i) 103 [optionstrs addObject:[NSString stringWithUTF8String:options[i].c_str()]]; 104 SkOSMenu::FindListIndex(*item->getEvent(), item->getSlotName(), &index); 105 option.fCell = [self createList:optionstrs current:index]; 106 [optionstrs release]; 107 } 108 else { 109 bool state = false; 110 SkString str; 111 SkOSMenu::TriState tristate; 112 switch (item->getType()) { 113 case SkOSMenu::kAction_Type: 114 option.fCell = [self createAction]; 115 break; 116 case SkOSMenu::kSlider_Type: 117 SkScalar min, max, value; 118 SkOSMenu::FindSliderValue(*item->getEvent(), item->getSlotName(), &value); 119 SkOSMenu::FindSliderMin(*item->getEvent(), &min); 120 SkOSMenu::FindSliderMax(*item->getEvent(), &max); 121 option.fCell = [self createSlider:value 122 min:min 123 max:max]; 124 break; 125 case SkOSMenu::kSwitch_Type: 126 SkOSMenu::FindSwitchState(*item->getEvent(), item->getSlotName(), &state); 127 option.fCell = [self createSwitch:(BOOL)state]; 128 break; 129 case SkOSMenu::kTriState_Type: 130 SkOSMenu::FindTriState(*item->getEvent(), item->getSlotName(), &tristate); 131 option.fCell = [self createTriState:[self triStateToNSState:tristate]]; 132 break; 133 case SkOSMenu::kTextField_Type: 134 SkOSMenu::FindText(*item->getEvent(),item->getSlotName(), &str); 135 option.fCell = [self createTextField:[NSString stringWithUTF8String:str.c_str()]]; 136 break; 137 default: 138 break; 139 } 140 } 141 [fItems addObject:option]; 142 [option release]; 143 } 144} 145 146- (NSInteger)numberOfRowsInTableView:(NSTableView *)tableView { 147 return [self.fItems count]; 148} 149 150- (id)tableView:(NSTableView *)tableView objectValueForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row { 151 NSInteger columnIndex = [tableView columnWithIdentifier:[tableColumn identifier]]; 152 if (columnIndex == 0) { 153 const SkOSMenu::Item* item = ((SkOptionItem*)[fItems objectAtIndex:row]).fItem; 154 NSString* label = [NSString stringWithUTF8String:item->getLabel()]; 155 if (fShowKeys) 156 return [NSString stringWithFormat:@"%@ (%c)", label, item->getKeyEquivalent()]; 157 else 158 return label; 159 } 160 else 161 return nil; 162} 163 164- (NSCell *)tableView:(NSTableView *)tableView dataCellForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row { 165 if (tableColumn) { 166 NSInteger columnIndex = [tableView columnWithIdentifier:[tableColumn identifier]]; 167 if (columnIndex == 1) 168 return [((SkOptionItem*)[fItems objectAtIndex:row]).fCell copy]; 169 else 170 return [[[SkTextFieldCell alloc] init] autorelease]; 171 } 172 return nil; 173} 174 175- (void)tableView:(NSTableView *)tableView willDisplayCell:(id)cell forTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row { 176 NSInteger columnIndex = [tableView columnWithIdentifier:[tableColumn identifier]]; 177 if (columnIndex == 1) { 178 SkOptionItem* option = (SkOptionItem*)[self.fItems objectAtIndex:row]; 179 NSCell* storedCell = option.fCell; 180 const SkOSMenu::Item* item = option.fItem; 181 switch (item->getType()) { 182 case SkOSMenu::kAction_Type: 183 break; 184 case SkOSMenu::kList_Type: 185 [cell selectItemAtIndex:[(NSPopUpButtonCell*)storedCell indexOfSelectedItem]]; 186 break; 187 case SkOSMenu::kSlider_Type: 188 [cell setFloatValue:[storedCell floatValue]]; 189 break; 190 case SkOSMenu::kSwitch_Type: 191 [cell setState:[(NSButtonCell*)storedCell state]]; 192 break; 193 case SkOSMenu::kTextField_Type: 194 if ([[storedCell stringValue] length] > 0) 195 [cell setStringValue:[storedCell stringValue]]; 196 break; 197 case SkOSMenu::kTriState_Type: 198 [cell setState:[(NSButtonCell*)storedCell state]]; 199 break; 200 default: 201 break; 202 } 203 } 204 else { 205 [(SkTextFieldCell*)cell setEditable:NO]; 206 } 207} 208 209- (void)tableView:(NSTableView *)tableView setObjectValue:(id)anObject forTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row { 210 NSInteger columnIndex = [tableView columnWithIdentifier:[tableColumn identifier]]; 211 if (columnIndex == 1) { 212 SkOptionItem* option = (SkOptionItem*)[self.fItems objectAtIndex:row]; 213 NSCell* cell = option.fCell; 214 const SkOSMenu::Item* item = option.fItem; 215 switch (item->getType()) { 216 case SkOSMenu::kAction_Type: 217 item->postEvent(); 218 break; 219 case SkOSMenu::kList_Type: 220 [(NSPopUpButtonCell*)cell selectItemAtIndex:[anObject intValue]]; 221 item->setInt([anObject intValue]); 222 break; 223 case SkOSMenu::kSlider_Type: 224 [cell setFloatValue:[anObject floatValue]]; 225 item->setScalar([anObject floatValue]); 226 break; 227 case SkOSMenu::kSwitch_Type: 228 [cell setState:[anObject boolValue]]; 229 item->setBool([anObject boolValue]); 230 break; 231 case SkOSMenu::kTextField_Type: 232 if ([anObject length] > 0) { 233 [cell setStringValue:anObject]; 234 item->setString([anObject UTF8String]); 235 } 236 break; 237 case SkOSMenu::kTriState_Type: 238 [cell setState:[anObject intValue]]; 239 item->setTriState((SkOSMenu::TriState)[anObject intValue]); 240 break; 241 default: 242 break; 243 } 244 item->postEvent(); 245 } 246} 247 248- (NSCell*)createAction{ 249 NSButtonCell* cell = [[[NSButtonCell alloc] init] autorelease]; 250 [cell setTitle:@""]; 251 [cell setButtonType:NSMomentaryPushInButton]; 252 [cell setBezelStyle:NSSmallSquareBezelStyle]; 253 return cell; 254} 255 256- (NSCell*)createList:(NSArray*)items current:(int)index { 257 NSPopUpButtonCell* cell = [[[NSPopUpButtonCell alloc] init] autorelease]; 258 [cell addItemsWithTitles:items]; 259 [cell selectItemAtIndex:index]; 260 [cell setArrowPosition:NSPopUpArrowAtBottom]; 261 [cell setBezelStyle:NSSmallSquareBezelStyle]; 262 return cell; 263} 264 265- (NSCell*)createSlider:(float)value min:(float)min max:(float)max { 266 NSSliderCell* cell = [[[NSSliderCell alloc] init] autorelease]; 267 [cell setFloatValue:value]; 268 [cell setMinValue:min]; 269 [cell setMaxValue:max]; 270 return cell; 271} 272 273- (NSCell*)createSwitch:(BOOL)state { 274 NSButtonCell* cell = [[[NSButtonCell alloc] init] autorelease]; 275 [cell setState:state]; 276 [cell setTitle:@""]; 277 [cell setButtonType:NSSwitchButton]; 278 return cell; 279} 280 281- (NSCell*)createTextField:(NSString*)placeHolder { 282 SkTextFieldCell* cell = [[[SkTextFieldCell alloc] init] autorelease]; 283 [cell setEditable:YES]; 284 [cell setStringValue:@""]; 285 [cell setPlaceholderString:placeHolder]; 286 return cell; 287} 288 289- (NSCell*)createTriState:(NSCellStateValue)state { 290 NSButtonCell* cell = [[[NSButtonCell alloc] init] autorelease]; 291 [cell setAllowsMixedState:TRUE]; 292 [cell setTitle:@""]; 293 [cell setState:(NSInteger)state]; 294 [cell setButtonType:NSSwitchButton]; 295 return cell; 296} 297@end 298