voice.js 116 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063
  1. /**
  2. * Copyright (c) 2006-2015, JGraph Ltd
  3. * Copyright (c) 2006-2015, Gaudenz Alder
  4. */
  5. /**
  6. * Voice plugin for draw.io
  7. *
  8. * Documentation:
  9. *
  10. * https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin
  11. *
  12. * TODO: Use grammer https://msdn.microsoft.com/en-us/library/ee800145.aspx
  13. */
  14. Draw.loadPlugin(function(ui) {
  15. // Speech recognition never supported without synthesis
  16. if (!('speechSynthesis' in window))
  17. {
  18. ui.showError('Error', 'Speech output not supported in this browser.', 'OK');
  19. return;
  20. }
  21. else
  22. {
  23. // Triggers loading of voices
  24. speechSynthesis.getVoices();
  25. }
  26. // Do no use in chromeless mode
  27. if (ui.editor.chromeless || ui.menubar == null)
  28. {
  29. return;
  30. }
  31. // Mic PNG image
  32. var outputImage = '';
  33. // True if we're on ChromOs
  34. var chromeOs = /\bCrOS\b/.test(navigator.userAgent);
  35. // Maximum length of message to speak
  36. var maxMessageLength = 50;
  37. // Maximum length of the label before the cell
  38. // is called by its shapename
  39. var maxLabelLength = 15;
  40. // Maximum length of output queue.
  41. var maxQueueLength = 3;
  42. // Specifies if speech output is enabled.
  43. var speechOutputEnabled = true;
  44. // Specifies if speech output is enabled.
  45. var speechInputEnabled = true;
  46. // Last message is never repeated
  47. var lastMessage = null;
  48. // Timestamp of last message
  49. var lastMessageTimestamp = null;
  50. // Sets global recognition language
  51. var lang = 'en-US';
  52. // Caches action names
  53. var actions = null;
  54. var actionList = null;
  55. // Caches shape names
  56. var shapeList = null;
  57. // Last inserted cell
  58. var lastInserted = null;
  59. // Current voice
  60. var currentVoice = 10;
  61. // Current recognition thread
  62. var recognizing = null;
  63. // Adds menu
  64. mxResources.parse('voiceType=Voice Type');
  65. mxResources.parse('speechOutput=Speech Output');
  66. mxResources.parse('speechListen=Listen');
  67. mxResources.parse('speechInstalled=Start with draw.io');
  68. mxResources.parse('speechListenContinuous=Start/Stop Listen');
  69. mxResources.parse('speechHint=Hint');
  70. mxResources.parse('speechHelp=Help');
  71. mxResources.parse('speechQuit=Quit');
  72. // Installs footer click handler
  73. function getOrCreateVoiceButton(ui)
  74. {
  75. if (ui.voiceButton == null)
  76. {
  77. ui.voiceButton = document.createElement('div');
  78. ui.voiceButton.className = 'geBtn';
  79. ui.voiceButton.style.width = '140px';
  80. ui.voiceButton.style.minWidth = '140px';
  81. ui.voiceButton.style.textOverflow = 'ellipsis';
  82. ui.voiceButton.style.overflowX = 'hidden';
  83. ui.voiceButton.style.fontWeight = 'bold';
  84. ui.voiceButton.style.textAlign = 'center';
  85. ui.voiceButton.style.display = 'inline-block';
  86. ui.voiceButton.style.padding = '0 10px 0 10px';
  87. ui.voiceButton.style.marginTop = '-4px';
  88. ui.voiceButton.style.height = '28px';
  89. ui.voiceButton.style.lineHeight = '28px';
  90. ui.voiceButton.style.color = '#235695';
  91. if (ui.buttonContainer.firstChild != null)
  92. {
  93. ui.buttonContainer.insertBefore(ui.voiceButton, ui.buttonContainer.firstChild);
  94. }
  95. else
  96. {
  97. ui.buttonContainer.appendChild(ui.voiceButton);
  98. }
  99. }
  100. return ui.voiceButton;
  101. };
  102. var td = getOrCreateVoiceButton(ui);
  103. if (td != null)
  104. {
  105. // Installs listener for start/stop listen
  106. mxEvent.addListener(td, 'click', function(evt)
  107. {
  108. App.listen(true);
  109. });
  110. }
  111. function setPluginInstalled(value)
  112. {
  113. if (mxSettings != null)
  114. {
  115. var plugins = mxSettings.getPlugins();
  116. var installed = mxUtils.indexOf(plugins, '/plugins/voice.js') >= 0;
  117. if (value != installed)
  118. {
  119. if (installed)
  120. {
  121. mxUtils.remove('/plugins/voice.js', plugins);
  122. }
  123. else
  124. {
  125. plugins.push('/plugins/voice.js');
  126. }
  127. mxSettings.setPlugins(plugins);
  128. mxSettings.save();
  129. }
  130. }
  131. };
  132. // Shows initial status message if only output is enable
  133. if (!('webkitSpeechRecognition' in window))
  134. {
  135. if (td != null)
  136. {
  137. td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + outputImage + '"/> Ready';
  138. td.style.color = '#235695';
  139. }
  140. }
  141. function updateStatusMessage()
  142. {
  143. if ('webkitSpeechRecognition' in window)
  144. {
  145. if (td != null)
  146. {
  147. if (recognizing != null)
  148. {
  149. td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + ui.micImage + '"/> Listening...';
  150. td.setAttribute('title', 'Click to Stop (Ctrl+O)');
  151. td.style.color = 'darkGray';
  152. }
  153. else
  154. {
  155. td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + ui.micImage + '"/> Click to Speak';
  156. td.setAttribute('title', 'Click to Speak (Ctrl+O)');
  157. td.style.color = '#235695';
  158. }
  159. }
  160. }
  161. };
  162. updateStatusMessage();
  163. var action = ui.actions.addAction('speechOutput', function()
  164. {
  165. speechOutputEnabled = !speechOutputEnabled;
  166. }, null, null, 'Ctrl/AltGr+Shift+Esc');
  167. action.setToggleAction(true);
  168. action.setSelectedCallback(function() { return speechOutputEnabled; });
  169. var action = ui.actions.addAction('speechInstalled', function()
  170. {
  171. setPluginInstalled();
  172. });
  173. action.setToggleAction(true);
  174. action.setSelectedCallback(function() { return mxUtils.indexOf(mxSettings.getPlugins(), '/plugins/voice.js') >= 0; });
  175. ui.actions.addAction('speechListen', function()
  176. {
  177. App.listen();
  178. }, null, null, 'Ctrl/AltGr+Esc');
  179. ui.actions.addAction('speechListenContinuous', function()
  180. {
  181. App.listen(true);
  182. }, null, null, 'Ctrl+O');
  183. ui.actions.addAction('speechHint', function()
  184. {
  185. App.sayHint();
  186. }, null, null, 'Shift+Esc');
  187. ui.actions.addAction('speechHelp', function()
  188. {
  189. window.open('https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin');
  190. });
  191. // Hijacks the settings for storing current voice
  192. if (mxSettings != null)
  193. {
  194. var tmp = mxSettings.settings.voice;
  195. if (tmp != null)
  196. {
  197. currentVoice = parseInt(tmp);
  198. }
  199. }
  200. ui.menus.put('voiceType', new Menu(mxUtils.bind(this, function(menu, parent)
  201. {
  202. var voices = speechSynthesis.getVoices();
  203. if (voices.length == 0)
  204. {
  205. menu.addItem('Loading...', null, function() {}, parent, null, false);
  206. }
  207. else
  208. {
  209. for (var i = 0; i < voices.length; i++)
  210. {
  211. (function(index)
  212. {
  213. var item = menu.addItem(voices[index].name + ' (' + voices[i].lang + ')', null, function()
  214. {
  215. currentVoice = index;
  216. App.say('hello');
  217. if (mxSettings != null)
  218. {
  219. mxSettings.settings.voice = currentVoice;
  220. mxSettings.save();
  221. }
  222. }, parent);
  223. if (index == currentVoice)
  224. {
  225. menu.addCheckmark(item, Editor.checkmarkImage);
  226. }
  227. })(i);
  228. }
  229. if (!mxClient.IS_QUIRKS)
  230. {
  231. parent.div.style.overflowX = 'hidden';
  232. parent.div.style.overflowY = 'auto';
  233. parent.div.style.maxHeight = '100%';
  234. // Workaround for document scrollbars with 100% max height in Chrome
  235. parent.div.style.marginBottom = '-20px';
  236. }
  237. }
  238. })));
  239. ui.actions.addAction('speechQuit', function()
  240. {
  241. // Hides UI
  242. speechOutputEnabled = false;
  243. menu.style.display = 'none';
  244. td.style.display = 'none';
  245. });
  246. var menu = ui.menubar.addMenu('Voice', function(menu, parent)
  247. {
  248. ui.menus.addSubmenu('voiceType', menu, parent);
  249. ui.menus.addMenuItems(menu, ['-', 'speechOutput', 'speechHint', '-', 'speechListen',
  250. 'speechListenContinuous', '-', 'speechInstalled',
  251. 'speechHelp', '-', 'speechQuit']);
  252. });
  253. // Inserts voice menu before help menu
  254. var menu = menu.parentNode.insertBefore(menu, menu.previousSibling.previousSibling.previousSibling);
  255. function insertShape(shape, done)
  256. {
  257. var searchTerm = mxUtils.trim(shape);
  258. ui.sidebar.searchEntries(searchTerm, 1, 0, function(results, len, more)
  259. {
  260. if (results.length > 0)
  261. {
  262. var elt = results[0]();
  263. // Click is blocked, must use mousedown/-up sequence
  264. // LATER: Use touchstart or pointerEvents depending on system
  265. dispatchEvent(elt, mouseEvent('mousedown', 1, 50, 1, 50));
  266. dispatchEvent(document.body, mouseEvent('mouseup', 1, 50, 1, 50));
  267. }
  268. else
  269. {
  270. App.say('{1} not found', [searchTerm]);
  271. }
  272. if (done != null)
  273. {
  274. done();
  275. }
  276. });
  277. };
  278. // http://stackoverflow.com/questions/11919065/sort-an-array-by-the-levenshtein-distance-with-best-performance-in-javascript
  279. //http://www.merriampark.com/ld.htm, http://www.mgilleland.com/ld/ldjavascript.htm, Damerau–Levenshtein distance (Wikipedia)
  280. var levenshteinDist = function(s, t) {
  281. var d = []; //2d matrix
  282. // Step 1
  283. var n = s.length;
  284. var m = t.length;
  285. if (n == 0) return m;
  286. if (m == 0) return n;
  287. //Create an array of arrays in javascript (a descending loop is quicker)
  288. for (var i = n; i >= 0; i--) d[i] = [];
  289. // Step 2
  290. for (var i = n; i >= 0; i--) d[i][0] = i;
  291. for (var j = m; j >= 0; j--) d[0][j] = j;
  292. // Step 3
  293. for (var i = 1; i <= n; i++) {
  294. var s_i = s.charAt(i - 1);
  295. // Step 4
  296. for (var j = 1; j <= m; j++) {
  297. //Check the jagged ld total so far
  298. if (i == j && d[i][j] > 4) return n;
  299. var t_j = t.charAt(j - 1);
  300. var cost = (s_i == t_j) ? 0 : 1; // Step 5
  301. //Calculate the minimum
  302. var mi = d[i - 1][j] + 1;
  303. var b = d[i][j - 1] + 1;
  304. var c = d[i - 1][j - 1] + cost;
  305. if (b < mi) mi = b;
  306. if (c < mi) mi = c;
  307. d[i][j] = mi; // Step 6
  308. //Damerau transposition
  309. if (i > 1 && j > 1 && s_i == t.charAt(j - 2) && s.charAt(i - 2) == t_j) {
  310. d[i][j] = Math.min(d[i][j], d[i - 2][j - 2] + cost);
  311. }
  312. }
  313. }
  314. // Step 7
  315. return d[n][m];
  316. }
  317. function naiveHammingDistance(str1, str2) {
  318. var dist = 0;
  319. str1 = str1.toLowerCase();
  320. str2 = str2.toLowerCase();
  321. for(var i = 0; i < str1.length; i++)
  322. {
  323. if (str2[i] && str2[i] !== str1[i])
  324. {
  325. dist += Math.abs(str1.charCodeAt(i) - str2.charCodeAt(i)) + Math.abs(str2.indexOf( str1[i] )) * 2;
  326. }
  327. else if (!str2[i])
  328. {
  329. // If there's no letter in the comparing string
  330. dist += dist;
  331. }
  332. }
  333. return dist;
  334. };
  335. function getBestWord(str1, words, useLevenshteinDist)
  336. {
  337. useLevenshteinDist = (useLevenshteinDist != null) ? useLevenshteinDist : true;
  338. var bestWord = words[0];
  339. var minDist = ((useLevenshteinDist) ? levenshteinDist(str1, bestWord) :
  340. naiveHammingDistance(str1, bestWord));
  341. for (var i = 1; i < words.length; i++)
  342. {
  343. var tmp = ((useLevenshteinDist) ? levenshteinDist(str1, words[i]) :
  344. ((str1 == words[i]) ? 0 : naiveHammingDistance(str1, words[i])));
  345. if (tmp < minDist || (tmp == minDist &&
  346. str1.length > bestWord.length &&
  347. bestWord.length < words[i].length))
  348. {
  349. bestWord = words[i];
  350. minDist = tmp;
  351. }
  352. if (bestWord == str1)
  353. {
  354. break;
  355. }
  356. }
  357. return bestWord;
  358. }
  359. function mouseEvent(type, sx, sy, cx, cy, shift)
  360. {
  361. var evt;
  362. var e = {
  363. bubbles: true,
  364. cancelable: (type != "mousemove"),
  365. view: window,
  366. detail: 0,
  367. screenX: sx,
  368. screenY: sy,
  369. clientX: cx,
  370. clientY: cy,
  371. ctrlKey: false,
  372. altKey: false,
  373. shiftKey: (shift != null) ? shift : false,
  374. metaKey: false,
  375. button: 0,
  376. relatedTarget: undefined
  377. };
  378. if (typeof( document.createEvent ) == "function")
  379. {
  380. evt = document.createEvent("MouseEvents");
  381. evt.initMouseEvent(type,
  382. e.bubbles, e.cancelable, e.view, e.detail,
  383. e.screenX, e.screenY, e.clientX, e.clientY,
  384. e.ctrlKey, e.altKey, e.shiftKey, e.metaKey,
  385. e.button, document.body.parentNode);
  386. }
  387. else if (document.createEventObject)
  388. {
  389. evt = document.createEventObject();
  390. for (prop in e)
  391. {
  392. evt[prop] = e[prop];
  393. }
  394. evt.button = { 0:1, 1:4, 2:2 }[evt.button] || evt.button;
  395. }
  396. return evt;
  397. };
  398. function dispatchEvent (el, evt)
  399. {
  400. if (el.dispatchEvent)
  401. {
  402. el.dispatchEvent(evt);
  403. }
  404. else if (el.fireEvent)
  405. {
  406. el.fireEvent('on' + type, evt);
  407. }
  408. return evt;
  409. };
  410. var keyHandlerEscape = ui.keyHandler.escape;
  411. ui.keyHandler.escape = function(evt)
  412. {
  413. if ((!mxClient.IS_MAC && mxEvent.isAltDown(evt)) ||
  414. ((mxClient.IS_MAC || chromeOs) && mxEvent.isControlDown(evt)))
  415. {
  416. if (speechOutputEnabled)
  417. {
  418. App.say('Speech output disabled');
  419. }
  420. speechOutputEnabled = !speechOutputEnabled;
  421. if (speechOutputEnabled)
  422. {
  423. App.say('Speech output enabled');
  424. }
  425. mxEvent.consume(evt);
  426. }
  427. else if (mxEvent.isShiftDown(evt))
  428. {
  429. App.sayHint();
  430. mxEvent.consume(evt);
  431. }
  432. else
  433. {
  434. keyHandlerEscape.apply(this, arguments);
  435. }
  436. };
  437. ui.keyHandler.bindAction(32, true, 'speechListen'); // Ctrl+SPACE
  438. ui.keyHandler.bindAction(79, true, 'speechListenContinuous'); // Ctrl+O
  439. /**
  440. * Plays a beep.
  441. */
  442. var beep = new Audio('data:audio/wav;base64,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qJrx4KOQjahZxWbcZgztj2c49nKmkId44S71j0c8eV9yDK6uPRzx5X18eDvjvQ6yKo9ZSS6l//8elePK/Lf//IInrOF/FvDoADYAGBMGb7FtErm5MXMlmPAJQVgWta7Zx2go+8xJ0UiCb8LHHdftWyLJE0QIAIsI+UbXu67dZMjmgDGCGl1H+vpF4NSDckSIkk7Vd+sxEhBQMRU8j/12UIRhzSaUdQ+rQU5kGeFxm+hb1oh6pWWmv3uvmReDl0UnvtapVaIzo1jZbf/pD6ElLqSX+rUmOQNpJFa/r+sa4e/pBlAABoAAAAA3CUgShLdGIxsY7AUABPRrgCABdDuQ5GC7DqPQCgbbJUAoRSUj+NIEig0YfyWUho1VBBBA//uQZB4ABZx5zfMakeAAAAmwAAAAF5F3P0w9GtAAACfAAAAAwLhMDmAYWMgVEG1U0FIGCBgXBXAtfMH10000EEEEEECUBYln03TTTdNBDZopopYvrTTdNa325mImNg3TTPV9q3pmY0xoO6bv3r00y+IDGid/9aaaZTGMuj9mpu9Mpio1dXrr5HERTZSmqU36A3CumzN/9Robv/Xx4v9ijkSRSNLQhAWumap82WRSBUqXStV/YcS+XVLnSS+WLDroqArFkMEsAS+eWmrUzrO0oEmE40RlMZ5+ODIkAyKAGUwZ3mVKmcamcJnMW26MRPgUw6j+LkhyHGVGYjSUUKNpuJUQoOIAyDvEyG8S5yfK6dhZc0Tx1KI/gviKL6qvvFs1+bWtaz58uUNnryq6kt5RzOCkPWlVqVX2a/EEBUdU1KrXLf40GoiiFXK///qpoiDXrOgqDR38JB0bw7SoL+ZB9o1RCkQjQ2CBYZKd/+VJxZRRZlqSkKiws0WFxUyCwsKiMy7hUVFhIaCrNQsKkTIsLivwKKigsj8XYlwt/WKi2N4d//uQRCSAAjURNIHpMZBGYiaQPSYyAAABLAAAAAAAACWAAAAApUF/Mg+0aohSIRobBAsMlO//Kk4soosy1JSFRYWaLC4qZBYWFRGZdwqKiwkNBVmoWFSJkWFxX4FFRQWR+LsS4W/rFRb/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////VEFHAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAU291bmRib3kuZGUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMjAwNGh0dHA6Ly93d3cuc291bmRib3kuZGUAAAAAAAAAACU=');
  443. var beep2 = new Audio('data:audio/wav;base64,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');
  444. App.beep = function(wav)
  445. {
  446. wav = (wav != null) ? wav : beep;
  447. wav.play();
  448. };
  449. // Thread to reset the label
  450. var resetStatus = null;
  451. /**
  452. *
  453. * Static method for speech output.
  454. */
  455. App.say = function(message, params)
  456. {
  457. if ('speechSynthesis' in window && message != null)
  458. {
  459. var text = mxResources.replacePlaceholders(message, params || []);
  460. lastMessageTimestamp = null;
  461. lastMessage = text;
  462. if (text != null && text.length > 0)
  463. {
  464. if (td != null)
  465. {
  466. var tmp = text;
  467. // Capitalize string
  468. if (tmp != null && tmp.length > 1)
  469. {
  470. tmp = tmp.charAt(0).toUpperCase() + tmp.slice(1);
  471. }
  472. td.innerHTML = ((speechOutputEnabled) ? '<img style="margin-right:4px;" align="absmiddle" border="0" src="' +
  473. outputImage + '"/>' : '') + ' ' + tmp;
  474. td.style.color = '#235695';
  475. if (resetStatus != null)
  476. {
  477. window.clearTimeout(resetStatus);
  478. resetStatus = null;
  479. }
  480. resetStatus = window.setTimeout(function()
  481. {
  482. updateStatusMessage();
  483. }, (recognizing != null) ? 1000 : 3000);
  484. }
  485. // Workaround for talking too much
  486. if (speechOutputEnabled && (!speechSynthesis.speaking || !speechSynthesis.pending))
  487. {
  488. if (text.length < maxMessageLength)
  489. {
  490. var msg = new SpeechSynthesisUtterance();
  491. // Picks random voice with same main locale
  492. var voices = speechSynthesis.getVoices();
  493. // Say "again" for same last message except more than 10 secs ago or shorter than again
  494. if (lastMessageTimestamp != null && text == lastMessage && text != null &&
  495. text.length > 5 && lastMessage != 'again')
  496. {
  497. if (lastMessageTimestamp != null &&
  498. new Date().getTime() - lastMessageTimestamp.getTime() < 10000)
  499. {
  500. text = 'repeat';
  501. }
  502. }
  503. msg.voice = voices[currentVoice];
  504. msg.voiceURI = 'native';
  505. //msg.lang = lang;
  506. msg.text = text;
  507. console.log('App.say speak:', msg.text);
  508. speechSynthesis.speak(msg);
  509. lastMessageTimestamp = new Date();
  510. }
  511. else
  512. {
  513. console.log('App.say ignored:', text);
  514. }
  515. }
  516. else
  517. {
  518. console.log('App.say skipped:', message);
  519. }
  520. }
  521. }
  522. };
  523. /**
  524. * Static method for speech output.
  525. */
  526. App.listen = function(continuous)
  527. {
  528. if ('webkitSpeechRecognition' in window && speechInputEnabled)
  529. {
  530. if (recognizing != null)
  531. {
  532. recognizing.stop();
  533. }
  534. else
  535. {
  536. var recognition = new webkitSpeechRecognition();
  537. recognition.interimResults = true;
  538. // TODO: Should use grammar instead of trying more alternatives
  539. recognition.maxAlternatives = 5;
  540. recognition.lang = lang;
  541. if (continuous != null)
  542. {
  543. recognition.continuous = continuous;
  544. }
  545. recognition.onstart = function(event)
  546. {
  547. updateStatusMessage();
  548. App.beep();
  549. };
  550. recognition.onresult = function(event)
  551. {
  552. for (var i = event.resultIndex; i < event.results.length; ++i)
  553. {
  554. if (td != null)
  555. {
  556. td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + ui.micImage +
  557. '"/> ' + event.results[i][0].transcript;
  558. td.style.color = (event.results[i].isFinal) ? '#235695' : 'darkGray';
  559. }
  560. if (event.results[i].isFinal)
  561. {
  562. var ok = false;
  563. for (var j = 0; j < event.results[i].length; j++)
  564. {
  565. if (App.executeVoiceCommand(event.results[i][j].transcript, recognition))
  566. {
  567. ok = true;
  568. break;
  569. }
  570. }
  571. if (!ok)
  572. {
  573. App.say('{1} not found', [event.results[i][0].transcript]);
  574. }
  575. if (td != null)
  576. {
  577. if (resetStatus != null)
  578. {
  579. window.clearTimeout(resetStatus);
  580. resetStatus = null;
  581. }
  582. resetStatus = window.setTimeout(function()
  583. {
  584. updateStatusMessage();
  585. }, 1000);
  586. }
  587. }
  588. }
  589. };
  590. recognition.onend = function(event)
  591. {
  592. // Overrides footer
  593. recognizing = null;
  594. updateStatusMessage();
  595. App.beep(beep2);
  596. };
  597. recognition.start();
  598. recognizing = recognition;
  599. }
  600. }
  601. else
  602. {
  603. speechOutputEnabled = !speechOutputEnabled;
  604. lastMessageTimestamp = null;
  605. App.say(lastMessage || 'Ready');
  606. lastMessageTimestamp = null;
  607. }
  608. };
  609. /**
  610. * Executes the given voice command.
  611. */
  612. App.executeVoiceCommand = function(command, recognition)
  613. {
  614. console.log('App.execute:', mxUtils.trim(command));
  615. var tokens = mxUtils.trim(command).split(' ');
  616. if (tokens.length > 0 && graph.isEnabled())
  617. {
  618. // Ask for Mic permissions
  619. // FIXME: Dialog seems to be hidden until tab is changed
  620. function resolveStylename(token, styles)
  621. {
  622. var tmp = token.toLowerCase().replace(/ /g, '');
  623. var style = null;
  624. for (var i = 0; i < styles.length; i++)
  625. {
  626. if (styles[i].toLowerCase() == tmp)
  627. {
  628. style = styles[i];
  629. break;
  630. }
  631. }
  632. return style;
  633. };
  634. // Main command
  635. tokens[0] = tokens[0].toLowerCase();
  636. // TODO: Use hamming distance for best match command but include all possible actions
  637. // which might be too slow
  638. // console.log('connect', naiveHammingDistance(tokens[0], 'connect'), naiveHammingDistance('disable', 'connect'), naiveHammingDistance('change', 'connect'));
  639. if (graph.isEditing())
  640. {
  641. if (tokens.length == 1 && tokens[0] == 'apply')
  642. {
  643. graph.stopEditing();
  644. }
  645. else if (tokens.length == 1 && tokens[0] == 'undo')
  646. {
  647. document.execCommand('undo', false, null);
  648. }
  649. else if (tokens.length == 1 && tokens[0] == 'redo')
  650. {
  651. document.execCommand('redo', false, null);
  652. }
  653. else
  654. {
  655. document.execCommand('insertHTML', false, command);
  656. }
  657. return true;
  658. }
  659. else if (tokens[0] == 'edit' && tokens[1] == 'text')
  660. {
  661. var cells = graph.getSelectionCells();
  662. if (cells.length == 1)
  663. {
  664. graph.startEditingAtCell(cells[0]);
  665. }
  666. return true;
  667. }
  668. else if (tokens[0] == 'hello' || tokens[0] == 'hi')
  669. {
  670. App.say('Hello! Try "Help", "Help Topic" or "Quick Start".');
  671. return true;
  672. }
  673. else if (tokens[0] == 'help')
  674. {
  675. var wnd = window.open('https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin');
  676. if (wnd == null)
  677. {
  678. App.say('Popup blocked');
  679. }
  680. else if (tokens.length > 1)
  681. {
  682. // Just used to check if popup windows are allowed
  683. wnd.close();
  684. var searchTerm = mxUtils.trim(command.substring(tokens[0].length));
  685. if (searchTerm != null && searchTerm.length > 0)
  686. {
  687. var form = document.createElement('div');
  688. form.style.display = 'inline';
  689. form.innerHTML = ':<form style="display:inline;margin-left:8px;" id="rw_search_form"' +
  690. 'target="_blank" method="get" action="https://support.draw.io/dosearchsite.action">' +
  691. '<input id="rw_search_query" type="text" name="queryString" size="25"></form>';
  692. var realForm = form.getElementsByTagName('form')[0]
  693. var input = form.getElementsByTagName('input')[0];
  694. input.setAttribute('value', searchTerm);
  695. realForm.submit();
  696. App.say(command);
  697. }
  698. }
  699. else
  700. {
  701. App.say('help');
  702. }
  703. return true;
  704. }
  705. else if ((tokens[0] == 'info' && tokens.length == 1) || mxUtils.trim(command).toLowerCase() == 'what\'s this')
  706. {
  707. App.sayHint();
  708. return true;
  709. }
  710. else if (tokens[0] == 'install' && tokens.length == 1)
  711. {
  712. setPluginInstalled(true);
  713. App.say('Installed');
  714. return true;
  715. }
  716. else if (tokens[0] == 'uninstall' && tokens.length == 1)
  717. {
  718. setPluginInstalled(false);
  719. App.say('Uninstalled');
  720. return true;
  721. }
  722. else if (tokens[0] == 'quick' && tokens.length > 0 && tokens[1].toLowerCase() == 'start')
  723. {
  724. var wnd = window.open('https://youtu.be/8OaMWa4R1SE?t=1');
  725. if (wnd == null)
  726. {
  727. App.say('Popup blocked');
  728. }
  729. return true;
  730. }
  731. else if (tokens[0] == 'search' && tokens.length > 1)
  732. {
  733. var searchToken = tokens.slice(1, tokens.length).join(' ').toLowerCase();
  734. var wnd = window.open('https://www.google.ch/search?q=' + encodeURIComponent(searchToken) + '&tbm=isch', 'voicePluginSearchResult');
  735. if (wnd == null)
  736. {
  737. App.say('Popup blocked');
  738. }
  739. return true;
  740. }
  741. else if (tokens[0] == 'shrink')
  742. {
  743. var cell = graph.getSelectionCell();
  744. var geo = graph.getCellGeometry(cell);
  745. if (graph.getModel().isVertex(cell) && geo != null)
  746. {
  747. geo = geo.clone();
  748. if (tokens.length == 1 || tokens[1] == 'height')
  749. {
  750. geo.height = graph.snap(Math.round(geo.height * 0.5));
  751. }
  752. if (tokens.length == 1 || tokens[1] != 'height')
  753. {
  754. geo.width = graph.snap(Math.round(geo.width * 0.5));
  755. }
  756. if (geo.width > graph.tolerance && geo.height > graph.tolerance)
  757. {
  758. graph.getModel().setGeometry(cell, geo);
  759. App.say('Resized');
  760. }
  761. else
  762. {
  763. App.say('Too small');
  764. }
  765. }
  766. else
  767. {
  768. App.say('No cell to resize');
  769. }
  770. return true;
  771. }
  772. else if (tokens[0] == 'double')
  773. {
  774. var cell = graph.getSelectionCell();
  775. var geo = graph.getCellGeometry(cell);
  776. if (graph.getModel().isVertex(cell) && geo != null)
  777. {
  778. geo = geo.clone();
  779. if (tokens.length == 1 || tokens[1] == 'height')
  780. {
  781. geo.height *= 2;
  782. }
  783. if (tokens.length == 1 || tokens[1] != 'height')
  784. {
  785. geo.width *= 2;
  786. }
  787. if (geo.width > graph.tolerance && geo.height > graph.tolerance)
  788. {
  789. graph.getModel().setGeometry(cell, geo);
  790. App.say('Resized');
  791. }
  792. else
  793. {
  794. App.say('Too small');
  795. }
  796. }
  797. else
  798. {
  799. App.say('No cell to resize');
  800. }
  801. return true;
  802. }
  803. else if (tokens[0] == 'width' || tokens[0] == 'with' || tokens[0] == 'height')
  804. {
  805. // Try numeric stylename if last token is numeric
  806. var lastToken = tokens[tokens.length - 1].toLowerCase();
  807. // Fixes some special cases for recoginition of short phrases
  808. // like "stroke width 2" where it tries to be clever
  809. if (lastToken == 'tool' || lastToken == 'tune' || lastToken == 'tomb' || lastToken == 'tube')
  810. {
  811. lastToken = '2';
  812. }
  813. else if (lastToken == 'd3')
  814. {
  815. lastToken = '3';
  816. }
  817. else if (lastToken == 'v')
  818. {
  819. lastToken = '5';
  820. }
  821. else
  822. {
  823. // Converts numeric words to numbers (system only seems to return written numbers for <= 4)
  824. var numbers = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine'];
  825. var tmpIndex = mxUtils.indexOf(numbers, lastToken);
  826. if (tmpIndex >= 0)
  827. {
  828. lastToken = tmpIndex;
  829. }
  830. }
  831. var lastValue = parseFloat(tokens[1].toLowerCase());
  832. if (tokens.length >= 2 && !isNaN(lastValue))
  833. {
  834. var cell = graph.getSelectionCell();
  835. var geo = graph.getCellGeometry(cell);
  836. if (graph.getModel().isVertex(cell) && geo != null)
  837. {
  838. geo = geo.clone();
  839. if (tokens[0] == 'height')
  840. {
  841. geo.height = lastValue;
  842. }
  843. else
  844. {
  845. geo.width = lastValue;
  846. }
  847. graph.getModel().setGeometry(cell, geo);
  848. App.say('Resized');
  849. }
  850. else
  851. {
  852. App.say('No cell to resize');
  853. }
  854. }
  855. return true;
  856. }
  857. else if (tokens[0] == 'insert' && tokens.length > 1)
  858. {
  859. // Fixes some common mistakes
  860. if (tokens[1].toLowerCase() == 'edits')
  861. {
  862. tokens[1] = 'ellipse';
  863. }
  864. var searchTerm = mxUtils.trim(tokens.slice(1, tokens.length).join(' '));
  865. var current = graph.getSelectionCell();
  866. // Clears selection to disable built-in connecting
  867. graph.clearSelection();
  868. insertShape(searchTerm, function()
  869. {
  870. var cell = graph.getSelectionCell();
  871. // Connects dangling edge of previously selected edge and moves cell
  872. if (graph.model.isVertex(cell) && graph.model.isEdge(current) &&
  873. (graph.model.getTerminal(current, true) == null ||
  874. graph.model.getTerminal(current, false) == null))
  875. {
  876. var edgeState = graph.view.getState(current);
  877. var vertexState = graph.view.getState(cell);
  878. if (vertexState != null && edgeState != null && edgeState.absolutePoints != null &&
  879. edgeState.absolutePoints.length > 1)
  880. {
  881. var source = graph.model.getTerminal(current, true) == null;
  882. var pts = edgeState.absolutePoints;
  883. var pt = pts[(source) ? 0 : pts.length - 1];
  884. var loc = new mxPoint(vertexState.getCenterX(), vertexState.getCenterY());
  885. if (loc != null && edgeState != null)
  886. {
  887. var s = graph.view.scale;
  888. var dx = pt.x - loc.x;
  889. var dy = pt.y - loc.y;
  890. // TODO: Should add insert to transaction but need absolute position
  891. graph.model.beginUpdate();
  892. try
  893. {
  894. graph.moveCells(graph.getSelectionCells(), dx / s, dy / s);
  895. graph.model.setTerminal(current, cell, source);
  896. }
  897. finally
  898. {
  899. graph.model.endUpdate();
  900. }
  901. }
  902. }
  903. }
  904. });
  905. return true;
  906. }
  907. else if (tokens[0] == 'connect' || tokens[0] == 'kinect' || (tokens[0] == 'clone' && tokens.length > 1))
  908. {
  909. var cell = graph.getSelectionCell();
  910. if (graph.getModel().isVertex(cell))
  911. {
  912. // Uses east direction if token not understood
  913. var direction = mxConstants.DIRECTION_EAST;
  914. if (tokens.length > 1)
  915. {
  916. tokens[1] = tokens[1].toLowerCase();
  917. // Guessing direction based on minimum hamming distance for given set
  918. var guess = getBestWord(tokens[1], ['up', 'left', 'down', 'right', 'north', 'south', 'east', 'west']);
  919. if (guess == 'up' || guess == mxConstants.DIRECTION_NORTH)
  920. {
  921. direction = mxConstants.DIRECTION_NORTH;
  922. }
  923. else if (guess == 'left' || guess == mxConstants.DIRECTION_WEST)
  924. {
  925. direction = mxConstants.DIRECTION_WEST;
  926. }
  927. else if (guess == 'down' || guess == mxConstants.DIRECTION_SOUTH)
  928. {
  929. direction = mxConstants.DIRECTION_SOUTH;
  930. }
  931. }
  932. var length = graph.defaultEdgeLength;
  933. var cloneSource = tokens[0] == 'clone';
  934. var evt = mouseEvent('click', 1, 50, 1, 50, !cloneSource);
  935. var cells = graph.connectVertex(cell, direction, length, evt);
  936. graph.selectCellsForConnectVertex(cells, evt, ui.hoverIcons);
  937. }
  938. return true;
  939. }
  940. // Fixes drive and ride common mistakes for right
  941. else if (mxUtils.indexOf(['and', 'drive', 'ride', 'move', 'up', 'left', 'down', 'right', 'north', 'south', 'east', 'west', 'downright'], tokens[0]) >= 0)
  942. {
  943. var cell = graph.getSelectionCell();
  944. if (!graph.isSelectionEmpty())
  945. {
  946. // Uses east direction if token not understood
  947. var dx = 0;
  948. var dy = 0;
  949. for (var i = 0; i < tokens.length; i++)
  950. {
  951. tokens[i] = tokens[i].toLowerCase();
  952. var direction = null;
  953. // Downright is a single word needs special handling
  954. if (tokens[i].toLowerCase() == 'downright')
  955. {
  956. dx += graph.defaultEdgeLength;
  957. dy += graph.defaultEdgeLength;
  958. }
  959. else
  960. {
  961. var guess = null;
  962. // Guessing direction based on minimum hamming distance for given set
  963. // Handle some common cases
  964. if (tokens[i] == 'drive' || tokens[i] == 'ride')
  965. {
  966. guess = 'right';
  967. }
  968. else
  969. {
  970. guess = getBestWord(tokens[i], ['move', 'and', 'up', 'left', 'right', 'north', 'south', 'east', 'west', 'down', 'downright']);
  971. }
  972. if (guess == 'up' || guess == mxConstants.DIRECTION_NORTH)
  973. {
  974. direction = mxConstants.DIRECTION_NORTH;
  975. }
  976. else if (guess == 'left' || guess == mxConstants.DIRECTION_WEST)
  977. {
  978. direction = mxConstants.DIRECTION_WEST;
  979. }
  980. else if (guess == 'down' || guess == mxConstants.DIRECTION_SOUTH)
  981. {
  982. direction = mxConstants.DIRECTION_SOUTH;
  983. }
  984. else if (guess == 'right' || guess == mxConstants.DIRECTION_EAST)
  985. {
  986. direction = mxConstants.DIRECTION_EAST;
  987. }
  988. if (direction != null)
  989. {
  990. if (direction == mxConstants.DIRECTION_NORTH)
  991. {
  992. dy += -graph.defaultEdgeLength;
  993. }
  994. else if (direction == mxConstants.DIRECTION_WEST)
  995. {
  996. dx += -graph.defaultEdgeLength;
  997. }
  998. else if (direction == mxConstants.DIRECTION_SOUTH)
  999. {
  1000. dy += graph.defaultEdgeLength;
  1001. }
  1002. else if (direction == mxConstants.DIRECTION_EAST)
  1003. {
  1004. dx += graph.defaultEdgeLength;
  1005. }
  1006. }
  1007. }
  1008. }
  1009. if (dx != 0 || dy != null)
  1010. {
  1011. graph.moveCells(graph.getSelectionCells(), dx, dy);
  1012. App.say('Moved');
  1013. }
  1014. return true;
  1015. }
  1016. }
  1017. else if (tokens[0] == 'text')
  1018. {
  1019. var cells = graph.getSelectionCells();
  1020. if (cells.length == 0 && graph.model.contains(lastInserted))
  1021. {
  1022. cells = [lastInserted];
  1023. }
  1024. if (cells.length == 0)
  1025. {
  1026. App.say('No cell for text');
  1027. }
  1028. else if (tokens[0].length >= 2)
  1029. {
  1030. var value = tokens.slice(1, tokens.length).join(' ');
  1031. if (value.length > 0)
  1032. {
  1033. // Capitalize string
  1034. if (value.length > 1)
  1035. {
  1036. value = value.charAt(0).toUpperCase() + value.slice(1);
  1037. }
  1038. graph.labelChanged(cells[0], value);
  1039. }
  1040. }
  1041. return true;
  1042. }
  1043. else if (tokens[0] == 'disconnect' && tokens.length == 1)
  1044. {
  1045. var cells = graph.getAllEdges(graph.getSelectionCells());
  1046. if (cells.length == 0)
  1047. {
  1048. App.say('No cell to disconnect');
  1049. }
  1050. else
  1051. {
  1052. graph.removeCells(cells);
  1053. }
  1054. return true;
  1055. }
  1056. else if (tokens[0] == 'deselect' && tokens.length == 1)
  1057. {
  1058. graph.clearSelection();
  1059. return true;
  1060. }
  1061. else if (tokens[0] === 'select' && (tokens.length == 1 || mxUtils.indexOf(
  1062. ['vertices', 'edges', 'none', 'all'], tokens[1].toLowerCase()) < 0))
  1063. {
  1064. // Handles some other shortcuts
  1065. if (tokens.length == 1 || mxUtils.indexOf(['last'], tokens[1].toLowerCase()) >= 0)
  1066. {
  1067. if (graph.model.contains(lastInserted))
  1068. {
  1069. graph.setSelectionCell(lastInserted);
  1070. App.say('{1} selected', [graph.getWordForCell(lastInserted).replace(/([A-Z])/g, ' $1')]);
  1071. return true;
  1072. }
  1073. }
  1074. // Handles alternative cases of edges
  1075. else if (mxUtils.indexOf(['connection', 'connections', 'inches'], tokens[1].toLowerCase()) >= 0)
  1076. {
  1077. var edges = graph.getAllEdges(graph.getSelectionCells());
  1078. if (edges.length > 0)
  1079. {
  1080. graph.setSelectionCells(edges);
  1081. }
  1082. else
  1083. {
  1084. ui.actions.get('selectEdges').funct();
  1085. }
  1086. App.sayHint();
  1087. return true;
  1088. }
  1089. // Handles alternative version of vertices
  1090. else if (mxUtils.indexOf(['shapes'], tokens[1].toLowerCase()) >= 0)
  1091. {
  1092. ui.actions.get('selectVertices').funct();
  1093. App.sayHint();
  1094. return true;
  1095. }
  1096. else if (mxUtils.indexOf(['previous'], tokens[1].toLowerCase()) >= 0)
  1097. {
  1098. if (graph.isSelectionEmpty())
  1099. {
  1100. // Selects the first vertex
  1101. var parent = graph.getDefaultParent();
  1102. var childCount = graph.model.getChildCount(parent);
  1103. for (var i = childCount - 1; i >= 0; i--)
  1104. {
  1105. var child = graph.model.getChildAt(parent, i);
  1106. if (graph.model.isVertex(child))
  1107. {
  1108. graph.setSelectionCell(child);
  1109. App.say('{1} selected', [graph.getWordForCell(child).replace(/([A-Z])/g, ' $1')]);
  1110. return true;
  1111. }
  1112. }
  1113. }
  1114. else
  1115. {
  1116. var cell = graph.getSelectionCell();
  1117. var model = graph.getModel();
  1118. var index = model.getParent(cell).getIndex(cell);
  1119. var childCount = model.getChildCount(model.getParent(cell));
  1120. if (index >= 0)
  1121. {
  1122. var next = model.getParent(cell).getChildAt(((index == 0) ? childCount : index) - 1);
  1123. if (next != null)
  1124. {
  1125. graph.setSelectionCell(next);
  1126. App.say('{1} selected', [graph.getWordForCell(next)]);
  1127. return true;
  1128. }
  1129. }
  1130. App.say('Previous not found');
  1131. }
  1132. }
  1133. else if (mxUtils.indexOf(['source', 'target'], tokens[1].toLowerCase()) >= 0)
  1134. {
  1135. var cell = graph.getSelectionCell();
  1136. if (graph.model.isEdge(cell))
  1137. {
  1138. var terminal = graph.model.getTerminal(cell, tokens[1].toLowerCase() == 'source');
  1139. if (terminal != null)
  1140. {
  1141. graph.setSelectionCell(terminal);
  1142. return true;
  1143. }
  1144. }
  1145. }
  1146. else if (mxUtils.indexOf(['next'], tokens[1].toLowerCase()) >= 0)
  1147. {
  1148. if (graph.isSelectionEmpty())
  1149. {
  1150. // Selects the first vertex
  1151. var parent = graph.getDefaultParent();
  1152. var childCount = graph.model.getChildCount(parent);
  1153. for (var i = 0; i < childCount; i++)
  1154. {
  1155. var child = graph.model.getChildAt(parent, i);
  1156. if (graph.model.isVertex(child))
  1157. {
  1158. graph.setSelectionCell(child);
  1159. App.say('{1} selected', [graph.getWordForCell(child).replace(/([A-Z])/g, ' $1')]);
  1160. return true;
  1161. }
  1162. }
  1163. }
  1164. else
  1165. {
  1166. var cell = graph.getSelectionCell();
  1167. var model = graph.getModel();
  1168. var index = model.getParent(cell).getIndex(cell);
  1169. var childCount = model.getChildCount(model.getParent(cell));
  1170. if (index < childCount)
  1171. {
  1172. var next = model.getParent(cell).getChildAt(((index == childCount - 1) ? 0 : index + 1));
  1173. if (next != null)
  1174. {
  1175. graph.setSelectionCell(next);
  1176. App.say('{1} selected', [graph.getWordForCell(next).replace(/([A-Z])/g, ' $1')]);
  1177. return true;
  1178. }
  1179. }
  1180. }
  1181. }
  1182. else
  1183. {
  1184. var states = graph.view.states.getValues();
  1185. var token = tokens[1].toLowerCase();
  1186. var searchToken = tokens.slice(1, tokens.length).join('').toLowerCase();
  1187. for (var i = states.length - 1; i > 0; i--)
  1188. {
  1189. // Simple matching for shape name or label
  1190. if (!graph.isCellSelected(states[i].cell))
  1191. {
  1192. // Tries label search first
  1193. var lab = graph.getLabel(states[i].cell);
  1194. if (lab != null && lab.length > 0)
  1195. {
  1196. lab = lab.toLowerCase().replace(/ /g, '');
  1197. // Some common mistakes
  1198. if (lab.charAt(lab.length - 1) == '2' && tokens[tokens.length - 1].toLowerCase() == 'to')
  1199. {
  1200. lab = lab.substring(0, lab.length - 1) + 'to';
  1201. }
  1202. var min = Math.min(searchToken.length, lab.length);
  1203. if (searchToken.substring(0, min) == lab.substring(0, min))
  1204. {
  1205. graph.setSelectionCell(states[i].cell);
  1206. App.say('{1} selected', [graph.getWordForCell(states[i].cell)]);
  1207. return true;
  1208. }
  1209. }
  1210. // Then tries shapename search
  1211. if (tokens.length == 2)
  1212. {
  1213. // Some common names
  1214. if (mxUtils.indexOf(['circle'], token) >= 0)
  1215. {
  1216. searchToken = 'ellipse';
  1217. }
  1218. else if (mxUtils.indexOf(['condition', 'decision'], token) >= 0)
  1219. {
  1220. searchToken = 'rhombus';
  1221. }
  1222. else if (mxUtils.indexOf(['preparation'], token) >= 0)
  1223. {
  1224. searchToken = 'hexagon';
  1225. }
  1226. else if (mxUtils.indexOf(['trapez'], token) >= 0)
  1227. {
  1228. searchToken = 'parallelogram';
  1229. }
  1230. }
  1231. var wrd = graph.getWordForCell(states[i].cell, true);
  1232. var min = Math.min(wrd.length, searchToken.length);
  1233. if (searchToken.substring(0, min) == wrd.substring(0, min))
  1234. {
  1235. graph.setSelectionCell(states[i].cell);
  1236. App.say('{1} selected', [graph.getWordForCell(states[i].cell)]);
  1237. return true;
  1238. }
  1239. }
  1240. }
  1241. }
  1242. }
  1243. else if (tokens[0] === 'remove')
  1244. {
  1245. var cells = graph.getSelectionCells();
  1246. if (cells.length == 0 && graph.model.contains(lastInserted))
  1247. {
  1248. cells = [lastInserted];
  1249. }
  1250. if (cells.length == 0)
  1251. {
  1252. App.say('No cells');
  1253. }
  1254. else if (tokens[1].toLowerCase() == 'text')
  1255. {
  1256. graph.labelChanged(cells[0], '');
  1257. }
  1258. else
  1259. {
  1260. // Remove style
  1261. var styleToken = mxUtils.trim(tokens.slice(1, tokens.length).join(' ').toLowerCase());
  1262. // Fixes common recognition errors
  1263. styleToken = styleToken.replace(/a line/g, 'align')
  1264. // Merges to single word
  1265. styleToken = styleToken.replace(/ /g, '')
  1266. // Fixes common speech errors that make no sense in the context
  1267. var keys = [mxConstants.STYLE_FILLCOLOR, mxConstants.STYLE_GRADIENTCOLOR,
  1268. mxConstants.STYLE_STROKECOLOR, mxConstants.STYLE_FONTCOLOR,
  1269. mxConstants.STYLE_LABEL_BACKGROUNDCOLOR, mxConstants.STYLE_LABEL_BORDERCOLOR,
  1270. mxConstants.STYLE_STROKEWIDTH, mxConstants.STYLE_FONTSIZE, mxConstants.STYLE_SPACING,
  1271. mxConstants.STYLE_SPACING_TOP, mxConstants.STYLE_SPACING_LEFT, mxConstants.STYLE_SPACING_RIGHT,
  1272. mxConstants.STYLE_SPACING_BOTTOM, mxConstants.STYLE_PERIMETER_SPACING,
  1273. mxConstants.STYLE_STARTSIZE, mxConstants.STYLE_ENDSIZE, mxConstants.STYLE_OPACITY,
  1274. mxConstants.STYLE_TEXT_OPACITY, mxConstants.STYLE_ROTATION,
  1275. mxConstants.STYLE_ALIGN, mxConstants.STYLE_VERTICAL_ALIGN, mxConstants.STYLE_FONTFAMILY,
  1276. mxConstants.STYLE_LABEL_POSITION, mxConstants.STYLE_VERTICAL_LABEL_POSITION,
  1277. mxConstants.STYLE_GRADIENT_DIRECTION];
  1278. // Guesses best word
  1279. var style = getBestWord(styleToken, keys);
  1280. if (style != null)
  1281. {
  1282. graph.setCellStyles(style, null, cells);
  1283. ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [null], 'cells', cells));
  1284. }
  1285. else
  1286. {
  1287. App.say('Unknown style {1}', [changeTokens[0]]);
  1288. }
  1289. }
  1290. return true;
  1291. }
  1292. else
  1293. {
  1294. var cells = graph.getSelectionCells();
  1295. if (cells.length == 0 && graph.model.contains(lastInserted))
  1296. {
  1297. cells = [lastInserted];
  1298. }
  1299. // Try numeric stylename if last token is numeric
  1300. var lastToken = tokens[tokens.length - 1].toLowerCase();
  1301. // Fixes some special cases for recoginition of short phrases
  1302. // like "stroke width 2" where it tries to be clever
  1303. if (lastToken == 'tool' || lastToken == 'tune' || lastToken == 'tomb' || lastToken == 'tube')
  1304. {
  1305. lastToken = '2';
  1306. }
  1307. else if (lastToken == 'd3')
  1308. {
  1309. lastToken = '3';
  1310. }
  1311. else if (lastToken == 'v')
  1312. {
  1313. lastToken = '5';
  1314. }
  1315. else
  1316. {
  1317. // Converts numeric words to numbers (system only seems to return written numbers for <= 4)
  1318. var numbers = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine'];
  1319. var tmpIndex = mxUtils.indexOf(numbers, lastToken);
  1320. if (tmpIndex >= 0)
  1321. {
  1322. lastToken = tmpIndex;
  1323. }
  1324. }
  1325. // If last token is numeric
  1326. var lastValue = parseFloat(lastToken);
  1327. if (tokens.length >= 2 && !isNaN(lastValue))
  1328. {
  1329. var styleToken = tokens.slice(0, tokens.length - 1).join('').toLowerCase();
  1330. // Numeric styles
  1331. var keys = [mxConstants.STYLE_STROKEWIDTH, mxConstants.STYLE_FONTSIZE, mxConstants.STYLE_SPACING,
  1332. mxConstants.STYLE_SPACING_TOP, mxConstants.STYLE_SPACING_LEFT, mxConstants.STYLE_SPACING_RIGHT,
  1333. mxConstants.STYLE_SPACING_BOTTOM, mxConstants.STYLE_PERIMETER_SPACING,
  1334. mxConstants.STYLE_STARTSIZE, mxConstants.STYLE_ENDSIZE, mxConstants.STYLE_OPACITY,
  1335. mxConstants.STYLE_TEXT_OPACITY, mxConstants.STYLE_ROTATION]
  1336. var style = null;
  1337. // Workaround for problem with "font size" is to say "size" ("font" is near "phone")
  1338. if (styleToken == 'size')
  1339. {
  1340. style = mxConstants.STYLE_FONTSIZE;
  1341. }
  1342. else
  1343. {
  1344. // Guesses best word
  1345. style = getBestWord(styleToken, keys);
  1346. }
  1347. if (style != null)
  1348. {
  1349. if (cells.length == 0)
  1350. {
  1351. App.say('No cell for {1}', [style]);
  1352. }
  1353. // Plausibility check
  1354. else if (lastValue <= 400)
  1355. {
  1356. graph.setCellStyles(style, lastValue, cells);
  1357. ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [lastValue], 'cells', cells));
  1358. }
  1359. else
  1360. {
  1361. App.say('{1} ignored', [lastValue]);
  1362. }
  1363. // Terminate
  1364. return true;
  1365. }
  1366. }
  1367. // If "color" appears in the command
  1368. if (tokens.length >= 2)
  1369. {
  1370. // Replaces GB spelling with US spelling for internal lookups
  1371. var colorCommand = mxUtils.trim(command.toLowerCase().replace(/colour/g, 'color'));
  1372. var colorIndex = colorCommand.indexOf('color');
  1373. // "Color" (word alone) matches to fill color
  1374. if (colorIndex >= 0)
  1375. {
  1376. // Text from first space after color* word
  1377. var colorToken = mxUtils.trim(colorCommand.substring(
  1378. colorCommand.indexOf(' ', colorIndex))).replace(/ /g, '');
  1379. var color = null;
  1380. // Checks for color code of token using ntc
  1381. for (var i = 0; i < ntc.names.length; i++)
  1382. {
  1383. if (ntc.names[i][1].toLowerCase().replace(/ /g, '') == colorToken)
  1384. {
  1385. color = '#' + ntc.names[i][0];
  1386. break;
  1387. }
  1388. }
  1389. // Color is known
  1390. if (color != null)
  1391. {
  1392. var styleToken = colorCommand.substring(0, colorIndex + 'color'.length).replace(/ /g, '');
  1393. keys = [mxConstants.STYLE_GRADIENTCOLOR,
  1394. mxConstants.STYLE_STROKECOLOR, mxConstants.STYLE_FONTCOLOR,
  1395. mxConstants.STYLE_LABEL_BACKGROUNDCOLOR, mxConstants.STYLE_LABEL_BORDERCOLOR,
  1396. mxConstants.STYLE_FILLCOLOR];
  1397. var style = null;
  1398. // Handles some special cases
  1399. if (styleToken == 'thecolor' || styleToken == 'color')
  1400. {
  1401. if (graph.model.isEdge(graph.getSelectionCell()))
  1402. {
  1403. style = mxConstants.STYLE_STROKECOLOR;
  1404. }
  1405. else
  1406. {
  1407. style = mxConstants.STYLE_FILLCOLOR;
  1408. }
  1409. }
  1410. else
  1411. {
  1412. // Guesses best word
  1413. style = getBestWord(styleToken, keys);
  1414. }
  1415. if (style != null)
  1416. {
  1417. if (cells.length == 0)
  1418. {
  1419. App.say('No cell for {1}', [style]);
  1420. }
  1421. else
  1422. {
  1423. graph.setCellStyles(style, color, cells);
  1424. ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [color], 'cells', cells));
  1425. }
  1426. // Terminate
  1427. return true;
  1428. }
  1429. }
  1430. }
  1431. }
  1432. // If the first token is a general stylename
  1433. if (tokens.length >= 2)
  1434. {
  1435. var keys = [mxConstants.STYLE_ALIGN, mxConstants.STYLE_VERTICAL_ALIGN,
  1436. mxConstants.STYLE_FONTFAMILY, mxConstants.STYLE_LABEL_POSITION,
  1437. mxConstants.STYLE_VERTICAL_LABEL_POSITION,
  1438. mxConstants.STYLE_GRADIENT_DIRECTION];
  1439. var styleToken = mxUtils.trim(tokens.slice(0, tokens.length - 1).join(' ').toLowerCase());
  1440. // Fixes common recognition errors
  1441. styleToken = styleToken.replace(/a line/g, 'align')
  1442. // Merges to single word
  1443. styleToken = styleToken.replace(/ /g, '')
  1444. var style = resolveStylename(styleToken, keys);
  1445. var value = mxUtils.trim(tokens[tokens.length - 1].toLowerCase());
  1446. if (style != null && value != null && value.length > 0)
  1447. {
  1448. if (cells.length == 0)
  1449. {
  1450. App.say('No cell for {1}', [style]);
  1451. }
  1452. else
  1453. {
  1454. graph.setCellStyles(style, value, cells);
  1455. ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [value], 'cells', cells));
  1456. }
  1457. // Terminate
  1458. return true;
  1459. }
  1460. }
  1461. // Checks if the command is a shape name
  1462. var vertices = [];
  1463. for (var i = 0; i < cells.length; i++)
  1464. {
  1465. var cell = cells[i];
  1466. if (graph.model.isVertex(cell))
  1467. {
  1468. vertices.push(cell);
  1469. }
  1470. }
  1471. var shapenameToken = mxUtils.trim(tokens.join('')).toLowerCase();
  1472. // Searches for registered shape names
  1473. // LATER: Using search like insert requires access to
  1474. // the cells of the search result items
  1475. if (shapeList == null)
  1476. {
  1477. shapeList = [];
  1478. for (var tmp in mxCellRenderer.prototype.defaultShapes)
  1479. {
  1480. shapeList.push(tmp.toLowerCase());
  1481. }
  1482. }
  1483. // Only exact matches allowed here
  1484. // LATER: Support rounded style
  1485. var shape = null;
  1486. // Some common mappings
  1487. if (mxUtils.indexOf(['cirlce', 'event', 'start', 'end'], shapenameToken) >= 0)
  1488. {
  1489. shapenameToken = shape = 'ellipse';
  1490. }
  1491. else if (mxUtils.indexOf(['condition', 'decision'], shapenameToken) >= 0)
  1492. {
  1493. shapenameToken = shape = 'rhombus';
  1494. }
  1495. else if (mxUtils.indexOf(['preparation'], shapenameToken) >= 0)
  1496. {
  1497. shapenameToken = shape = 'hexagon';
  1498. }
  1499. else if (mxUtils.indexOf(['trapez'], shapenameToken) >= 0)
  1500. {
  1501. shapenameToken = shape = 'parallelogram';
  1502. }
  1503. else
  1504. {
  1505. shape = getBestWord(shapenameToken, shapeList);
  1506. }
  1507. // LATER: Ignore all edge shapes
  1508. if (shape != null && shape.toLowerCase() == shapenameToken && shape != 'connector' &&
  1509. shape != 'arrow' && shape != 'flexarrow' && shape != 'arrowconnector' &&
  1510. shape != 'link')
  1511. {
  1512. if (cells.length == 0)
  1513. {
  1514. App.say('No cell for {1}', [shape]);
  1515. }
  1516. else if (vertices.length == 0)
  1517. {
  1518. App.say('Connections ignored');
  1519. }
  1520. else
  1521. {
  1522. // LATER: Add perimeter style etc
  1523. graph.setCellStyles(mxConstants.STYLE_SHAPE, shapenameToken, vertices);
  1524. ui.fireEvent(new mxEventObject('styleChanged', 'keys', [mxConstants.STYLE_SHAPE],
  1525. 'values', [shapenameToken], 'cells', vertices));
  1526. // Terminate
  1527. return true;
  1528. }
  1529. }
  1530. // Lazy creation of cache for action names
  1531. if (actions == null)
  1532. {
  1533. actions = {};
  1534. actionList = [];
  1535. for (var name in ui.actions.actions)
  1536. {
  1537. var shortname = mxUtils.trim(name).toLowerCase().replace(/ /g, '');
  1538. actions[shortname] = ui.actions.actions[name];
  1539. actionList.push(shortname);
  1540. }
  1541. }
  1542. var name = mxUtils.trim(command).toLowerCase().replace(/ /g, '');
  1543. if (urlParams['dev'] == '1')
  1544. {
  1545. var guess = getBestWord(name, actionList, true);
  1546. console.log('App.say guess:', name, '=>', guess, '(' +
  1547. naiveHammingDistance(name, guess) + ', ' +
  1548. levenshteinDist(name, guess) +
  1549. ')');
  1550. }
  1551. // Some common redirects and mistakes
  1552. if (tokens[0] == 'back')
  1553. {
  1554. command = name = 'undo';
  1555. }
  1556. else if (tokens[0] == 'restore')
  1557. {
  1558. command = name = 'redo';
  1559. }
  1560. else if (tokens[0] == 'clone')
  1561. {
  1562. command = name = 'duplicate';
  1563. }
  1564. else if (mxUtils.indexOf(['all', 'small', 'wall', 'call'], name) >= 0)
  1565. {
  1566. command = name = 'selectall';
  1567. }
  1568. else if (mxUtils.indexOf(['both', 'boat', 'phone', 'don\'t', 'food', 'bolt', 'bull'], tokens[0]) >= 0)
  1569. {
  1570. command = name = 'bold';
  1571. }
  1572. var action = actions[name];
  1573. if (action != null)
  1574. {
  1575. if (action.isEnabled())
  1576. {
  1577. App.say(name);
  1578. action.funct();
  1579. }
  1580. else
  1581. {
  1582. // Cannot use name here as it has no spaces
  1583. App.say('{1} disabled', [command]);
  1584. }
  1585. return true;
  1586. }
  1587. return false;
  1588. }
  1589. }
  1590. };
  1591. if ('speechSynthesis' in window)
  1592. {
  1593. // Shows dialog for mic access
  1594. if (navigator.webkitGetUserMedia)
  1595. {
  1596. window.addEventListener('load', function()
  1597. {
  1598. navigator.webkitGetUserMedia({audio:true}, function()
  1599. {
  1600. //console.log('access to mic ok');
  1601. }, function(e)
  1602. {
  1603. //alert('Error getting audio');
  1604. console.log(e);
  1605. });
  1606. });
  1607. }
  1608. // Installs helper methods and listeners for speech output
  1609. var graph = ui.editor.graph;
  1610. App.reloadVoicePlugin = function()
  1611. {
  1612. // Shows UI
  1613. speechOutputEnabled = true;
  1614. menu.style.display = '';
  1615. td.style.display = 'inline-block';
  1616. };
  1617. App.sayHint = function()
  1618. {
  1619. if (graph.getSelectionCount() == 0)
  1620. {
  1621. App.say('No cell selected.' + ((ui.isDiagramEmpty()) ? ' Diagram is empty.' : ''));
  1622. }
  1623. else if (graph.getSelectionCount() == 1)
  1624. {
  1625. var cell = graph.getSelectionCell();
  1626. if (graph.getModel().isVertex(cell))
  1627. {
  1628. var geo = graph.getCellGeometry(cell)
  1629. if (geo != null)
  1630. {
  1631. App.say('{1} at {2} and {3} is {4} times {5} pixels', [graph.getWordForCell(cell, true),
  1632. geo.x, geo.y, geo.width, geo.height]);
  1633. }
  1634. }
  1635. else
  1636. {
  1637. App.say('One connection selected');
  1638. }
  1639. var lab = mxUtils.trim(graph.getLabel(cell));
  1640. if (lab != null && lab.length > 0 && lab.length < 20)
  1641. {
  1642. App.say('Label is {1}', [lab]);
  1643. }
  1644. }
  1645. else
  1646. {
  1647. var cells = graph.getSelectionCells();
  1648. var vertexCount = 0;
  1649. var edgeCount = 0;
  1650. for (var i = 0; i < cells.length; i++)
  1651. {
  1652. var cell = cells[i];
  1653. if (graph.model.isEdge(cell))
  1654. {
  1655. edgeCount++;
  1656. }
  1657. else if (graph.model.isVertex(cell))
  1658. {
  1659. vertexCount++;
  1660. }
  1661. }
  1662. if (vertexCount > 0 && edgeCount > 0)
  1663. {
  1664. var tmp = '';
  1665. if (vertexCount == 1)
  1666. {
  1667. tmp += 'One shape';
  1668. }
  1669. else
  1670. {
  1671. tmp += '{1} shapes';
  1672. }
  1673. tmp += ' and ';
  1674. if (edgeCount == 1)
  1675. {
  1676. tmp += 'one connection';
  1677. }
  1678. else
  1679. {
  1680. tmp += '{2} connections';
  1681. }
  1682. App.say(tmp + ' selected', [vertexCount, edgeCount]);
  1683. }
  1684. else if (vertexCount > 0)
  1685. {
  1686. App.say('{1} shapes selected', [vertexCount]);
  1687. }
  1688. else
  1689. {
  1690. App.say('{1} connections selected', [edgeCount]);
  1691. }
  1692. }
  1693. };
  1694. // Can return more than first word
  1695. function firstWord(value)
  1696. {
  1697. // TODO Use regex
  1698. if (value != null && value.length > maxLabelLength)
  1699. {
  1700. var space = value.indexOf(' ');
  1701. // Add second word if label is short
  1702. var tmp = value.indexOf(' ', space + 1);
  1703. if (tmp >= 0 && tmp <= maxLabelLength)
  1704. {
  1705. space = tmp;
  1706. // Add third word if label is short
  1707. var tmp = value.indexOf(' ', space + 1);
  1708. if (tmp >= 0 && tmp <= maxLabelLength)
  1709. {
  1710. space = tmp;
  1711. }
  1712. }
  1713. if (space >= 0)
  1714. {
  1715. value = value.substring(0, space);
  1716. }
  1717. }
  1718. return value;
  1719. };
  1720. graph.getWordForCell = function(cell, ignoreLabel)
  1721. {
  1722. var label = 'no';
  1723. if (cell != null)
  1724. {
  1725. label = (ignoreLabel) ? null : firstWord(this.getLabel(cell));
  1726. if (label == null || label.length == 0 || label.length > maxLabelLength || mxUtils.isNumeric(label))
  1727. {
  1728. var state = this.view.getState(cell);
  1729. var style = (state != null) ? state.style : this.getCellStyle(cell);
  1730. var tmp = style[mxConstants.STYLE_SHAPE];
  1731. if (tmp == 'label')
  1732. {
  1733. label = 'rectangle';
  1734. }
  1735. else if (tmp != null)
  1736. {
  1737. var dot = tmp.lastIndexOf('.');
  1738. if (dot >= 0)
  1739. {
  1740. tmp = tmp.substring(dot + 1);
  1741. }
  1742. label = tmp
  1743. }
  1744. }
  1745. var div = document.createElement('div');
  1746. div.innerHTML = label;
  1747. label = div.innerText;
  1748. }
  1749. return label;
  1750. };
  1751. ui.addListener('styleChanged', function(sender, evt)
  1752. {
  1753. //console.log('styleChanged', evt, evt.getProperty('keys'));
  1754. var cells = evt.getProperty('cells');
  1755. var keys = evt.getProperty('keys');
  1756. var values = evt.getProperty('values');
  1757. if (cells != null && keys != null && keys.length == 1 && values.length == 1)
  1758. {
  1759. var tmp = values[0];
  1760. if (typeof tmp === 'string' && tmp.charAt(0) == '#')
  1761. {
  1762. var n_match = ntc.name(tmp);
  1763. //if (n_match[2]) /* exact match */
  1764. {
  1765. tmp = n_match[1];
  1766. }
  1767. }
  1768. if (tmp == mxConstants.NONE || tmp == null)
  1769. {
  1770. App.say('Removed {1}', [keys[0]]);
  1771. }
  1772. else
  1773. {
  1774. // Replaces camel case notation with spaces
  1775. var key = keys[0].replace(/([A-Z])/g, ' $1')
  1776. App.say('{1} {2}', [key, tmp]);
  1777. }
  1778. }
  1779. });
  1780. graph.addListener(mxEvent.LABEL_CHANGED, function(sender, evt)
  1781. {
  1782. //console.log('CELL_CONNECTED', evt);
  1783. var cell = evt.getProperty('cell');
  1784. var old = evt.getProperty('old');
  1785. // Resolves placeholders in new label
  1786. var value = this.getLabel(cell);
  1787. if ((value != null && value.length > 20) ||
  1788. (old != null && old.length > 20))
  1789. {
  1790. App.say('Label changed');
  1791. }
  1792. else if (value != null && value.length == 0)
  1793. {
  1794. App.say('Label removed');
  1795. }
  1796. else if (value != null)
  1797. {
  1798. App.say('{1}', [value]);
  1799. }
  1800. });
  1801. graph.addListener(mxEvent.CELL_CONNECTED, function(sender, evt)
  1802. {
  1803. // console.log('CELL_CONNECTED', evt);
  1804. // var edge = evt.getProperty('edge');
  1805. // var terminal = evt.getProperty('terminal');
  1806. // var other = graph.model.getTerminal(edge, !evt.getProperty('source'));
  1807. //
  1808. // if (other != null && terminal != null)
  1809. // {
  1810. // App.say('connected {1} to {2}', [getWordForCell(other), getWordForCell(terminal)]);
  1811. // }
  1812. });
  1813. graph.addListener(mxEvent.CELLS_ADDED, function(sender, evt)
  1814. {
  1815. //console.log('CELLS_ADDED', evt);
  1816. var cells = evt.getProperty('cells');
  1817. if (cells.length > 1)
  1818. {
  1819. var vertexCount = 0;
  1820. var edgeCount = 0;
  1821. for (var i = 0; i < cells.length; i++)
  1822. {
  1823. var cell = cells[i];
  1824. if (graph.model.isEdge(cell))
  1825. {
  1826. edgeCount++;
  1827. }
  1828. else if (graph.model.isVertex(cell))
  1829. {
  1830. vertexCount++;
  1831. }
  1832. }
  1833. if (vertexCount > 0 && edgeCount > 0)
  1834. {
  1835. var tmp = '';
  1836. if (vertexCount == 1)
  1837. {
  1838. tmp += 'One shape';
  1839. }
  1840. else
  1841. {
  1842. tmp += '{1} shapes';
  1843. }
  1844. tmp += ' and ';
  1845. if (edgeCount == 1)
  1846. {
  1847. tmp += 'one connection';
  1848. }
  1849. else
  1850. {
  1851. tmp += '{2} connections';
  1852. }
  1853. App.say(tmp + ' inserted', [vertexCount, edgeCount]);
  1854. }
  1855. else if (vertexCount > 0)
  1856. {
  1857. App.say('{1} shapes', [vertexCount]);
  1858. }
  1859. else
  1860. {
  1861. App.say('{1} connections', [edgeCount]);
  1862. }
  1863. }
  1864. else
  1865. {
  1866. cell = cells[0];
  1867. if (graph.model.isEdge(cell) && graph.model.getTerminal(cell, true) != null &&
  1868. graph.model.getTerminal(cell, false) != null)
  1869. {
  1870. if (graph.getWordForCell(graph.model.getTerminal(cell, true)) ==
  1871. graph.getWordForCell(graph.model.getTerminal(cell, false)))
  1872. {
  1873. App.say('Connected');
  1874. }
  1875. else
  1876. {
  1877. App.say('{1} connected to {2}', [graph.getWordForCell(graph.model.getTerminal(cell, true)),
  1878. graph.getWordForCell(graph.model.getTerminal(cell, false))]);
  1879. }
  1880. }
  1881. else if (graph.model.isVertex(cell))
  1882. {
  1883. lastInserted = cell;
  1884. // Replaces camel case notation with spaces
  1885. var word = graph.getWordForCell(cell).replace(/([A-Z])/g, ' $1');
  1886. App.say('{1}', [word]);
  1887. }
  1888. }
  1889. });
  1890. graph.addListener(mxEvent.CELLS_REMOVED, function(sender, evt)
  1891. {
  1892. //console.log('CELLS_REMOVED', evt);
  1893. var cells = evt.getProperty('cells');
  1894. if (cells.length > 1)
  1895. {
  1896. App.say('{1} cells deleted', [cells.length]);
  1897. }
  1898. else
  1899. {
  1900. cell = cells[0];
  1901. App.say('{1} deleted', [graph.getWordForCell(cell)]);
  1902. }
  1903. });
  1904. }
  1905. });
  1906. /** Code for color to voice based on ntc (name that color)
  1907. /*
  1908. +-----------------------------------------------------------------+
  1909. | Created by Chirag Mehta - http://chir.ag/projects/ntc |
  1910. |-----------------------------------------------------------------|
  1911. | ntc js (Name that Color JavaScript) |
  1912. +-----------------------------------------------------------------+
  1913. All the functions, code, lists etc. have been written specifically
  1914. for the Name that Color JavaScript by Chirag Mehta unless otherwise
  1915. specified.
  1916. This script is released under the: Creative Commons License:
  1917. Attribution 2.5 http://creativecommons.org/licenses/by/2.5/
  1918. Sample Usage:
  1919. <script type="text/javascript" src="ntc.js"></script>
  1920. <script type="text/javascript">
  1921. var n_match = ntc.name("#6195ED");
  1922. n_rgb = n_match[0]; // This is the RGB value of the closest matching color
  1923. n_name = n_match[1]; // This is the text string for the name of the match
  1924. n_exactmatch = n_match[2]; // True if exact color match, False if close-match
  1925. alert(n_match);
  1926. </script>
  1927. */
  1928. var ntc = {
  1929. init: function() {
  1930. var color, rgb, hsl;
  1931. for(var i = 0; i < ntc.names.length; i++)
  1932. {
  1933. color = "#" + ntc.names[i][0];
  1934. rgb = ntc.rgb(color);
  1935. hsl = ntc.hsl(color);
  1936. ntc.names[i].push(rgb[0], rgb[1], rgb[2], hsl[0], hsl[1], hsl[2]);
  1937. }
  1938. },
  1939. name: function(color) {
  1940. color = color.toUpperCase();
  1941. if(color.length < 3 || color.length > 7)
  1942. return ["#000000", "Invalid Color: " + color, false];
  1943. if(color.length % 3 == 0)
  1944. color = "#" + color;
  1945. if(color.length == 4)
  1946. color = "#" + color.substr(1, 1) + color.substr(1, 1) + color.substr(2, 1) + color.substr(2, 1) + color.substr(3, 1) + color.substr(3, 1);
  1947. var rgb = ntc.rgb(color);
  1948. var r = rgb[0], g = rgb[1], b = rgb[2];
  1949. var hsl = ntc.hsl(color);
  1950. var h = hsl[0], s = hsl[1], l = hsl[2];
  1951. var ndf1 = 0; ndf2 = 0; ndf = 0;
  1952. var cl = -1, df = -1;
  1953. for(var i = 0; i < ntc.names.length; i++)
  1954. {
  1955. if(color == "#" + ntc.names[i][0])
  1956. return ["#" + ntc.names[i][0], ntc.names[i][1], true];
  1957. ndf1 = Math.pow(r - ntc.names[i][2], 2) + Math.pow(g - ntc.names[i][3], 2) + Math.pow(b - ntc.names[i][4], 2);
  1958. ndf2 = Math.pow(h - ntc.names[i][5], 2) + Math.pow(s - ntc.names[i][6], 2) + Math.pow(l - ntc.names[i][7], 2);
  1959. ndf = ndf1 + ndf2 * 2;
  1960. if(df < 0 || df > ndf)
  1961. {
  1962. df = ndf;
  1963. cl = i;
  1964. }
  1965. }
  1966. return (cl < 0 ? ["#000000", "Invalid Color: " + color, false] : ["#" + ntc.names[cl][0], ntc.names[cl][1], false]);
  1967. },
  1968. // adopted from: Farbtastic 1.2
  1969. // http://acko.net/dev/farbtastic
  1970. hsl: function (color) {
  1971. var rgb = [parseInt('0x' + color.substring(1, 3)) / 255, parseInt('0x' + color.substring(3, 5)) / 255, parseInt('0x' + color.substring(5, 7)) / 255];
  1972. var min, max, delta, h, s, l;
  1973. var r = rgb[0], g = rgb[1], b = rgb[2];
  1974. min = Math.min(r, Math.min(g, b));
  1975. max = Math.max(r, Math.max(g, b));
  1976. delta = max - min;
  1977. l = (min + max) / 2;
  1978. s = 0;
  1979. if(l > 0 && l < 1)
  1980. s = delta / (l < 0.5 ? (2 * l) : (2 - 2 * l));
  1981. h = 0;
  1982. if(delta > 0)
  1983. {
  1984. if (max == r && max != g) h += (g - b) / delta;
  1985. if (max == g && max != b) h += (2 + (b - r) / delta);
  1986. if (max == b && max != r) h += (4 + (r - g) / delta);
  1987. h /= 6;
  1988. }
  1989. return [parseInt(h * 255), parseInt(s * 255), parseInt(l * 255)];
  1990. },
  1991. // adopted from: Farbtastic 1.2
  1992. // http://acko.net/dev/farbtastic
  1993. rgb: function(color) {
  1994. return [parseInt('0x' + color.substring(1, 3)), parseInt('0x' + color.substring(3, 5)), parseInt('0x' + color.substring(5, 7))];
  1995. },
  1996. names: [
  1997. // Added all standard HTML color names / gaudenz, oct-2015
  1998. ["F0F8FF", "aliceblue"],
  1999. ["FAEBD7", "antiquewhite"],
  2000. ["00FFFF", "aqua"],
  2001. ["7FFFD4", "aquamarine"],
  2002. ["F0FFFF", "azure"],
  2003. ["F5F5DC", "beige"],
  2004. ["FFE4C4", "bisque"],
  2005. ["000000", "black"],
  2006. ["FFEBCD", "blanchedalmond"],
  2007. ["0000FF", "blue"],
  2008. ["8A2BE2", "blueviolet"],
  2009. ["A52A2A", "brown"],
  2010. ["DEB887", "burlywood"],
  2011. ["5F9EA0", "cadetblue"],
  2012. ["7FFF00", "chartreuse"],
  2013. ["D2691E", "chocolate"],
  2014. ["FF7F50", "coral"],
  2015. ["6495ED", "cornflowerblue"],
  2016. ["FFF8DC", "cornsilk"],
  2017. ["DC143C", "crimson"],
  2018. ["00FFFF", "cyan"],
  2019. ["00008B", "darkblue"],
  2020. ["008B8B", "darkcyan"],
  2021. ["B8860B", "darkgoldenrod"],
  2022. ["A9A9A9", "darkgray"],
  2023. ["A9A9A9", "darkgrey"],
  2024. ["006400", "darkgreen"],
  2025. ["BDB76B", "darkkhaki"],
  2026. ["8B008B", "darkmagenta"],
  2027. ["556B2F", "darkolivegreen"],
  2028. ["FF8C00", "darkorange"],
  2029. ["9932CC", "darkorchid"],
  2030. ["8B0000", "darkred"],
  2031. ["E9967A", "darksalmon"],
  2032. ["8FBC8F", "darkseagreen"],
  2033. ["483D8B", "darkslateblue"],
  2034. ["2F4F4F", "darkslategray"],
  2035. ["2F4F4F", "darkslategrey"],
  2036. ["00CED1", "darkturquoise"],
  2037. ["9400D3", "darkviolet"],
  2038. ["FF1493", "deeppink"],
  2039. ["00BFFF", "deepskyblue"],
  2040. ["696969", "dimgray"],
  2041. ["696969", "dimgrey"],
  2042. ["1E90FF", "dodgerblue"],
  2043. ["B22222", "firebrick"],
  2044. ["FFFAF0", "floralwhite"],
  2045. ["228B22", "forestgreen"],
  2046. ["FF00FF", "fuchsia"],
  2047. ["DCDCDC", "gainsboro"],
  2048. ["F8F8FF", "ghostwhite"],
  2049. ["FFD700", "gold"],
  2050. ["DAA520", "goldenrod"],
  2051. ["808080", "gray"],
  2052. ["808080", "grey"],
  2053. ["008000", "green"],
  2054. ["ADFF2F", "greenyellow"],
  2055. ["F0FFF0", "honeydew"],
  2056. ["FF69B4", "hotpink"],
  2057. ["CD5C5C", "indianred "],
  2058. ["4B0082", "indigo "],
  2059. ["FFFFF0", "ivory"],
  2060. ["F0E68C", "khaki"],
  2061. ["E6E6FA", "lavender"],
  2062. ["FFF0F5", "lavenderblush"],
  2063. ["7CFC00", "lawngreen"],
  2064. ["FFFACD", "lemonchiffon"],
  2065. ["ADD8E6", "lightblue"],
  2066. ["F08080", "lightcoral"],
  2067. ["E0FFFF", "lightcyan"],
  2068. ["FAFAD2", "lightgoldenrodyellow"],
  2069. ["D3D3D3", "lightgray"],
  2070. ["D3D3D3", "lightgrey"],
  2071. ["90EE90", "lightgreen"],
  2072. ["FFB6C1", "lightpink"],
  2073. ["FFA07A", "lightsalmon"],
  2074. ["20B2AA", "lightseagreen"],
  2075. ["87CEFA", "lightskyblue"],
  2076. ["778899", "lightslategray"],
  2077. ["778899", "lightslategrey"],
  2078. ["B0C4DE", "lightsteelblue"],
  2079. ["FFFFE0", "lightyellow"],
  2080. ["00FF00", "lime"],
  2081. ["32CD32", "limegreen"],
  2082. ["FAF0E6", "linen"],
  2083. ["FF00FF", "magenta"],
  2084. ["800000", "maroon"],
  2085. ["66CDAA", "mediumaquamarine"],
  2086. ["0000CD", "mediumblue"],
  2087. ["BA55D3", "mediumorchid"],
  2088. ["9370DB", "mediumpurple"],
  2089. ["3CB371", "mediumseagreen"],
  2090. ["7B68EE", "mediumslateblue"],
  2091. ["00FA9A", "mediumspringgreen"],
  2092. ["48D1CC", "mediumturquoise"],
  2093. ["C71585", "mediumvioletred"],
  2094. ["191970", "midnightblue"],
  2095. ["F5FFFA", "mintcream"],
  2096. ["FFE4E1", "mistyrose"],
  2097. ["FFE4B5", "moccasin"],
  2098. ["FFDEAD", "navajowhite"],
  2099. ["000080", "navy"],
  2100. ["FDF5E6", "oldlace"],
  2101. ["808000", "olive"],
  2102. ["6B8E23", "olivedrab"],
  2103. ["FFA500", "orange"],
  2104. ["FF4500", "orangered"],
  2105. ["DA70D6", "orchid"],
  2106. ["EEE8AA", "palegoldenrod"],
  2107. ["98FB98", "palegreen"],
  2108. ["AFEEEE", "paleturquoise"],
  2109. ["DB7093", "palevioletred"],
  2110. ["FFEFD5", "papayawhip"],
  2111. ["FFDAB9", "peachpuff"],
  2112. ["CD853F", "peru"],
  2113. ["FFC0CB", "pink"],
  2114. ["DDA0DD", "plum"],
  2115. ["B0E0E6", "powderblue"],
  2116. ["800080", "purple"],
  2117. ["FF0000", "red"],
  2118. ["BC8F8F", "rosybrown"],
  2119. ["4169E1", "royalblue"],
  2120. ["8B4513", "saddlebrown"],
  2121. ["FA8072", "salmon"],
  2122. ["F4A460", "sandybrown"],
  2123. ["2E8B57", "seagreen"],
  2124. ["FFF5EE", "seashell"],
  2125. ["A0522D", "sienna"],
  2126. ["C0C0C0", "silver"],
  2127. ["87CEEB", "skyblue"],
  2128. ["6A5ACD", "slateblue"],
  2129. ["708090", "slategray"],
  2130. ["708090", "slategrey"],
  2131. ["FFFAFA", "snow"],
  2132. ["00FF7F", "springgreen"],
  2133. ["4682B4", "steelblue"],
  2134. ["D2B48C", "tan"],
  2135. ["008080", "teal"],
  2136. ["D8BFD8", "thistle"],
  2137. ["FF6347", "tomato"],
  2138. ["40E0D0", "turquoise"],
  2139. ["EE82EE", "violet"],
  2140. ["F5DEB3", "wheat"],
  2141. ["FFFFFF", "white"],
  2142. ["F5F5F5", "whitesmoke"],
  2143. ["FFFF00", "yellow"],
  2144. ["9ACD32", "yellowgreen"],
  2145. // Existing table
  2146. ["000000", "Black"],
  2147. ["000080", "Navy Blue"],
  2148. ["0000C8", "Dark Blue"],
  2149. ["0000FF", "Blue"],
  2150. ["000741", "Stratos"],
  2151. ["001B1C", "Swamp"],
  2152. ["002387", "Resolution Blue"],
  2153. ["002900", "Deep Fir"],
  2154. ["002E20", "Burnham"],
  2155. ["002FA7", "International Klein Blue"],
  2156. ["003153", "Prussian Blue"],
  2157. ["003366", "Midnight Blue"],
  2158. ["003399", "Smalt"],
  2159. ["003532", "Deep Teal"],
  2160. ["003E40", "Cyprus"],
  2161. ["004620", "Kaitoke Green"],
  2162. ["0047AB", "Cobalt"],
  2163. ["004816", "Crusoe"],
  2164. ["004950", "Sherpa Blue"],
  2165. ["0056A7", "Endeavour"],
  2166. ["00581A", "Camarone"],
  2167. ["0066CC", "Science Blue"],
  2168. ["0066FF", "Blue Ribbon"],
  2169. ["00755E", "Tropical Rain Forest"],
  2170. ["0076A3", "Allports"],
  2171. ["007BA7", "Deep Cerulean"],
  2172. ["007EC7", "Lochmara"],
  2173. ["007FFF", "Azure Radiance"],
  2174. ["008080", "Teal"],
  2175. ["0095B6", "Bondi Blue"],
  2176. ["009DC4", "Pacific Blue"],
  2177. ["00A693", "Persian Green"],
  2178. ["00A86B", "Jade"],
  2179. ["00CC99", "Caribbean Green"],
  2180. ["00CCCC", "Robin's Egg Blue"],
  2181. ["00FF00", "Green"],
  2182. ["00FF7F", "Spring Green"],
  2183. ["00FFFF", "Cyan / Aqua"],
  2184. ["010D1A", "Blue Charcoal"],
  2185. ["011635", "Midnight"],
  2186. ["011D13", "Holly"],
  2187. ["012731", "Daintree"],
  2188. ["01361C", "Cardin Green"],
  2189. ["01371A", "County Green"],
  2190. ["013E62", "Astronaut Blue"],
  2191. ["013F6A", "Regal Blue"],
  2192. ["014B43", "Aqua Deep"],
  2193. ["015E85", "Orient"],
  2194. ["016162", "Blue Stone"],
  2195. ["016D39", "Fun Green"],
  2196. ["01796F", "Pine Green"],
  2197. ["017987", "Blue Lagoon"],
  2198. ["01826B", "Deep Sea"],
  2199. ["01A368", "Green Haze"],
  2200. ["022D15", "English Holly"],
  2201. ["02402C", "Sherwood Green"],
  2202. ["02478E", "Congress Blue"],
  2203. ["024E46", "Evening Sea"],
  2204. ["026395", "Bahama Blue"],
  2205. ["02866F", "Observatory"],
  2206. ["02A4D3", "Cerulean"],
  2207. ["03163C", "Tangaroa"],
  2208. ["032B52", "Green Vogue"],
  2209. ["036A6E", "Mosque"],
  2210. ["041004", "Midnight Moss"],
  2211. ["041322", "Black Pearl"],
  2212. ["042E4C", "Blue Whale"],
  2213. ["044022", "Zuccini"],
  2214. ["044259", "Teal Blue"],
  2215. ["051040", "Deep Cove"],
  2216. ["051657", "Gulf Blue"],
  2217. ["055989", "Venice Blue"],
  2218. ["056F57", "Watercourse"],
  2219. ["062A78", "Catalina Blue"],
  2220. ["063537", "Tiber"],
  2221. ["069B81", "Gossamer"],
  2222. ["06A189", "Niagara"],
  2223. ["073A50", "Tarawera"],
  2224. ["080110", "Jaguar"],
  2225. ["081910", "Black Bean"],
  2226. ["082567", "Deep Sapphire"],
  2227. ["088370", "Elf Green"],
  2228. ["08E8DE", "Bright Turquoise"],
  2229. ["092256", "Downriver"],
  2230. ["09230F", "Palm Green"],
  2231. ["09255D", "Madison"],
  2232. ["093624", "Bottle Green"],
  2233. ["095859", "Deep Sea Green"],
  2234. ["097F4B", "Salem"],
  2235. ["0A001C", "Black Russian"],
  2236. ["0A480D", "Dark Fern"],
  2237. ["0A6906", "Japanese Laurel"],
  2238. ["0A6F75", "Atoll"],
  2239. ["0B0B0B", "Cod Gray"],
  2240. ["0B0F08", "Marshland"],
  2241. ["0B1107", "Gordons Green"],
  2242. ["0B1304", "Black Forest"],
  2243. ["0B6207", "San Felix"],
  2244. ["0BDA51", "Malachite"],
  2245. ["0C0B1D", "Ebony"],
  2246. ["0C0D0F", "Woodsmoke"],
  2247. ["0C1911", "Racing Green"],
  2248. ["0C7A79", "Surfie Green"],
  2249. ["0C8990", "Blue Chill"],
  2250. ["0D0332", "Black Rock"],
  2251. ["0D1117", "Bunker"],
  2252. ["0D1C19", "Aztec"],
  2253. ["0D2E1C", "Bush"],
  2254. ["0E0E18", "Cinder"],
  2255. ["0E2A30", "Firefly"],
  2256. ["0F2D9E", "Torea Bay"],
  2257. ["10121D", "Vulcan"],
  2258. ["101405", "Green Waterloo"],
  2259. ["105852", "Eden"],
  2260. ["110C6C", "Arapawa"],
  2261. ["120A8F", "Ultramarine"],
  2262. ["123447", "Elephant"],
  2263. ["126B40", "Jewel"],
  2264. ["130000", "Diesel"],
  2265. ["130A06", "Asphalt"],
  2266. ["13264D", "Blue Zodiac"],
  2267. ["134F19", "Parsley"],
  2268. ["140600", "Nero"],
  2269. ["1450AA", "Tory Blue"],
  2270. ["151F4C", "Bunting"],
  2271. ["1560BD", "Denim"],
  2272. ["15736B", "Genoa"],
  2273. ["161928", "Mirage"],
  2274. ["161D10", "Hunter Green"],
  2275. ["162A40", "Big Stone"],
  2276. ["163222", "Celtic"],
  2277. ["16322C", "Timber Green"],
  2278. ["163531", "Gable Green"],
  2279. ["171F04", "Pine Tree"],
  2280. ["175579", "Chathams Blue"],
  2281. ["182D09", "Deep Forest Green"],
  2282. ["18587A", "Blumine"],
  2283. ["19330E", "Palm Leaf"],
  2284. ["193751", "Nile Blue"],
  2285. ["1959A8", "Fun Blue"],
  2286. ["1A1A68", "Lucky Point"],
  2287. ["1AB385", "Mountain Meadow"],
  2288. ["1B0245", "Tolopea"],
  2289. ["1B1035", "Haiti"],
  2290. ["1B127B", "Deep Koamaru"],
  2291. ["1B1404", "Acadia"],
  2292. ["1B2F11", "Seaweed"],
  2293. ["1B3162", "Biscay"],
  2294. ["1B659D", "Matisse"],
  2295. ["1C1208", "Crowshead"],
  2296. ["1C1E13", "Rangoon Green"],
  2297. ["1C39BB", "Persian Blue"],
  2298. ["1C402E", "Everglade"],
  2299. ["1C7C7D", "Elm"],
  2300. ["1D6142", "Green Pea"],
  2301. ["1E0F04", "Creole"],
  2302. ["1E1609", "Karaka"],
  2303. ["1E1708", "El Paso"],
  2304. ["1E385B", "Cello"],
  2305. ["1E433C", "Te Papa Green"],
  2306. ["1E90FF", "Dodger Blue"],
  2307. ["1E9AB0", "Eastern Blue"],
  2308. ["1F120F", "Night Rider"],
  2309. ["1FC2C2", "Java"],
  2310. ["20208D", "Jacksons Purple"],
  2311. ["202E54", "Cloud Burst"],
  2312. ["204852", "Blue Dianne"],
  2313. ["211A0E", "Eternity"],
  2314. ["220878", "Deep Blue"],
  2315. ["228B22", "Forest Green"],
  2316. ["233418", "Mallard"],
  2317. ["240A40", "Violet"],
  2318. ["240C02", "Kilamanjaro"],
  2319. ["242A1D", "Log Cabin"],
  2320. ["242E16", "Black Olive"],
  2321. ["24500F", "Green House"],
  2322. ["251607", "Graphite"],
  2323. ["251706", "Cannon Black"],
  2324. ["251F4F", "Port Gore"],
  2325. ["25272C", "Shark"],
  2326. ["25311C", "Green Kelp"],
  2327. ["2596D1", "Curious Blue"],
  2328. ["260368", "Paua"],
  2329. ["26056A", "Paris M"],
  2330. ["261105", "Wood Bark"],
  2331. ["261414", "Gondola"],
  2332. ["262335", "Steel Gray"],
  2333. ["26283B", "Ebony Clay"],
  2334. ["273A81", "Bay of Many"],
  2335. ["27504B", "Plantation"],
  2336. ["278A5B", "Eucalyptus"],
  2337. ["281E15", "Oil"],
  2338. ["283A77", "Astronaut"],
  2339. ["286ACD", "Mariner"],
  2340. ["290C5E", "Violent Violet"],
  2341. ["292130", "Bastille"],
  2342. ["292319", "Zeus"],
  2343. ["292937", "Charade"],
  2344. ["297B9A", "Jelly Bean"],
  2345. ["29AB87", "Jungle Green"],
  2346. ["2A0359", "Cherry Pie"],
  2347. ["2A140E", "Coffee Bean"],
  2348. ["2A2630", "Baltic Sea"],
  2349. ["2A380B", "Turtle Green"],
  2350. ["2A52BE", "Cerulean Blue"],
  2351. ["2B0202", "Sepia Black"],
  2352. ["2B194F", "Valhalla"],
  2353. ["2B3228", "Heavy Metal"],
  2354. ["2C0E8C", "Blue Gem"],
  2355. ["2C1632", "Revolver"],
  2356. ["2C2133", "Bleached Cedar"],
  2357. ["2C8C84", "Lochinvar"],
  2358. ["2D2510", "Mikado"],
  2359. ["2D383A", "Outer Space"],
  2360. ["2D569B", "St Tropaz"],
  2361. ["2E0329", "Jacaranda"],
  2362. ["2E1905", "Jacko Bean"],
  2363. ["2E3222", "Rangitoto"],
  2364. ["2E3F62", "Rhino"],
  2365. ["2E8B57", "Sea Green"],
  2366. ["2EBFD4", "Scooter"],
  2367. ["2F270E", "Onion"],
  2368. ["2F3CB3", "Governor Bay"],
  2369. ["2F519E", "Sapphire"],
  2370. ["2F5A57", "Spectra"],
  2371. ["2F6168", "Casal"],
  2372. ["300529", "Melanzane"],
  2373. ["301F1E", "Cocoa Brown"],
  2374. ["302A0F", "Woodrush"],
  2375. ["304B6A", "San Juan"],
  2376. ["30D5C8", "Turquoise"],
  2377. ["311C17", "Eclipse"],
  2378. ["314459", "Pickled Bluewood"],
  2379. ["315BA1", "Azure"],
  2380. ["31728D", "Calypso"],
  2381. ["317D82", "Paradiso"],
  2382. ["32127A", "Persian Indigo"],
  2383. ["32293A", "Blackcurrant"],
  2384. ["323232", "Mine Shaft"],
  2385. ["325D52", "Stromboli"],
  2386. ["327C14", "Bilbao"],
  2387. ["327DA0", "Astral"],
  2388. ["33036B", "Christalle"],
  2389. ["33292F", "Thunder"],
  2390. ["33CC99", "Shamrock"],
  2391. ["341515", "Tamarind"],
  2392. ["350036", "Mardi Gras"],
  2393. ["350E42", "Valentino"],
  2394. ["350E57", "Jagger"],
  2395. ["353542", "Tuna"],
  2396. ["354E8C", "Chambray"],
  2397. ["363050", "Martinique"],
  2398. ["363534", "Tuatara"],
  2399. ["363C0D", "Waiouru"],
  2400. ["36747D", "Ming"],
  2401. ["368716", "La Palma"],
  2402. ["370202", "Chocolate"],
  2403. ["371D09", "Clinker"],
  2404. ["37290E", "Brown Tumbleweed"],
  2405. ["373021", "Birch"],
  2406. ["377475", "Oracle"],
  2407. ["380474", "Blue Diamond"],
  2408. ["381A51", "Grape"],
  2409. ["383533", "Dune"],
  2410. ["384555", "Oxford Blue"],
  2411. ["384910", "Clover"],
  2412. ["394851", "Limed Spruce"],
  2413. ["396413", "Dell"],
  2414. ["3A0020", "Toledo"],
  2415. ["3A2010", "Sambuca"],
  2416. ["3A2A6A", "Jacarta"],
  2417. ["3A686C", "William"],
  2418. ["3A6A47", "Killarney"],
  2419. ["3AB09E", "Keppel"],
  2420. ["3B000B", "Temptress"],
  2421. ["3B0910", "Aubergine"],
  2422. ["3B1F1F", "Jon"],
  2423. ["3B2820", "Treehouse"],
  2424. ["3B7A57", "Amazon"],
  2425. ["3B91B4", "Boston Blue"],
  2426. ["3C0878", "Windsor"],
  2427. ["3C1206", "Rebel"],
  2428. ["3C1F76", "Meteorite"],
  2429. ["3C2005", "Dark Ebony"],
  2430. ["3C3910", "Camouflage"],
  2431. ["3C4151", "Bright Gray"],
  2432. ["3C4443", "Cape Cod"],
  2433. ["3C493A", "Lunar Green"],
  2434. ["3D0C02", "Bean "],
  2435. ["3D2B1F", "Bistre"],
  2436. ["3D7D52", "Goblin"],
  2437. ["3E0480", "Kingfisher Daisy"],
  2438. ["3E1C14", "Cedar"],
  2439. ["3E2B23", "English Walnut"],
  2440. ["3E2C1C", "Black Marlin"],
  2441. ["3E3A44", "Ship Gray"],
  2442. ["3EABBF", "Pelorous"],
  2443. ["3F2109", "Bronze"],
  2444. ["3F2500", "Cola"],
  2445. ["3F3002", "Madras"],
  2446. ["3F307F", "Minsk"],
  2447. ["3F4C3A", "Cabbage Pont"],
  2448. ["3F583B", "Tom Thumb"],
  2449. ["3F5D53", "Mineral Green"],
  2450. ["3FC1AA", "Puerto Rico"],
  2451. ["3FFF00", "Harlequin"],
  2452. ["401801", "Brown Pod"],
  2453. ["40291D", "Cork"],
  2454. ["403B38", "Masala"],
  2455. ["403D19", "Thatch Green"],
  2456. ["405169", "Fiord"],
  2457. ["40826D", "Viridian"],
  2458. ["40A860", "Chateau Green"],
  2459. ["410056", "Ripe Plum"],
  2460. ["411F10", "Paco"],
  2461. ["412010", "Deep Oak"],
  2462. ["413C37", "Merlin"],
  2463. ["414257", "Gun Powder"],
  2464. ["414C7D", "East Bay"],
  2465. ["4169E1", "Royal Blue"],
  2466. ["41AA78", "Ocean Green"],
  2467. ["420303", "Burnt Maroon"],
  2468. ["423921", "Lisbon Brown"],
  2469. ["427977", "Faded Jade"],
  2470. ["431560", "Scarlet Gum"],
  2471. ["433120", "Iroko"],
  2472. ["433E37", "Armadillo"],
  2473. ["434C59", "River Bed"],
  2474. ["436A0D", "Green Leaf"],
  2475. ["44012D", "Barossa"],
  2476. ["441D00", "Morocco Brown"],
  2477. ["444954", "Mako"],
  2478. ["454936", "Kelp"],
  2479. ["456CAC", "San Marino"],
  2480. ["45B1E8", "Picton Blue"],
  2481. ["460B41", "Loulou"],
  2482. ["462425", "Crater Brown"],
  2483. ["465945", "Gray Asparagus"],
  2484. ["4682B4", "Steel Blue"],
  2485. ["480404", "Rustic Red"],
  2486. ["480607", "Bulgarian Rose"],
  2487. ["480656", "Clairvoyant"],
  2488. ["481C1C", "Cocoa Bean"],
  2489. ["483131", "Woody Brown"],
  2490. ["483C32", "Taupe"],
  2491. ["49170C", "Van Cleef"],
  2492. ["492615", "Brown Derby"],
  2493. ["49371B", "Metallic Bronze"],
  2494. ["495400", "Verdun Green"],
  2495. ["496679", "Blue Bayoux"],
  2496. ["497183", "Bismark"],
  2497. ["4A2A04", "Bracken"],
  2498. ["4A3004", "Deep Bronze"],
  2499. ["4A3C30", "Mondo"],
  2500. ["4A4244", "Tundora"],
  2501. ["4A444B", "Gravel"],
  2502. ["4A4E5A", "Trout"],
  2503. ["4B0082", "Pigment Indigo"],
  2504. ["4B5D52", "Nandor"],
  2505. ["4C3024", "Saddle"],
  2506. ["4C4F56", "Abbey"],
  2507. ["4D0135", "Blackberry"],
  2508. ["4D0A18", "Cab Sav"],
  2509. ["4D1E01", "Indian Tan"],
  2510. ["4D282D", "Cowboy"],
  2511. ["4D282E", "Livid Brown"],
  2512. ["4D3833", "Rock"],
  2513. ["4D3D14", "Punga"],
  2514. ["4D400F", "Bronzetone"],
  2515. ["4D5328", "Woodland"],
  2516. ["4E0606", "Mahogany"],
  2517. ["4E2A5A", "Bossanova"],
  2518. ["4E3B41", "Matterhorn"],
  2519. ["4E420C", "Bronze Olive"],
  2520. ["4E4562", "Mulled Wine"],
  2521. ["4E6649", "Axolotl"],
  2522. ["4E7F9E", "Wedgewood"],
  2523. ["4EABD1", "Shakespeare"],
  2524. ["4F1C70", "Honey Flower"],
  2525. ["4F2398", "Daisy Bush"],
  2526. ["4F69C6", "Indigo"],
  2527. ["4F7942", "Fern Green"],
  2528. ["4F9D5D", "Fruit Salad"],
  2529. ["4FA83D", "Apple"],
  2530. ["504351", "Mortar"],
  2531. ["507096", "Kashmir Blue"],
  2532. ["507672", "Cutty Sark"],
  2533. ["50C878", "Emerald"],
  2534. ["514649", "Emperor"],
  2535. ["516E3D", "Chalet Green"],
  2536. ["517C66", "Como"],
  2537. ["51808F", "Smalt Blue"],
  2538. ["52001F", "Castro"],
  2539. ["520C17", "Maroon Oak"],
  2540. ["523C94", "Gigas"],
  2541. ["533455", "Voodoo"],
  2542. ["534491", "Victoria"],
  2543. ["53824B", "Hippie Green"],
  2544. ["541012", "Heath"],
  2545. ["544333", "Judge Gray"],
  2546. ["54534D", "Fuscous Gray"],
  2547. ["549019", "Vida Loca"],
  2548. ["55280C", "Cioccolato"],
  2549. ["555B10", "Saratoga"],
  2550. ["556D56", "Finlandia"],
  2551. ["5590D9", "Havelock Blue"],
  2552. ["56B4BE", "Fountain Blue"],
  2553. ["578363", "Spring Leaves"],
  2554. ["583401", "Saddle Brown"],
  2555. ["585562", "Scarpa Flow"],
  2556. ["587156", "Cactus"],
  2557. ["589AAF", "Hippie Blue"],
  2558. ["591D35", "Wine Berry"],
  2559. ["592804", "Brown Bramble"],
  2560. ["593737", "Congo Brown"],
  2561. ["594433", "Millbrook"],
  2562. ["5A6E9C", "Waikawa Gray"],
  2563. ["5A87A0", "Horizon"],
  2564. ["5B3013", "Jambalaya"],
  2565. ["5C0120", "Bordeaux"],
  2566. ["5C0536", "Mulberry Wood"],
  2567. ["5C2E01", "Carnaby Tan"],
  2568. ["5C5D75", "Comet"],
  2569. ["5D1E0F", "Redwood"],
  2570. ["5D4C51", "Don Juan"],
  2571. ["5D5C58", "Chicago"],
  2572. ["5D5E37", "Verdigris"],
  2573. ["5D7747", "Dingley"],
  2574. ["5DA19F", "Breaker Bay"],
  2575. ["5E483E", "Kabul"],
  2576. ["5E5D3B", "Hemlock"],
  2577. ["5F3D26", "Irish Coffee"],
  2578. ["5F5F6E", "Mid Gray"],
  2579. ["5F6672", "Shuttle Gray"],
  2580. ["5FA777", "Aqua Forest"],
  2581. ["5FB3AC", "Tradewind"],
  2582. ["604913", "Horses Neck"],
  2583. ["605B73", "Smoky"],
  2584. ["606E68", "Corduroy"],
  2585. ["6093D1", "Danube"],
  2586. ["612718", "Espresso"],
  2587. ["614051", "Eggplant"],
  2588. ["615D30", "Costa Del Sol"],
  2589. ["61845F", "Glade Green"],
  2590. ["622F30", "Buccaneer"],
  2591. ["623F2D", "Quincy"],
  2592. ["624E9A", "Butterfly Bush"],
  2593. ["625119", "West Coast"],
  2594. ["626649", "Finch"],
  2595. ["639A8F", "Patina"],
  2596. ["63B76C", "Fern"],
  2597. ["6456B7", "Blue Violet"],
  2598. ["646077", "Dolphin"],
  2599. ["646463", "Storm Dust"],
  2600. ["646A54", "Siam"],
  2601. ["646E75", "Nevada"],
  2602. ["6495ED", "Cornflower Blue"],
  2603. ["64CCDB", "Viking"],
  2604. ["65000B", "Rosewood"],
  2605. ["651A14", "Cherrywood"],
  2606. ["652DC1", "Purple Heart"],
  2607. ["657220", "Fern Frond"],
  2608. ["65745D", "Willow Grove"],
  2609. ["65869F", "Hoki"],
  2610. ["660045", "Pompadour"],
  2611. ["660099", "Purple"],
  2612. ["66023C", "Tyrian Purple"],
  2613. ["661010", "Dark Tan"],
  2614. ["66B58F", "Silver Tree"],
  2615. ["66FF00", "Bright Green"],
  2616. ["66FF66", "Screamin' Green"],
  2617. ["67032D", "Black Rose"],
  2618. ["675FA6", "Scampi"],
  2619. ["676662", "Ironside Gray"],
  2620. ["678975", "Viridian Green"],
  2621. ["67A712", "Christi"],
  2622. ["683600", "Nutmeg Wood Finish"],
  2623. ["685558", "Zambezi"],
  2624. ["685E6E", "Salt Box"],
  2625. ["692545", "Tawny Port"],
  2626. ["692D54", "Finn"],
  2627. ["695F62", "Scorpion"],
  2628. ["697E9A", "Lynch"],
  2629. ["6A442E", "Spice"],
  2630. ["6A5D1B", "Himalaya"],
  2631. ["6A6051", "Soya Bean"],
  2632. ["6B2A14", "Hairy Heath"],
  2633. ["6B3FA0", "Royal Purple"],
  2634. ["6B4E31", "Shingle Fawn"],
  2635. ["6B5755", "Dorado"],
  2636. ["6B8BA2", "Bermuda Gray"],
  2637. ["6B8E23", "Olive Drab"],
  2638. ["6C3082", "Eminence"],
  2639. ["6CDAE7", "Turquoise Blue"],
  2640. ["6D0101", "Lonestar"],
  2641. ["6D5E54", "Pine Cone"],
  2642. ["6D6C6C", "Dove Gray"],
  2643. ["6D9292", "Juniper"],
  2644. ["6D92A1", "Gothic"],
  2645. ["6E0902", "Red Oxide"],
  2646. ["6E1D14", "Moccaccino"],
  2647. ["6E4826", "Pickled Bean"],
  2648. ["6E4B26", "Dallas"],
  2649. ["6E6D57", "Kokoda"],
  2650. ["6E7783", "Pale Sky"],
  2651. ["6F440C", "Cafe Royale"],
  2652. ["6F6A61", "Flint"],
  2653. ["6F8E63", "Highland"],
  2654. ["6F9D02", "Limeade"],
  2655. ["6FD0C5", "Downy"],
  2656. ["701C1C", "Persian Plum"],
  2657. ["704214", "Sepia"],
  2658. ["704A07", "Antique Bronze"],
  2659. ["704F50", "Ferra"],
  2660. ["706555", "Coffee"],
  2661. ["708090", "Slate Gray"],
  2662. ["711A00", "Cedar Wood Finish"],
  2663. ["71291D", "Metallic Copper"],
  2664. ["714693", "Affair"],
  2665. ["714AB2", "Studio"],
  2666. ["715D47", "Tobacco Brown"],
  2667. ["716338", "Yellow Metal"],
  2668. ["716B56", "Peat"],
  2669. ["716E10", "Olivetone"],
  2670. ["717486", "Storm Gray"],
  2671. ["718080", "Sirocco"],
  2672. ["71D9E2", "Aquamarine Blue"],
  2673. ["72010F", "Venetian Red"],
  2674. ["724A2F", "Old Copper"],
  2675. ["726D4E", "Go Ben"],
  2676. ["727B89", "Raven"],
  2677. ["731E8F", "Seance"],
  2678. ["734A12", "Raw Umber"],
  2679. ["736C9F", "Kimberly"],
  2680. ["736D58", "Crocodile"],
  2681. ["737829", "Crete"],
  2682. ["738678", "Xanadu"],
  2683. ["74640D", "Spicy Mustard"],
  2684. ["747D63", "Limed Ash"],
  2685. ["747D83", "Rolling Stone"],
  2686. ["748881", "Blue Smoke"],
  2687. ["749378", "Laurel"],
  2688. ["74C365", "Mantis"],
  2689. ["755A57", "Russett"],
  2690. ["7563A8", "Deluge"],
  2691. ["76395D", "Cosmic"],
  2692. ["7666C6", "Blue Marguerite"],
  2693. ["76BD17", "Lima"],
  2694. ["76D7EA", "Sky Blue"],
  2695. ["770F05", "Dark Burgundy"],
  2696. ["771F1F", "Crown of Thorns"],
  2697. ["773F1A", "Walnut"],
  2698. ["776F61", "Pablo"],
  2699. ["778120", "Pacifika"],
  2700. ["779E86", "Oxley"],
  2701. ["77DD77", "Pastel Green"],
  2702. ["780109", "Japanese Maple"],
  2703. ["782D19", "Mocha"],
  2704. ["782F16", "Peanut"],
  2705. ["78866B", "Camouflage Green"],
  2706. ["788A25", "Wasabi"],
  2707. ["788BBA", "Ship Cove"],
  2708. ["78A39C", "Sea Nymph"],
  2709. ["795D4C", "Roman Coffee"],
  2710. ["796878", "Old Lavender"],
  2711. ["796989", "Rum"],
  2712. ["796A78", "Fedora"],
  2713. ["796D62", "Sandstone"],
  2714. ["79DEEC", "Spray"],
  2715. ["7A013A", "Siren"],
  2716. ["7A58C1", "Fuchsia Blue"],
  2717. ["7A7A7A", "Boulder"],
  2718. ["7A89B8", "Wild Blue Yonder"],
  2719. ["7AC488", "De York"],
  2720. ["7B3801", "Red Beech"],
  2721. ["7B3F00", "Cinnamon"],
  2722. ["7B6608", "Yukon Gold"],
  2723. ["7B7874", "Tapa"],
  2724. ["7B7C94", "Waterloo "],
  2725. ["7B8265", "Flax Smoke"],
  2726. ["7B9F80", "Amulet"],
  2727. ["7BA05B", "Asparagus"],
  2728. ["7C1C05", "Kenyan Copper"],
  2729. ["7C7631", "Pesto"],
  2730. ["7C778A", "Topaz"],
  2731. ["7C7B7A", "Concord"],
  2732. ["7C7B82", "Jumbo"],
  2733. ["7C881A", "Trendy Green"],
  2734. ["7CA1A6", "Gumbo"],
  2735. ["7CB0A1", "Acapulco"],
  2736. ["7CB7BB", "Neptune"],
  2737. ["7D2C14", "Pueblo"],
  2738. ["7DA98D", "Bay Leaf"],
  2739. ["7DC8F7", "Malibu"],
  2740. ["7DD8C6", "Bermuda"],
  2741. ["7E3A15", "Copper Canyon"],
  2742. ["7F1734", "Claret"],
  2743. ["7F3A02", "Peru Tan"],
  2744. ["7F626D", "Falcon"],
  2745. ["7F7589", "Mobster"],
  2746. ["7F76D3", "Moody Blue"],
  2747. ["7FFF00", "Chartreuse"],
  2748. ["7FFFD4", "Aquamarine"],
  2749. ["800000", "Maroon"],
  2750. ["800B47", "Rose Bud Cherry"],
  2751. ["801818", "Falu Red"],
  2752. ["80341F", "Red Robin"],
  2753. ["803790", "Vivid Violet"],
  2754. ["80461B", "Russet"],
  2755. ["807E79", "Friar Gray"],
  2756. ["808000", "Olive"],
  2757. ["808080", "Gray"],
  2758. ["80B3AE", "Gulf Stream"],
  2759. ["80B3C4", "Glacier"],
  2760. ["80CCEA", "Seagull"],
  2761. ["81422C", "Nutmeg"],
  2762. ["816E71", "Spicy Pink"],
  2763. ["817377", "Empress"],
  2764. ["819885", "Spanish Green"],
  2765. ["826F65", "Sand Dune"],
  2766. ["828685", "Gunsmoke"],
  2767. ["828F72", "Battleship Gray"],
  2768. ["831923", "Merlot"],
  2769. ["837050", "Shadow"],
  2770. ["83AA5D", "Chelsea Cucumber"],
  2771. ["83D0C6", "Monte Carlo"],
  2772. ["843179", "Plum"],
  2773. ["84A0A0", "Granny Smith"],
  2774. ["8581D9", "Chetwode Blue"],
  2775. ["858470", "Bandicoot"],
  2776. ["859FAF", "Bali Hai"],
  2777. ["85C4CC", "Half Baked"],
  2778. ["860111", "Red Devil"],
  2779. ["863C3C", "Lotus"],
  2780. ["86483C", "Ironstone"],
  2781. ["864D1E", "Bull Shot"],
  2782. ["86560A", "Rusty Nail"],
  2783. ["868974", "Bitter"],
  2784. ["86949F", "Regent Gray"],
  2785. ["871550", "Disco"],
  2786. ["87756E", "Americano"],
  2787. ["877C7B", "Hurricane"],
  2788. ["878D91", "Oslo Gray"],
  2789. ["87AB39", "Sushi"],
  2790. ["885342", "Spicy Mix"],
  2791. ["886221", "Kumera"],
  2792. ["888387", "Suva Gray"],
  2793. ["888D65", "Avocado"],
  2794. ["893456", "Camelot"],
  2795. ["893843", "Solid Pink"],
  2796. ["894367", "Cannon Pink"],
  2797. ["897D6D", "Makara"],
  2798. ["8A3324", "Burnt Umber"],
  2799. ["8A73D6", "True V"],
  2800. ["8A8360", "Clay Creek"],
  2801. ["8A8389", "Monsoon"],
  2802. ["8A8F8A", "Stack"],
  2803. ["8AB9F1", "Jordy Blue"],
  2804. ["8B00FF", "Electric Violet"],
  2805. ["8B0723", "Monarch"],
  2806. ["8B6B0B", "Corn Harvest"],
  2807. ["8B8470", "Olive Haze"],
  2808. ["8B847E", "Schooner"],
  2809. ["8B8680", "Natural Gray"],
  2810. ["8B9C90", "Mantle"],
  2811. ["8B9FEE", "Portage"],
  2812. ["8BA690", "Envy"],
  2813. ["8BA9A5", "Cascade"],
  2814. ["8BE6D8", "Riptide"],
  2815. ["8C055E", "Cardinal Pink"],
  2816. ["8C472F", "Mule Fawn"],
  2817. ["8C5738", "Potters Clay"],
  2818. ["8C6495", "Trendy Pink"],
  2819. ["8D0226", "Paprika"],
  2820. ["8D3D38", "Sanguine Brown"],
  2821. ["8D3F3F", "Tosca"],
  2822. ["8D7662", "Cement"],
  2823. ["8D8974", "Granite Green"],
  2824. ["8D90A1", "Manatee"],
  2825. ["8DA8CC", "Polo Blue"],
  2826. ["8E0000", "Red Berry"],
  2827. ["8E4D1E", "Rope"],
  2828. ["8E6F70", "Opium"],
  2829. ["8E775E", "Domino"],
  2830. ["8E8190", "Mamba"],
  2831. ["8EABC1", "Nepal"],
  2832. ["8F021C", "Pohutukawa"],
  2833. ["8F3E33", "El Salva"],
  2834. ["8F4B0E", "Korma"],
  2835. ["8F8176", "Squirrel"],
  2836. ["8FD6B4", "Vista Blue"],
  2837. ["900020", "Burgundy"],
  2838. ["901E1E", "Old Brick"],
  2839. ["907874", "Hemp"],
  2840. ["907B71", "Almond Frost"],
  2841. ["908D39", "Sycamore"],
  2842. ["92000A", "Sangria"],
  2843. ["924321", "Cumin"],
  2844. ["926F5B", "Beaver"],
  2845. ["928573", "Stonewall"],
  2846. ["928590", "Venus"],
  2847. ["9370DB", "Medium Purple"],
  2848. ["93CCEA", "Cornflower"],
  2849. ["93DFB8", "Algae Green"],
  2850. ["944747", "Copper Rust"],
  2851. ["948771", "Arrowtown"],
  2852. ["950015", "Scarlett"],
  2853. ["956387", "Strikemaster"],
  2854. ["959396", "Mountain Mist"],
  2855. ["960018", "Carmine"],
  2856. ["964B00", "Brown"],
  2857. ["967059", "Leather"],
  2858. ["9678B6", "Purple Mountain's Majesty"],
  2859. ["967BB6", "Lavender Purple"],
  2860. ["96A8A1", "Pewter"],
  2861. ["96BBAB", "Summer Green"],
  2862. ["97605D", "Au Chico"],
  2863. ["9771B5", "Wisteria"],
  2864. ["97CD2D", "Atlantis"],
  2865. ["983D61", "Vin Rouge"],
  2866. ["9874D3", "Lilac Bush"],
  2867. ["98777B", "Bazaar"],
  2868. ["98811B", "Hacienda"],
  2869. ["988D77", "Pale Oyster"],
  2870. ["98FF98", "Mint Green"],
  2871. ["990066", "Fresh Eggplant"],
  2872. ["991199", "Violet Eggplant"],
  2873. ["991613", "Tamarillo"],
  2874. ["991B07", "Totem Pole"],
  2875. ["996666", "Copper Rose"],
  2876. ["9966CC", "Amethyst"],
  2877. ["997A8D", "Mountbatten Pink"],
  2878. ["9999CC", "Blue Bell"],
  2879. ["9A3820", "Prairie Sand"],
  2880. ["9A6E61", "Toast"],
  2881. ["9A9577", "Gurkha"],
  2882. ["9AB973", "Olivine"],
  2883. ["9AC2B8", "Shadow Green"],
  2884. ["9B4703", "Oregon"],
  2885. ["9B9E8F", "Lemon Grass"],
  2886. ["9C3336", "Stiletto"],
  2887. ["9D5616", "Hawaiian Tan"],
  2888. ["9DACB7", "Gull Gray"],
  2889. ["9DC209", "Pistachio"],
  2890. ["9DE093", "Granny Smith Apple"],
  2891. ["9DE5FF", "Anakiwa"],
  2892. ["9E5302", "Chelsea Gem"],
  2893. ["9E5B40", "Sepia Skin"],
  2894. ["9EA587", "Sage"],
  2895. ["9EA91F", "Citron"],
  2896. ["9EB1CD", "Rock Blue"],
  2897. ["9EDEE0", "Morning Glory"],
  2898. ["9F381D", "Cognac"],
  2899. ["9F821C", "Reef Gold"],
  2900. ["9F9F9C", "Star Dust"],
  2901. ["9FA0B1", "Santas Gray"],
  2902. ["9FD7D3", "Sinbad"],
  2903. ["9FDD8C", "Feijoa"],
  2904. ["A02712", "Tabasco"],
  2905. ["A1750D", "Buttered Rum"],
  2906. ["A1ADB5", "Hit Gray"],
  2907. ["A1C50A", "Citrus"],
  2908. ["A1DAD7", "Aqua Island"],
  2909. ["A1E9DE", "Water Leaf"],
  2910. ["A2006D", "Flirt"],
  2911. ["A23B6C", "Rouge"],
  2912. ["A26645", "Cape Palliser"],
  2913. ["A2AAB3", "Gray Chateau"],
  2914. ["A2AEAB", "Edward"],
  2915. ["A3807B", "Pharlap"],
  2916. ["A397B4", "Amethyst Smoke"],
  2917. ["A3E3ED", "Blizzard Blue"],
  2918. ["A4A49D", "Delta"],
  2919. ["A4A6D3", "Wistful"],
  2920. ["A4AF6E", "Green Smoke"],
  2921. ["A50B5E", "Jazzberry Jam"],
  2922. ["A59B91", "Zorba"],
  2923. ["A5CB0C", "Bahia"],
  2924. ["A62F20", "Roof Terracotta"],
  2925. ["A65529", "Paarl"],
  2926. ["A68B5B", "Barley Corn"],
  2927. ["A69279", "Donkey Brown"],
  2928. ["A6A29A", "Dawn"],
  2929. ["A72525", "Mexican Red"],
  2930. ["A7882C", "Luxor Gold"],
  2931. ["A85307", "Rich Gold"],
  2932. ["A86515", "Reno Sand"],
  2933. ["A86B6B", "Coral Tree"],
  2934. ["A8989B", "Dusty Gray"],
  2935. ["A899E6", "Dull Lavender"],
  2936. ["A8A589", "Tallow"],
  2937. ["A8AE9C", "Bud"],
  2938. ["A8AF8E", "Locust"],
  2939. ["A8BD9F", "Norway"],
  2940. ["A8E3BD", "Chinook"],
  2941. ["A9A491", "Gray Olive"],
  2942. ["A9ACB6", "Aluminium"],
  2943. ["A9B2C3", "Cadet Blue"],
  2944. ["A9B497", "Schist"],
  2945. ["A9BDBF", "Tower Gray"],
  2946. ["A9BEF2", "Perano"],
  2947. ["A9C6C2", "Opal"],
  2948. ["AA375A", "Night Shadz"],
  2949. ["AA4203", "Fire"],
  2950. ["AA8B5B", "Muesli"],
  2951. ["AA8D6F", "Sandal"],
  2952. ["AAA5A9", "Shady Lady"],
  2953. ["AAA9CD", "Logan"],
  2954. ["AAABB7", "Spun Pearl"],
  2955. ["AAD6E6", "Regent St Blue"],
  2956. ["AAF0D1", "Magic Mint"],
  2957. ["AB0563", "Lipstick"],
  2958. ["AB3472", "Royal Heath"],
  2959. ["AB917A", "Sandrift"],
  2960. ["ABA0D9", "Cold Purple"],
  2961. ["ABA196", "Bronco"],
  2962. ["AC8A56", "Limed Oak"],
  2963. ["AC91CE", "East Side"],
  2964. ["AC9E22", "Lemon Ginger"],
  2965. ["ACA494", "Napa"],
  2966. ["ACA586", "Hillary"],
  2967. ["ACA59F", "Cloudy"],
  2968. ["ACACAC", "Silver Chalice"],
  2969. ["ACB78E", "Swamp Green"],
  2970. ["ACCBB1", "Spring Rain"],
  2971. ["ACDD4D", "Conifer"],
  2972. ["ACE1AF", "Celadon"],
  2973. ["AD781B", "Mandalay"],
  2974. ["ADBED1", "Casper"],
  2975. ["ADDFAD", "Moss Green"],
  2976. ["ADE6C4", "Padua"],
  2977. ["ADFF2F", "Green Yellow"],
  2978. ["AE4560", "Hippie Pink"],
  2979. ["AE6020", "Desert"],
  2980. ["AE809E", "Bouquet"],
  2981. ["AF4035", "Medium Carmine"],
  2982. ["AF4D43", "Apple Blossom"],
  2983. ["AF593E", "Brown Rust"],
  2984. ["AF8751", "Driftwood"],
  2985. ["AF8F2C", "Alpine"],
  2986. ["AF9F1C", "Lucky"],
  2987. ["AFA09E", "Martini"],
  2988. ["AFB1B8", "Bombay"],
  2989. ["AFBDD9", "Pigeon Post"],
  2990. ["B04C6A", "Cadillac"],
  2991. ["B05D54", "Matrix"],
  2992. ["B05E81", "Tapestry"],
  2993. ["B06608", "Mai Tai"],
  2994. ["B09A95", "Del Rio"],
  2995. ["B0E0E6", "Powder Blue"],
  2996. ["B0E313", "Inch Worm"],
  2997. ["B10000", "Bright Red"],
  2998. ["B14A0B", "Vesuvius"],
  2999. ["B1610B", "Pumpkin Skin"],
  3000. ["B16D52", "Santa Fe"],
  3001. ["B19461", "Teak"],
  3002. ["B1E2C1", "Fringy Flower"],
  3003. ["B1F4E7", "Ice Cold"],
  3004. ["B20931", "Shiraz"],
  3005. ["B2A1EA", "Biloba Flower"],
  3006. ["B32D29", "Tall Poppy"],
  3007. ["B35213", "Fiery Orange"],
  3008. ["B38007", "Hot Toddy"],
  3009. ["B3AF95", "Taupe Gray"],
  3010. ["B3C110", "La Rioja"],
  3011. ["B43332", "Well Read"],
  3012. ["B44668", "Blush"],
  3013. ["B4CFD3", "Jungle Mist"],
  3014. ["B57281", "Turkish Rose"],
  3015. ["B57EDC", "Lavender"],
  3016. ["B5A27F", "Mongoose"],
  3017. ["B5B35C", "Olive Green"],
  3018. ["B5D2CE", "Jet Stream"],
  3019. ["B5ECDF", "Cruise"],
  3020. ["B6316C", "Hibiscus"],
  3021. ["B69D98", "Thatch"],
  3022. ["B6B095", "Heathered Gray"],
  3023. ["B6BAA4", "Eagle"],
  3024. ["B6D1EA", "Spindle"],
  3025. ["B6D3BF", "Gum Leaf"],
  3026. ["B7410E", "Rust"],
  3027. ["B78E5C", "Muddy Waters"],
  3028. ["B7A214", "Sahara"],
  3029. ["B7A458", "Husk"],
  3030. ["B7B1B1", "Nobel"],
  3031. ["B7C3D0", "Heather"],
  3032. ["B7F0BE", "Madang"],
  3033. ["B81104", "Milano Red"],
  3034. ["B87333", "Copper"],
  3035. ["B8B56A", "Gimblet"],
  3036. ["B8C1B1", "Green Spring"],
  3037. ["B8C25D", "Celery"],
  3038. ["B8E0F9", "Sail"],
  3039. ["B94E48", "Chestnut"],
  3040. ["B95140", "Crail"],
  3041. ["B98D28", "Marigold"],
  3042. ["B9C46A", "Wild Willow"],
  3043. ["B9C8AC", "Rainee"],
  3044. ["BA0101", "Guardsman Red"],
  3045. ["BA450C", "Rock Spray"],
  3046. ["BA6F1E", "Bourbon"],
  3047. ["BA7F03", "Pirate Gold"],
  3048. ["BAB1A2", "Nomad"],
  3049. ["BAC7C9", "Submarine"],
  3050. ["BAEEF9", "Charlotte"],
  3051. ["BB3385", "Medium Red Violet"],
  3052. ["BB8983", "Brandy Rose"],
  3053. ["BBD009", "Rio Grande"],
  3054. ["BBD7C1", "Surf"],
  3055. ["BCC9C2", "Powder Ash"],
  3056. ["BD5E2E", "Tuscany"],
  3057. ["BD978E", "Quicksand"],
  3058. ["BDB1A8", "Silk"],
  3059. ["BDB2A1", "Malta"],
  3060. ["BDB3C7", "Chatelle"],
  3061. ["BDBBD7", "Lavender Gray"],
  3062. ["BDBDC6", "French Gray"],
  3063. ["BDC8B3", "Clay Ash"],
  3064. ["BDC9CE", "Loblolly"],
  3065. ["BDEDFD", "French Pass"],
  3066. ["BEA6C3", "London Hue"],
  3067. ["BEB5B7", "Pink Swan"],
  3068. ["BEDE0D", "Fuego"],
  3069. ["BF5500", "Rose of Sharon"],
  3070. ["BFB8B0", "Tide"],
  3071. ["BFBED8", "Blue Haze"],
  3072. ["BFC1C2", "Silver Sand"],
  3073. ["BFC921", "Key Lime Pie"],
  3074. ["BFDBE2", "Ziggurat"],
  3075. ["BFFF00", "Lime"],
  3076. ["C02B18", "Thunderbird"],
  3077. ["C04737", "Mojo"],
  3078. ["C08081", "Old Rose"],
  3079. ["C0C0C0", "Silver"],
  3080. ["C0D3B9", "Pale Leaf"],
  3081. ["C0D8B6", "Pixie Green"],
  3082. ["C1440E", "Tia Maria"],
  3083. ["C154C1", "Fuchsia Pink"],
  3084. ["C1A004", "Buddha Gold"],
  3085. ["C1B7A4", "Bison Hide"],
  3086. ["C1BAB0", "Tea"],
  3087. ["C1BECD", "Gray Suit"],
  3088. ["C1D7B0", "Sprout"],
  3089. ["C1F07C", "Sulu"],
  3090. ["C26B03", "Indochine"],
  3091. ["C2955D", "Twine"],
  3092. ["C2BDB6", "Cotton Seed"],
  3093. ["C2CAC4", "Pumice"],
  3094. ["C2E8E5", "Jagged Ice"],
  3095. ["C32148", "Maroon Flush"],
  3096. ["C3B091", "Indian Khaki"],
  3097. ["C3BFC1", "Pale Slate"],
  3098. ["C3C3BD", "Gray Nickel"],
  3099. ["C3CDE6", "Periwinkle Gray"],
  3100. ["C3D1D1", "Tiara"],
  3101. ["C3DDF9", "Tropical Blue"],
  3102. ["C41E3A", "Cardinal"],
  3103. ["C45655", "Fuzzy Wuzzy Brown"],
  3104. ["C45719", "Orange Roughy"],
  3105. ["C4C4BC", "Mist Gray"],
  3106. ["C4D0B0", "Coriander"],
  3107. ["C4F4EB", "Mint Tulip"],
  3108. ["C54B8C", "Mulberry"],
  3109. ["C59922", "Nugget"],
  3110. ["C5994B", "Tussock"],
  3111. ["C5DBCA", "Sea Mist"],
  3112. ["C5E17A", "Yellow Green"],
  3113. ["C62D42", "Brick Red"],
  3114. ["C6726B", "Contessa"],
  3115. ["C69191", "Oriental Pink"],
  3116. ["C6A84B", "Roti"],
  3117. ["C6C3B5", "Ash"],
  3118. ["C6C8BD", "Kangaroo"],
  3119. ["C6E610", "Las Palmas"],
  3120. ["C7031E", "Monza"],
  3121. ["C71585", "Red Violet"],
  3122. ["C7BCA2", "Coral Reef"],
  3123. ["C7C1FF", "Melrose"],
  3124. ["C7C4BF", "Cloud"],
  3125. ["C7C9D5", "Ghost"],
  3126. ["C7CD90", "Pine Glade"],
  3127. ["C7DDE5", "Botticelli"],
  3128. ["C88A65", "Antique Brass"],
  3129. ["C8A2C8", "Lilac"],
  3130. ["C8A528", "Hokey Pokey"],
  3131. ["C8AABF", "Lily"],
  3132. ["C8B568", "Laser"],
  3133. ["C8E3D7", "Edgewater"],
  3134. ["C96323", "Piper"],
  3135. ["C99415", "Pizza"],
  3136. ["C9A0DC", "Light Wisteria"],
  3137. ["C9B29B", "Rodeo Dust"],
  3138. ["C9B35B", "Sundance"],
  3139. ["C9B93B", "Earls Green"],
  3140. ["C9C0BB", "Silver Rust"],
  3141. ["C9D9D2", "Conch"],
  3142. ["C9FFA2", "Reef"],
  3143. ["C9FFE5", "Aero Blue"],
  3144. ["CA3435", "Flush Mahogany"],
  3145. ["CABB48", "Turmeric"],
  3146. ["CADCD4", "Paris White"],
  3147. ["CAE00D", "Bitter Lemon"],
  3148. ["CAE6DA", "Skeptic"],
  3149. ["CB8FA9", "Viola"],
  3150. ["CBCAB6", "Foggy Gray"],
  3151. ["CBD3B0", "Green Mist"],
  3152. ["CBDBD6", "Nebula"],
  3153. ["CC3333", "Persian Red"],
  3154. ["CC5500", "Burnt Orange"],
  3155. ["CC7722", "Ochre"],
  3156. ["CC8899", "Puce"],
  3157. ["CCCAA8", "Thistle Green"],
  3158. ["CCCCFF", "Periwinkle"],
  3159. ["CCFF00", "Electric Lime"],
  3160. ["CD5700", "Tenn"],
  3161. ["CD5C5C", "Chestnut Rose"],
  3162. ["CD8429", "Brandy Punch"],
  3163. ["CDF4FF", "Onahau"],
  3164. ["CEB98F", "Sorrell Brown"],
  3165. ["CEBABA", "Cold Turkey"],
  3166. ["CEC291", "Yuma"],
  3167. ["CEC7A7", "Chino"],
  3168. ["CFA39D", "Eunry"],
  3169. ["CFB53B", "Old Gold"],
  3170. ["CFDCCF", "Tasman"],
  3171. ["CFE5D2", "Surf Crest"],
  3172. ["CFF9F3", "Humming Bird"],
  3173. ["CFFAF4", "Scandal"],
  3174. ["D05F04", "Red Stage"],
  3175. ["D06DA1", "Hopbush"],
  3176. ["D07D12", "Meteor"],
  3177. ["D0BEF8", "Perfume"],
  3178. ["D0C0E5", "Prelude"],
  3179. ["D0F0C0", "Tea Green"],
  3180. ["D18F1B", "Geebung"],
  3181. ["D1BEA8", "Vanilla"],
  3182. ["D1C6B4", "Soft Amber"],
  3183. ["D1D2CA", "Celeste"],
  3184. ["D1D2DD", "Mischka"],
  3185. ["D1E231", "Pear"],
  3186. ["D2691E", "Hot Cinnamon"],
  3187. ["D27D46", "Raw Sienna"],
  3188. ["D29EAA", "Careys Pink"],
  3189. ["D2B48C", "Tan"],
  3190. ["D2DA97", "Deco"],
  3191. ["D2F6DE", "Blue Romance"],
  3192. ["D2F8B0", "Gossip"],
  3193. ["D3CBBA", "Sisal"],
  3194. ["D3CDC5", "Swirl"],
  3195. ["D47494", "Charm"],
  3196. ["D4B6AF", "Clam Shell"],
  3197. ["D4BF8D", "Straw"],
  3198. ["D4C4A8", "Akaroa"],
  3199. ["D4CD16", "Bird Flower"],
  3200. ["D4D7D9", "Iron"],
  3201. ["D4DFE2", "Geyser"],
  3202. ["D4E2FC", "Hawkes Blue"],
  3203. ["D54600", "Grenadier"],
  3204. ["D591A4", "Can Can"],
  3205. ["D59A6F", "Whiskey"],
  3206. ["D5D195", "Winter Hazel"],
  3207. ["D5F6E3", "Granny Apple"],
  3208. ["D69188", "My Pink"],
  3209. ["D6C562", "Tacha"],
  3210. ["D6CEF6", "Moon Raker"],
  3211. ["D6D6D1", "Quill Gray"],
  3212. ["D6FFDB", "Snowy Mint"],
  3213. ["D7837F", "New York Pink"],
  3214. ["D7C498", "Pavlova"],
  3215. ["D7D0FF", "Fog"],
  3216. ["D84437", "Valencia"],
  3217. ["D87C63", "Japonica"],
  3218. ["D8BFD8", "Thistle"],
  3219. ["D8C2D5", "Maverick"],
  3220. ["D8FCFA", "Foam"],
  3221. ["D94972", "Cabaret"],
  3222. ["D99376", "Burning Sand"],
  3223. ["D9B99B", "Cameo"],
  3224. ["D9D6CF", "Timberwolf"],
  3225. ["D9DCC1", "Tana"],
  3226. ["D9E4F5", "Link Water"],
  3227. ["D9F7FF", "Mabel"],
  3228. ["DA3287", "Cerise"],
  3229. ["DA5B38", "Flame Pea"],
  3230. ["DA6304", "Bamboo"],
  3231. ["DA6A41", "Red Damask"],
  3232. ["DA70D6", "Orchid"],
  3233. ["DA8A67", "Copperfield"],
  3234. ["DAA520", "Golden Grass"],
  3235. ["DAECD6", "Zanah"],
  3236. ["DAF4F0", "Iceberg"],
  3237. ["DAFAFF", "Oyster Bay"],
  3238. ["DB5079", "Cranberry"],
  3239. ["DB9690", "Petite Orchid"],
  3240. ["DB995E", "Di Serria"],
  3241. ["DBDBDB", "Alto"],
  3242. ["DBFFF8", "Frosted Mint"],
  3243. ["DC143C", "Crimson"],
  3244. ["DC4333", "Punch"],
  3245. ["DCB20C", "Galliano"],
  3246. ["DCB4BC", "Blossom"],
  3247. ["DCD747", "Wattle"],
  3248. ["DCD9D2", "Westar"],
  3249. ["DCDDCC", "Moon Mist"],
  3250. ["DCEDB4", "Caper"],
  3251. ["DCF0EA", "Swans Down"],
  3252. ["DDD6D5", "Swiss Coffee"],
  3253. ["DDF9F1", "White Ice"],
  3254. ["DE3163", "Cerise Red"],
  3255. ["DE6360", "Roman"],
  3256. ["DEA681", "Tumbleweed"],
  3257. ["DEBA13", "Gold Tips"],
  3258. ["DEC196", "Brandy"],
  3259. ["DECBC6", "Wafer"],
  3260. ["DED4A4", "Sapling"],
  3261. ["DED717", "Barberry"],
  3262. ["DEE5C0", "Beryl Green"],
  3263. ["DEF5FF", "Pattens Blue"],
  3264. ["DF73FF", "Heliotrope"],
  3265. ["DFBE6F", "Apache"],
  3266. ["DFCD6F", "Chenin"],
  3267. ["DFCFDB", "Lola"],
  3268. ["DFECDA", "Willow Brook"],
  3269. ["DFFF00", "Chartreuse Yellow"],
  3270. ["E0B0FF", "Mauve"],
  3271. ["E0B646", "Anzac"],
  3272. ["E0B974", "Harvest Gold"],
  3273. ["E0C095", "Calico"],
  3274. ["E0FFFF", "Baby Blue"],
  3275. ["E16865", "Sunglo"],
  3276. ["E1BC64", "Equator"],
  3277. ["E1C0C8", "Pink Flare"],
  3278. ["E1E6D6", "Periglacial Blue"],
  3279. ["E1EAD4", "Kidnapper"],
  3280. ["E1F6E8", "Tara"],
  3281. ["E25465", "Mandy"],
  3282. ["E2725B", "Terracotta"],
  3283. ["E28913", "Golden Bell"],
  3284. ["E292C0", "Shocking"],
  3285. ["E29418", "Dixie"],
  3286. ["E29CD2", "Light Orchid"],
  3287. ["E2D8ED", "Snuff"],
  3288. ["E2EBED", "Mystic"],
  3289. ["E2F3EC", "Apple Green"],
  3290. ["E30B5C", "Razzmatazz"],
  3291. ["E32636", "Alizarin Crimson"],
  3292. ["E34234", "Cinnabar"],
  3293. ["E3BEBE", "Cavern Pink"],
  3294. ["E3F5E1", "Peppermint"],
  3295. ["E3F988", "Mindaro"],
  3296. ["E47698", "Deep Blush"],
  3297. ["E49B0F", "Gamboge"],
  3298. ["E4C2D5", "Melanie"],
  3299. ["E4CFDE", "Twilight"],
  3300. ["E4D1C0", "Bone"],
  3301. ["E4D422", "Sunflower"],
  3302. ["E4D5B7", "Grain Brown"],
  3303. ["E4D69B", "Zombie"],
  3304. ["E4F6E7", "Frostee"],
  3305. ["E4FFD1", "Snow Flurry"],
  3306. ["E52B50", "Amaranth"],
  3307. ["E5841B", "Zest"],
  3308. ["E5CCC9", "Dust Storm"],
  3309. ["E5D7BD", "Stark White"],
  3310. ["E5D8AF", "Hampton"],
  3311. ["E5E0E1", "Bon Jour"],
  3312. ["E5E5E5", "Mercury"],
  3313. ["E5F9F6", "Polar"],
  3314. ["E64E03", "Trinidad"],
  3315. ["E6BE8A", "Gold Sand"],
  3316. ["E6BEA5", "Cashmere"],
  3317. ["E6D7B9", "Double Spanish White"],
  3318. ["E6E4D4", "Satin Linen"],
  3319. ["E6F2EA", "Harp"],
  3320. ["E6F8F3", "Off Green"],
  3321. ["E6FFE9", "Hint of Green"],
  3322. ["E6FFFF", "Tranquil"],
  3323. ["E77200", "Mango Tango"],
  3324. ["E7730A", "Christine"],
  3325. ["E79F8C", "Tonys Pink"],
  3326. ["E79FC4", "Kobi"],
  3327. ["E7BCB4", "Rose Fog"],
  3328. ["E7BF05", "Corn"],
  3329. ["E7CD8C", "Putty"],
  3330. ["E7ECE6", "Gray Nurse"],
  3331. ["E7F8FF", "Lily White"],
  3332. ["E7FEFF", "Bubbles"],
  3333. ["E89928", "Fire Bush"],
  3334. ["E8B9B3", "Shilo"],
  3335. ["E8E0D5", "Pearl Bush"],
  3336. ["E8EBE0", "Green White"],
  3337. ["E8F1D4", "Chrome White"],
  3338. ["E8F2EB", "Gin"],
  3339. ["E8F5F2", "Aqua Squeeze"],
  3340. ["E96E00", "Clementine"],
  3341. ["E97451", "Burnt Sienna"],
  3342. ["E97C07", "Tahiti Gold"],
  3343. ["E9CECD", "Oyster Pink"],
  3344. ["E9D75A", "Confetti"],
  3345. ["E9E3E3", "Ebb"],
  3346. ["E9F8ED", "Ottoman"],
  3347. ["E9FFFD", "Clear Day"],
  3348. ["EA88A8", "Carissma"],
  3349. ["EAAE69", "Porsche"],
  3350. ["EAB33B", "Tulip Tree"],
  3351. ["EAC674", "Rob Roy"],
  3352. ["EADAB8", "Raffia"],
  3353. ["EAE8D4", "White Rock"],
  3354. ["EAF6EE", "Panache"],
  3355. ["EAF6FF", "Solitude"],
  3356. ["EAF9F5", "Aqua Spring"],
  3357. ["EAFFFE", "Dew"],
  3358. ["EB9373", "Apricot"],
  3359. ["EBC2AF", "Zinnwaldite"],
  3360. ["ECA927", "Fuel Yellow"],
  3361. ["ECC54E", "Ronchi"],
  3362. ["ECC7EE", "French Lilac"],
  3363. ["ECCDB9", "Just Right"],
  3364. ["ECE090", "Wild Rice"],
  3365. ["ECEBBD", "Fall Green"],
  3366. ["ECEBCE", "Aths Special"],
  3367. ["ECF245", "Starship"],
  3368. ["ED0A3F", "Red Ribbon"],
  3369. ["ED7A1C", "Tango"],
  3370. ["ED9121", "Carrot Orange"],
  3371. ["ED989E", "Sea Pink"],
  3372. ["EDB381", "Tacao"],
  3373. ["EDC9AF", "Desert Sand"],
  3374. ["EDCDAB", "Pancho"],
  3375. ["EDDCB1", "Chamois"],
  3376. ["EDEA99", "Primrose"],
  3377. ["EDF5DD", "Frost"],
  3378. ["EDF5F5", "Aqua Haze"],
  3379. ["EDF6FF", "Zumthor"],
  3380. ["EDF9F1", "Narvik"],
  3381. ["EDFC84", "Honeysuckle"],
  3382. ["EE82EE", "Lavender Magenta"],
  3383. ["EEC1BE", "Beauty Bush"],
  3384. ["EED794", "Chalky"],
  3385. ["EED9C4", "Almond"],
  3386. ["EEDC82", "Flax"],
  3387. ["EEDEDA", "Bizarre"],
  3388. ["EEE3AD", "Double Colonial White"],
  3389. ["EEEEE8", "Cararra"],
  3390. ["EEEF78", "Manz"],
  3391. ["EEF0C8", "Tahuna Sands"],
  3392. ["EEF0F3", "Athens Gray"],
  3393. ["EEF3C3", "Tusk"],
  3394. ["EEF4DE", "Loafer"],
  3395. ["EEF6F7", "Catskill White"],
  3396. ["EEFDFF", "Twilight Blue"],
  3397. ["EEFF9A", "Jonquil"],
  3398. ["EEFFE2", "Rice Flower"],
  3399. ["EF863F", "Jaffa"],
  3400. ["EFEFEF", "Gallery"],
  3401. ["EFF2F3", "Porcelain"],
  3402. ["F091A9", "Mauvelous"],
  3403. ["F0D52D", "Golden Dream"],
  3404. ["F0DB7D", "Golden Sand"],
  3405. ["F0DC82", "Buff"],
  3406. ["F0E2EC", "Prim"],
  3407. ["F0E68C", "Khaki"],
  3408. ["F0EEFD", "Selago"],
  3409. ["F0EEFF", "Titan White"],
  3410. ["F0F8FF", "Alice Blue"],
  3411. ["F0FCEA", "Feta"],
  3412. ["F18200", "Gold Drop"],
  3413. ["F19BAB", "Wewak"],
  3414. ["F1E788", "Sahara Sand"],
  3415. ["F1E9D2", "Parchment"],
  3416. ["F1E9FF", "Blue Chalk"],
  3417. ["F1EEC1", "Mint Julep"],
  3418. ["F1F1F1", "Seashell"],
  3419. ["F1F7F2", "Saltpan"],
  3420. ["F1FFAD", "Tidal"],
  3421. ["F1FFC8", "Chiffon"],
  3422. ["F2552A", "Flamingo"],
  3423. ["F28500", "Tangerine"],
  3424. ["F2C3B2", "Mandys Pink"],
  3425. ["F2F2F2", "Concrete"],
  3426. ["F2FAFA", "Black Squeeze"],
  3427. ["F34723", "Pomegranate"],
  3428. ["F3AD16", "Buttercup"],
  3429. ["F3D69D", "New Orleans"],
  3430. ["F3D9DF", "Vanilla Ice"],
  3431. ["F3E7BB", "Sidecar"],
  3432. ["F3E9E5", "Dawn Pink"],
  3433. ["F3EDCF", "Wheatfield"],
  3434. ["F3FB62", "Canary"],
  3435. ["F3FBD4", "Orinoco"],
  3436. ["F3FFD8", "Carla"],
  3437. ["F400A1", "Hollywood Cerise"],
  3438. ["F4A460", "Sandy brown"],
  3439. ["F4C430", "Saffron"],
  3440. ["F4D81C", "Ripe Lemon"],
  3441. ["F4EBD3", "Janna"],
  3442. ["F4F2EE", "Pampas"],
  3443. ["F4F4F4", "Wild Sand"],
  3444. ["F4F8FF", "Zircon"],
  3445. ["F57584", "Froly"],
  3446. ["F5C85C", "Cream Can"],
  3447. ["F5C999", "Manhattan"],
  3448. ["F5D5A0", "Maize"],
  3449. ["F5DEB3", "Wheat"],
  3450. ["F5E7A2", "Sandwisp"],
  3451. ["F5E7E2", "Pot Pourri"],
  3452. ["F5E9D3", "Albescent White"],
  3453. ["F5EDEF", "Soft Peach"],
  3454. ["F5F3E5", "Ecru White"],
  3455. ["F5F5DC", "Beige"],
  3456. ["F5FB3D", "Golden Fizz"],
  3457. ["F5FFBE", "Australian Mint"],
  3458. ["F64A8A", "French Rose"],
  3459. ["F653A6", "Brilliant Rose"],
  3460. ["F6A4C9", "Illusion"],
  3461. ["F6F0E6", "Merino"],
  3462. ["F6F7F7", "Black Haze"],
  3463. ["F6FFDC", "Spring Sun"],
  3464. ["F7468A", "Violet Red"],
  3465. ["F77703", "Chilean Fire"],
  3466. ["F77FBE", "Persian Pink"],
  3467. ["F7B668", "Rajah"],
  3468. ["F7C8DA", "Azalea"],
  3469. ["F7DBE6", "We Peep"],
  3470. ["F7F2E1", "Quarter Spanish White"],
  3471. ["F7F5FA", "Whisper"],
  3472. ["F7FAF7", "Snow Drift"],
  3473. ["F8B853", "Casablanca"],
  3474. ["F8C3DF", "Chantilly"],
  3475. ["F8D9E9", "Cherub"],
  3476. ["F8DB9D", "Marzipan"],
  3477. ["F8DD5C", "Energy Yellow"],
  3478. ["F8E4BF", "Givry"],
  3479. ["F8F0E8", "White Linen"],
  3480. ["F8F4FF", "Magnolia"],
  3481. ["F8F6F1", "Spring Wood"],
  3482. ["F8F7DC", "Coconut Cream"],
  3483. ["F8F7FC", "White Lilac"],
  3484. ["F8F8F7", "Desert Storm"],
  3485. ["F8F99C", "Texas"],
  3486. ["F8FACD", "Corn Field"],
  3487. ["F8FDD3", "Mimosa"],
  3488. ["F95A61", "Carnation"],
  3489. ["F9BF58", "Saffron Mango"],
  3490. ["F9E0ED", "Carousel Pink"],
  3491. ["F9E4BC", "Dairy Cream"],
  3492. ["F9E663", "Portica"],
  3493. ["F9EAF3", "Amour"],
  3494. ["F9F8E4", "Rum Swizzle"],
  3495. ["F9FF8B", "Dolly"],
  3496. ["F9FFF6", "Sugar Cane"],
  3497. ["FA7814", "Ecstasy"],
  3498. ["FA9D5A", "Tan Hide"],
  3499. ["FAD3A2", "Corvette"],
  3500. ["FADFAD", "Peach Yellow"],
  3501. ["FAE600", "Turbo"],
  3502. ["FAEAB9", "Astra"],
  3503. ["FAECCC", "Champagne"],
  3504. ["FAF0E6", "Linen"],
  3505. ["FAF3F0", "Fantasy"],
  3506. ["FAF7D6", "Citrine White"],
  3507. ["FAFAFA", "Alabaster"],
  3508. ["FAFDE4", "Hint of Yellow"],
  3509. ["FAFFA4", "Milan"],
  3510. ["FB607F", "Brink Pink"],
  3511. ["FB8989", "Geraldine"],
  3512. ["FBA0E3", "Lavender Rose"],
  3513. ["FBA129", "Sea Buckthorn"],
  3514. ["FBAC13", "Sun"],
  3515. ["FBAED2", "Lavender Pink"],
  3516. ["FBB2A3", "Rose Bud"],
  3517. ["FBBEDA", "Cupid"],
  3518. ["FBCCE7", "Classic Rose"],
  3519. ["FBCEB1", "Apricot Peach"],
  3520. ["FBE7B2", "Banana Mania"],
  3521. ["FBE870", "Marigold Yellow"],
  3522. ["FBE96C", "Festival"],
  3523. ["FBEA8C", "Sweet Corn"],
  3524. ["FBEC5D", "Candy Corn"],
  3525. ["FBF9F9", "Hint of Red"],
  3526. ["FBFFBA", "Shalimar"],
  3527. ["FC0FC0", "Shocking Pink"],
  3528. ["FC80A5", "Tickle Me Pink"],
  3529. ["FC9C1D", "Tree Poppy"],
  3530. ["FCC01E", "Lightning Yellow"],
  3531. ["FCD667", "Goldenrod"],
  3532. ["FCD917", "Candlelight"],
  3533. ["FCDA98", "Cherokee"],
  3534. ["FCF4D0", "Double Pearl Lusta"],
  3535. ["FCF4DC", "Pearl Lusta"],
  3536. ["FCF8F7", "Vista White"],
  3537. ["FCFBF3", "Bianca"],
  3538. ["FCFEDA", "Moon Glow"],
  3539. ["FCFFE7", "China Ivory"],
  3540. ["FCFFF9", "Ceramic"],
  3541. ["FD0E35", "Torch Red"],
  3542. ["FD5B78", "Wild Watermelon"],
  3543. ["FD7B33", "Crusta"],
  3544. ["FD7C07", "Sorbus"],
  3545. ["FD9FA2", "Sweet Pink"],
  3546. ["FDD5B1", "Light Apricot"],
  3547. ["FDD7E4", "Pig Pink"],
  3548. ["FDE1DC", "Cinderella"],
  3549. ["FDE295", "Golden Glow"],
  3550. ["FDE910", "Lemon"],
  3551. ["FDF5E6", "Old Lace"],
  3552. ["FDF6D3", "Half Colonial White"],
  3553. ["FDF7AD", "Drover"],
  3554. ["FDFEB8", "Pale Prim"],
  3555. ["FDFFD5", "Cumulus"],
  3556. ["FE28A2", "Persian Rose"],
  3557. ["FE4C40", "Sunset Orange"],
  3558. ["FE6F5E", "Bittersweet"],
  3559. ["FE9D04", "California"],
  3560. ["FEA904", "Yellow Sea"],
  3561. ["FEBAAD", "Melon"],
  3562. ["FED33C", "Bright Sun"],
  3563. ["FED85D", "Dandelion"],
  3564. ["FEDB8D", "Salomie"],
  3565. ["FEE5AC", "Cape Honey"],
  3566. ["FEEBF3", "Remy"],
  3567. ["FEEFCE", "Oasis"],
  3568. ["FEF0EC", "Bridesmaid"],
  3569. ["FEF2C7", "Beeswax"],
  3570. ["FEF3D8", "Bleach White"],
  3571. ["FEF4CC", "Pipi"],
  3572. ["FEF4DB", "Half Spanish White"],
  3573. ["FEF4F8", "Wisp Pink"],
  3574. ["FEF5F1", "Provincial Pink"],
  3575. ["FEF7DE", "Half Dutch White"],
  3576. ["FEF8E2", "Solitaire"],
  3577. ["FEF8FF", "White Pointer"],
  3578. ["FEF9E3", "Off Yellow"],
  3579. ["FEFCED", "Orange White"],
  3580. ["FF0000", "Red"],
  3581. ["FF007F", "Rose"],
  3582. ["FF00CC", "Purple Pizzazz"],
  3583. ["FF00FF", "Magenta / Fuchsia"],
  3584. ["FF2400", "Scarlet"],
  3585. ["FF3399", "Wild Strawberry"],
  3586. ["FF33CC", "Razzle Dazzle Rose"],
  3587. ["FF355E", "Radical Red"],
  3588. ["FF3F34", "Red Orange"],
  3589. ["FF4040", "Coral Red"],
  3590. ["FF4D00", "Vermilion"],
  3591. ["FF4F00", "International Orange"],
  3592. ["FF6037", "Outrageous Orange"],
  3593. ["FF6600", "Blaze Orange"],
  3594. ["FF66FF", "Pink Flamingo"],
  3595. ["FF681F", "Orange"],
  3596. ["FF69B4", "Hot Pink"],
  3597. ["FF6B53", "Persimmon"],
  3598. ["FF6FFF", "Blush Pink"],
  3599. ["FF7034", "Burning Orange"],
  3600. ["FF7518", "Pumpkin"],
  3601. ["FF7D07", "Flamenco"],
  3602. ["FF7F00", "Flush Orange"],
  3603. ["FF7F50", "Coral"],
  3604. ["FF8C69", "Salmon"],
  3605. ["FF9000", "Pizazz"],
  3606. ["FF910F", "West Side"],
  3607. ["FF91A4", "Pink Salmon"],
  3608. ["FF9933", "Neon Carrot"],
  3609. ["FF9966", "Atomic Tangerine"],
  3610. ["FF9980", "Vivid Tangerine"],
  3611. ["FF9E2C", "Sunshade"],
  3612. ["FFA000", "Orange Peel"],
  3613. ["FFA194", "Mona Lisa"],
  3614. ["FFA500", "Web Orange"],
  3615. ["FFA6C9", "Carnation Pink"],
  3616. ["FFAB81", "Hit Pink"],
  3617. ["FFAE42", "Yellow Orange"],
  3618. ["FFB0AC", "Cornflower Lilac"],
  3619. ["FFB1B3", "Sundown"],
  3620. ["FFB31F", "My Sin"],
  3621. ["FFB555", "Texas Rose"],
  3622. ["FFB7D5", "Cotton Candy"],
  3623. ["FFB97B", "Macaroni and Cheese"],
  3624. ["FFBA00", "Selective Yellow"],
  3625. ["FFBD5F", "Koromiko"],
  3626. ["FFBF00", "Amber"],
  3627. ["FFC0A8", "Wax Flower"],
  3628. ["FFC0CB", "Pink"],
  3629. ["FFC3C0", "Your Pink"],
  3630. ["FFC901", "Supernova"],
  3631. ["FFCBA4", "Flesh"],
  3632. ["FFCC33", "Sunglow"],
  3633. ["FFCC5C", "Golden Tainoi"],
  3634. ["FFCC99", "Peach Orange"],
  3635. ["FFCD8C", "Chardonnay"],
  3636. ["FFD1DC", "Pastel Pink"],
  3637. ["FFD2B7", "Romantic"],
  3638. ["FFD38C", "Grandis"],
  3639. ["FFD700", "Gold"],
  3640. ["FFD800", "School bus Yellow"],
  3641. ["FFD8D9", "Cosmos"],
  3642. ["FFDB58", "Mustard"],
  3643. ["FFDCD6", "Peach Schnapps"],
  3644. ["FFDDAF", "Caramel"],
  3645. ["FFDDCD", "Tuft Bush"],
  3646. ["FFDDCF", "Watusi"],
  3647. ["FFDDF4", "Pink Lace"],
  3648. ["FFDEAD", "Navajo White"],
  3649. ["FFDEB3", "Frangipani"],
  3650. ["FFE1DF", "Pippin"],
  3651. ["FFE1F2", "Pale Rose"],
  3652. ["FFE2C5", "Negroni"],
  3653. ["FFE5A0", "Cream Brulee"],
  3654. ["FFE5B4", "Peach"],
  3655. ["FFE6C7", "Tequila"],
  3656. ["FFE772", "Kournikova"],
  3657. ["FFEAC8", "Sandy Beach"],
  3658. ["FFEAD4", "Karry"],
  3659. ["FFEC13", "Broom"],
  3660. ["FFEDBC", "Colonial White"],
  3661. ["FFEED8", "Derby"],
  3662. ["FFEFA1", "Vis Vis"],
  3663. ["FFEFC1", "Egg White"],
  3664. ["FFEFD5", "Papaya Whip"],
  3665. ["FFEFEC", "Fair Pink"],
  3666. ["FFF0DB", "Peach Cream"],
  3667. ["FFF0F5", "Lavender blush"],
  3668. ["FFF14F", "Gorse"],
  3669. ["FFF1B5", "Buttermilk"],
  3670. ["FFF1D8", "Pink Lady"],
  3671. ["FFF1EE", "Forget Me Not"],
  3672. ["FFF1F9", "Tutu"],
  3673. ["FFF39D", "Picasso"],
  3674. ["FFF3F1", "Chardon"],
  3675. ["FFF46E", "Paris Daisy"],
  3676. ["FFF4CE", "Barley White"],
  3677. ["FFF4DD", "Egg Sour"],
  3678. ["FFF4E0", "Sazerac"],
  3679. ["FFF4E8", "Serenade"],
  3680. ["FFF4F3", "Chablis"],
  3681. ["FFF5EE", "Seashell Peach"],
  3682. ["FFF5F3", "Sauvignon"],
  3683. ["FFF6D4", "Milk Punch"],
  3684. ["FFF6DF", "Varden"],
  3685. ["FFF6F5", "Rose White"],
  3686. ["FFF8D1", "Baja White"],
  3687. ["FFF9E2", "Gin Fizz"],
  3688. ["FFF9E6", "Early Dawn"],
  3689. ["FFFACD", "Lemon Chiffon"],
  3690. ["FFFAF4", "Bridal Heath"],
  3691. ["FFFBDC", "Scotch Mist"],
  3692. ["FFFBF9", "Soapstone"],
  3693. ["FFFC99", "Witch Haze"],
  3694. ["FFFCEA", "Buttery White"],
  3695. ["FFFCEE", "Island Spice"],
  3696. ["FFFDD0", "Cream"],
  3697. ["FFFDE6", "Chilean Heath"],
  3698. ["FFFDE8", "Travertine"],
  3699. ["FFFDF3", "Orchid White"],
  3700. ["FFFDF4", "Quarter Pearl Lusta"],
  3701. ["FFFEE1", "Half and Half"],
  3702. ["FFFEEC", "Apricot White"],
  3703. ["FFFEF0", "Rice Cake"],
  3704. ["FFFEF6", "Black White"],
  3705. ["FFFEFD", "Romance"],
  3706. ["FFFF00", "Yellow"],
  3707. ["FFFF66", "Laser Lemon"],
  3708. ["FFFF99", "Pale Canary"],
  3709. ["FFFFB4", "Portafino"],
  3710. ["FFFFF0", "Ivory"],
  3711. ["FFFFFF", "White"]
  3712. ]
  3713. }
  3714. ntc.init();