epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2011 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 7 | #include <stdarg.h> |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 8 | #include "SkOSMenu.h" |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 9 | #include "SkThread.h" |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 10 | |
| 11 | static int gOSMenuCmd = 7000; |
| 12 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 13 | SkOSMenu::SkOSMenu(const char title[]) { |
| 14 | fTitle.set(title); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 15 | } |
| 16 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 17 | SkOSMenu::~SkOSMenu() { |
| 18 | this->reset(); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 19 | } |
| 20 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 21 | void SkOSMenu::reset() { |
| 22 | fItems.deleteAll(); |
| 23 | fTitle.reset(); |
| 24 | } |
| 25 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 26 | void SkOSMenu::assignKeyEquivalentToItem(int itemID, SkUnichar key) { |
| 27 | for (int i = 0; i < fItems.count(); ++i) { |
| 28 | if (itemID == fItems[i]->getID()) |
| 29 | fItems[i]->setKeyEquivalent(key); |
| 30 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 31 | } |
| 32 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 33 | bool SkOSMenu::handleKeyEquivalent(SkUnichar key) { |
| 34 | int value = 0, size = 0; |
| 35 | bool state; |
| 36 | SkOSMenu::TriState tristate; |
| 37 | for (int i = 0; i < fItems.count(); ++i) { |
| 38 | Item* item = fItems[i]; |
| 39 | if (item->getKeyEquivalent()== key) { |
| 40 | SkString list; |
| 41 | switch (item->getType()) { |
| 42 | case kList_Type: |
| 43 | SkOSMenu::FindListItemCount(item->getEvent(), &size); |
| 44 | SkOSMenu::FindListIndex(item->getEvent(), item->getSlotName(), &value); |
| 45 | value = (value + 1) % size; |
| 46 | item->postEventWithInt(value); |
| 47 | break; |
| 48 | case kSwitch_Type: |
| 49 | SkOSMenu::FindSwitchState(item->getEvent(), item->getSlotName(), &state); |
| 50 | item->postEventWithBool(!state); |
| 51 | break; |
| 52 | case kTriState_Type: |
| 53 | SkOSMenu::FindTriState(item->getEvent(), item->getSlotName(), &tristate); |
| 54 | if (kOnState == tristate) |
| 55 | tristate = kMixedState; |
| 56 | else |
| 57 | tristate = (SkOSMenu::TriState)((int)tristate + 1); |
| 58 | item->postEventWithInt(tristate); |
| 59 | break; |
| 60 | case kAction_Type: |
| 61 | case kCustom_Type: |
| 62 | case kSlider_Type: |
| 63 | case kTextField_Type: |
| 64 | default: |
| 65 | item->postEvent(); |
| 66 | break; |
| 67 | } |
| 68 | return true; |
| 69 | } |
| 70 | } |
| 71 | return false; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 72 | } |
| 73 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 74 | //////////////////////////////////////////////////////////////////////////////// |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 75 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 76 | SkOSMenu::Item::Item(const char label[], SkOSMenu::Type type, |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 77 | const char slotName[], SkEvent* evt) { |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 78 | fLabel.set(label); |
| 79 | fSlotName.set(slotName); |
| 80 | fType = type; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 81 | fEvent = evt; |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 82 | fKey = 0; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 83 | fID = sk_atomic_inc(&gOSMenuCmd); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 84 | } |
| 85 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 86 | void SkOSMenu::Item::postEventWithBool(bool value) const { |
| 87 | SkASSERT(SkOSMenu::kSwitch_Type == fType); |
| 88 | fEvent->setBool(fSlotName.c_str(), value); |
| 89 | this->postEvent(); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 90 | } |
| 91 | |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 92 | void SkOSMenu::Item::postEventWithScalar(SkScalar value) const { |
| 93 | SkASSERT(SkOSMenu::kSlider_Type == fType); |
| 94 | fEvent->setScalar(fSlotName.c_str(), value); |
| 95 | this->postEvent(); |
| 96 | } |
| 97 | |
| 98 | void SkOSMenu::Item::postEventWithInt(int value) const { |
| 99 | SkASSERT(SkOSMenu::kList_Type == fType || SkOSMenu::kTriState_Type == fType); |
| 100 | fEvent->setS32(fSlotName.c_str(), value); |
| 101 | this->postEvent(); |
| 102 | } |
| 103 | |
| 104 | void SkOSMenu::Item::postEventWithString(const char value[]) const { |
| 105 | SkASSERT(SkOSMenu::kTextField_Type == fType); |
| 106 | fEvent->setString(fSlotName.c_str(), value); |
| 107 | this->postEvent(); |
| 108 | } |
| 109 | |
| 110 | //////////////////////////////////////////////////////////////////////////////// |
| 111 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 112 | static const char* gMenuEventType = "SkOSMenuEventType"; |
| 113 | static const char* gSlider_Min_Scalar = "SkOSMenuSlider_Min"; |
| 114 | static const char* gSlider_Max_Scalar = "SkOSMenuSlider_Max"; |
| 115 | static const char* gDelimiter = "|"; |
| 116 | static const char* gList_Items_Str = "SkOSMenuList_Items"; |
| 117 | static const char* gList_ItemCount_S32 = "SkOSMenuList_ItemCount"; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 118 | |
| 119 | int SkOSMenu::appendItem(const char label[], Type type, const char slotName[], |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 120 | SkEvent* evt) { |
| 121 | SkOSMenu::Item* item = new Item(label, type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 122 | fItems.append(1, &item); |
| 123 | return item->getID(); |
| 124 | } |
| 125 | |
| 126 | int SkOSMenu::appendAction(const char label[], SkEventSinkID target) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 127 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 128 | //Store label in event so it can be used to identify the action later |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 129 | evt->setString(label, label); |
| 130 | return appendItem(label, SkOSMenu::kAction_Type, "", evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 131 | } |
| 132 | |
| 133 | int SkOSMenu::appendList(const char label[], const char slotName[], |
| 134 | SkEventSinkID target, int index, const char option[], ...) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 135 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 136 | va_list args; |
| 137 | if (option) { |
| 138 | SkString str(option); |
| 139 | va_start(args, option); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 140 | int count = 1; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 141 | for (const char* arg = va_arg(args, const char*); arg != NULL; arg = va_arg(args, const char*)) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 142 | str += gDelimiter; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 143 | str += arg; |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 144 | ++count; |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 145 | } |
| 146 | va_end(args); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 147 | evt->setString(gList_Items_Str, str); |
| 148 | evt->setS32(gList_ItemCount_S32, count); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 149 | evt->setS32(slotName, index); |
| 150 | } |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 151 | return appendItem(label, SkOSMenu::kList_Type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 152 | } |
| 153 | |
| 154 | int SkOSMenu::appendSlider(const char label[], const char slotName[], |
| 155 | SkEventSinkID target, SkScalar min, SkScalar max, |
| 156 | SkScalar defaultValue) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 157 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
| 158 | evt->setScalar(gSlider_Min_Scalar, min); |
| 159 | evt->setScalar(gSlider_Max_Scalar, max); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 160 | evt->setScalar(slotName, defaultValue); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 161 | return appendItem(label, SkOSMenu::kSlider_Type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 162 | } |
| 163 | |
| 164 | int SkOSMenu::appendSwitch(const char label[], const char slotName[], |
| 165 | SkEventSinkID target, bool defaultState) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 166 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 167 | evt->setBool(slotName, defaultState); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 168 | return appendItem(label, SkOSMenu::kSwitch_Type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 169 | } |
| 170 | |
| 171 | int SkOSMenu::appendTriState(const char label[], const char slotName[], |
| 172 | SkEventSinkID target, SkOSMenu::TriState defaultState) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 173 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 174 | evt->setS32(slotName, defaultState); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 175 | return appendItem(label, SkOSMenu::kTriState_Type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 176 | } |
| 177 | |
| 178 | int SkOSMenu::appendTextField(const char label[], const char slotName[], |
| 179 | SkEventSinkID target, const char placeholder[]) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 180 | SkEvent* evt = new SkEvent(gMenuEventType, target); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 181 | evt->setString(slotName, placeholder); |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 182 | return appendItem(label, SkOSMenu::kTextField_Type, slotName, evt); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 183 | } |
| 184 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 185 | bool SkOSMenu::FindListItemCount(const SkEvent* evt, int* count) { |
| 186 | return evt->isType(gMenuEventType) && evt->findS32(gList_ItemCount_S32, count); |
| 187 | } |
| 188 | |
| 189 | bool SkOSMenu::FindListItems(const SkEvent* evt, SkString items[]) { |
| 190 | if (evt->isType(gMenuEventType) && NULL != items) { |
| 191 | const char* text = evt->findString(gList_Items_Str); |
| 192 | char temp[strlen(text)]; |
| 193 | memcpy(temp, text, strlen(text) + 1); //make sure to copy the null terminator |
| 194 | char* token = strtok(temp, gDelimiter); |
| 195 | int index = 0; |
| 196 | while (token != NULL) { |
| 197 | items[index].set(token, strlen(token)); |
| 198 | token = strtok (NULL, gDelimiter); |
| 199 | ++index; |
| 200 | } |
| 201 | return true; |
| 202 | } |
| 203 | return false; |
| 204 | } |
| 205 | |
| 206 | bool SkOSMenu::FindSliderMin(const SkEvent* evt, SkScalar* min) { |
| 207 | return evt->isType(gMenuEventType) && evt->findScalar(gSlider_Min_Scalar, min); |
| 208 | } |
| 209 | |
| 210 | bool SkOSMenu::FindSliderMax(const SkEvent* evt, SkScalar* max) { |
| 211 | return evt->isType(gMenuEventType) && evt->findScalar(gSlider_Max_Scalar, max); |
| 212 | } |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 213 | |
| 214 | bool SkOSMenu::FindAction(const SkEvent* evt, const char label[]) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 215 | return evt->isType(gMenuEventType) && evt->findString(label); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 216 | } |
| 217 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 218 | bool SkOSMenu::FindListIndex(const SkEvent* evt, const char slotName[], int* value) { |
| 219 | return evt->isType(gMenuEventType) && evt->findS32(slotName, value); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 220 | } |
| 221 | |
| 222 | bool SkOSMenu::FindSliderValue(const SkEvent* evt, const char slotName[], SkScalar* value) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 223 | return evt->isType(gMenuEventType) && evt->findScalar(slotName, value); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 224 | } |
| 225 | |
| 226 | bool SkOSMenu::FindSwitchState(const SkEvent* evt, const char slotName[], bool* value) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 227 | return evt->isType(gMenuEventType) && evt->findBool(slotName, value); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 228 | } |
| 229 | |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 230 | bool SkOSMenu::FindTriState(const SkEvent* evt, const char slotName[], SkOSMenu::TriState* value) { |
| 231 | return evt->isType(gMenuEventType) && evt->findS32(slotName, (int*)value); |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 232 | } |
| 233 | |
| 234 | bool SkOSMenu::FindText(const SkEvent* evt, const char slotName[], SkString* value) { |
yangsu@google.com | e55f533 | 2011-08-05 22:11:41 +0000 | [diff] [blame^] | 235 | if (evt->isType(gMenuEventType)) { |
yangsu@google.com | 654d72f | 2011-08-01 17:27:33 +0000 | [diff] [blame] | 236 | const char* text = evt->findString(slotName); |
| 237 | if (!text || !*text) |
| 238 | return false; |
| 239 | else { |
| 240 | value->set(text); |
| 241 | return true; |
| 242 | } |
| 243 | } |
| 244 | return false; |
| 245 | } |