drawio/war/plugins/voice.js
Gaudenz Alder 7926da7222 6.3.4 release
Former-commit-id: 4ba28cd89f
2017-03-15 08:57:48 +01:00

4069 lines
116 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/**
* Copyright (c) 2006-2015, JGraph Ltd
* Copyright (c) 2006-2015, Gaudenz Alder
*/
/**
* Voice plugin for draw.io
*
* Documentation:
*
* https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin
*
* TODO: Use grammer https://msdn.microsoft.com/en-us/library/ee800145.aspx
*/
Draw.loadPlugin(function(ui) {
var micImage = '';
// Speech recognition never supported without synthesis
if (!('speechSynthesis' in window))
{
ui.showError('Error', 'Speech output not supported in this browser.', 'OK');
return;
}
else
{
// Triggers loading of voices
speechSynthesis.getVoices();
}
// Do no use in chromeless mode
if (ui.editor.chromeless || ui.menubar == null)
{
return;
}
// Mic PNG image
var outputImage = '';
// True if we're on ChromOs
var chromeOs = /\bCrOS\b/.test(navigator.userAgent);
// Maximum length of message to speak
var maxMessageLength = 1000;
// Maximum length of the label before the cell
// is called by its shapename
var maxLabelLength = 15;
// Maximum length of output queue.
var maxQueueLength = 3;
// Specifies if speech output is enabled.
var speechOutputEnabled = true;
// Specifies if speech output is enabled.
var speechInputEnabled = true;
// Last message is never repeated
var lastMessage = null;
// Timestamp of last message
var lastMessageTimestamp = null;
// Sets global recognition language
var lang = 'en-US';
// Caches action names
var actions = null;
var actionList = null;
// Caches shape names
var shapeList = null;
// Last inserted cell
var lastInserted = null;
// Current voice
var currentVoice = 10;
// Current recognition thread
var recognizing = null;
// Adds menu
mxResources.parse('voiceType=Voice Type');
mxResources.parse('speechOutput=Speech Output');
mxResources.parse('speechListen=Listen');
mxResources.parse('speechInstalled=Start with draw.io');
mxResources.parse('speechListenContinuous=Start/Stop Listen');
mxResources.parse('speechHint=Hint');
mxResources.parse('speechHelp=Help');
mxResources.parse('speechQuit=Quit');
// Installs footer click handler
function getOrCreateVoiceButton(ui)
{
if (ui.voiceButton == null)
{
ui.voiceButton = document.createElement('div');
ui.voiceButton.className = 'geBtn';
ui.voiceButton.style.width = '140px';
ui.voiceButton.style.minWidth = '140px';
ui.voiceButton.style.textOverflow = 'ellipsis';
ui.voiceButton.style.overflowX = 'hidden';
ui.voiceButton.style.fontWeight = 'bold';
ui.voiceButton.style.textAlign = 'center';
ui.voiceButton.style.display = 'inline-block';
ui.voiceButton.style.padding = '0 10px 0 10px';
ui.voiceButton.style.marginTop = '-4px';
ui.voiceButton.style.height = '28px';
ui.voiceButton.style.lineHeight = '28px';
ui.voiceButton.style.color = '#235695';
if (ui.buttonContainer.firstChild != null)
{
ui.buttonContainer.insertBefore(ui.voiceButton, ui.buttonContainer.firstChild);
}
else
{
ui.buttonContainer.appendChild(ui.voiceButton);
}
}
return ui.voiceButton;
};
var td = getOrCreateVoiceButton(ui);
if (td != null)
{
// Installs listener for start/stop listen
mxEvent.addListener(td, 'click', function(evt)
{
if (speechSynthesis.speaking)
{
speechSynthesis.cancel();
}
App.listen(true);
});
}
function setPluginInstalled(value)
{
if (mxSettings != null)
{
var plugins = mxSettings.getPlugins();
var installed = mxUtils.indexOf(plugins, '/plugins/voice.js') >= 0;
if (value != installed)
{
if (installed)
{
mxUtils.remove('/plugins/voice.js', plugins);
}
else
{
plugins.push('/plugins/voice.js');
}
mxSettings.setPlugins(plugins);
mxSettings.save();
}
}
};
// Shows initial status message if only output is enable
if (!('webkitSpeechRecognition' in window))
{
if (td != null)
{
td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + outputImage + '"/> Ready';
td.style.color = '#235695';
}
}
function updateStatusMessage()
{
if ('webkitSpeechRecognition' in window)
{
if (td != null)
{
if (recognizing != null)
{
td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + micImage + '"/> Listening...';
td.setAttribute('title', 'Click to Stop (Ctrl+O)');
td.style.color = 'darkGray';
}
else
{
td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + micImage + '"/> Click to Speak';
td.setAttribute('title', 'Click to Speak (Ctrl+O)');
td.style.color = '#235695';
}
}
}
};
updateStatusMessage();
var action = ui.actions.addAction('speechOutput', function()
{
speechOutputEnabled = !speechOutputEnabled;
}, null, null, 'Ctrl/AltGr+Shift+Esc');
action.setToggleAction(true);
action.setSelectedCallback(function() { return speechOutputEnabled; });
var action = ui.actions.addAction('speechInstalled', function()
{
setPluginInstalled();
});
action.setToggleAction(true);
action.setSelectedCallback(function() { return mxUtils.indexOf(mxSettings.getPlugins(), '/plugins/voice.js') >= 0; });
ui.actions.addAction('speechListen', function()
{
App.listen();
}, null, null, 'Ctrl/AltGr+Esc');
ui.actions.addAction('speechListenContinuous', function()
{
App.listen(true);
}, null, null, 'Ctrl+O');
ui.actions.addAction('speechHint', function()
{
App.sayHint();
}, null, null, 'Shift+Esc');
ui.actions.addAction('speechHelp', function()
{
window.open('https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin');
});
// Hijacks the settings for storing current voice
if (mxSettings != null)
{
var tmp = mxSettings.settings.voice;
if (tmp != null)
{
currentVoice = parseInt(tmp);
}
}
ui.menus.put('voiceType', new Menu(mxUtils.bind(this, function(menu, parent)
{
var voices = speechSynthesis.getVoices();
if (voices.length == 0)
{
menu.addItem('Loading...', null, function() {}, parent, null, false);
}
else
{
for (var i = 0; i < voices.length; i++)
{
(function(index)
{
var item = menu.addItem(voices[index].name + ' (' + voices[i].lang + ')', null, function()
{
currentVoice = index;
App.say('hello');
if (mxSettings != null)
{
mxSettings.settings.voice = currentVoice;
mxSettings.save();
}
}, parent);
if (index == currentVoice)
{
menu.addCheckmark(item, Editor.checkmarkImage);
}
})(i);
}
if (!mxClient.IS_QUIRKS)
{
parent.div.style.overflowX = 'hidden';
parent.div.style.overflowY = 'auto';
parent.div.style.maxHeight = '100%';
// Workaround for document scrollbars with 100% max height in Chrome
parent.div.style.marginBottom = '-20px';
}
}
})));
ui.actions.addAction('speechQuit', function()
{
// Hides UI
speechOutputEnabled = false;
menu.style.display = 'none';
td.style.display = 'none';
});
var menu = ui.menubar.addMenu('Voice', function(menu, parent)
{
ui.menus.addSubmenu('voiceType', menu, parent);
ui.menus.addMenuItems(menu, ['-', 'speechOutput', 'speechHint', '-', 'speechListen',
'speechListenContinuous', '-', 'speechInstalled',
'speechHelp', '-', 'speechQuit']);
});
// Inserts voice menu before help menu
var menu = menu.parentNode.insertBefore(menu, menu.previousSibling.previousSibling.previousSibling);
function insertShape(shape, done)
{
var searchTerm = mxUtils.trim(shape);
ui.sidebar.searchEntries(searchTerm, 1, 0, function(results, len, more)
{
if (results.length > 0)
{
var elt = results[0]();
// Click is blocked, must use mousedown/-up sequence
// LATER: Use touchstart or pointerEvents depending on system
dispatchEvent(elt, mouseEvent('mousedown', 1, 50, 1, 50));
dispatchEvent(document.body, mouseEvent('mouseup', 1, 50, 1, 50));
}
else
{
App.say('{1} not found', [searchTerm]);
}
if (done != null)
{
done();
}
});
};
// http://stackoverflow.com/questions/11919065/sort-an-array-by-the-levenshtein-distance-with-best-performance-in-javascript
//http://www.merriampark.com/ld.htm, http://www.mgilleland.com/ld/ldjavascript.htm, DamerauLevenshtein distance (Wikipedia)
var levenshteinDist = function(s, t) {
var d = []; //2d matrix
// Step 1
var n = s.length;
var m = t.length;
if (n == 0) return m;
if (m == 0) return n;
//Create an array of arrays in javascript (a descending loop is quicker)
for (var i = n; i >= 0; i--) d[i] = [];
// Step 2
for (var i = n; i >= 0; i--) d[i][0] = i;
for (var j = m; j >= 0; j--) d[0][j] = j;
// Step 3
for (var i = 1; i <= n; i++) {
var s_i = s.charAt(i - 1);
// Step 4
for (var j = 1; j <= m; j++) {
//Check the jagged ld total so far
if (i == j && d[i][j] > 4) return n;
var t_j = t.charAt(j - 1);
var cost = (s_i == t_j) ? 0 : 1; // Step 5
//Calculate the minimum
var mi = d[i - 1][j] + 1;
var b = d[i][j - 1] + 1;
var c = d[i - 1][j - 1] + cost;
if (b < mi) mi = b;
if (c < mi) mi = c;
d[i][j] = mi; // Step 6
//Damerau transposition
if (i > 1 && j > 1 && s_i == t.charAt(j - 2) && s.charAt(i - 2) == t_j) {
d[i][j] = Math.min(d[i][j], d[i - 2][j - 2] + cost);
}
}
}
// Step 7
return d[n][m];
}
function naiveHammingDistance(str1, str2) {
var dist = 0;
str1 = str1.toLowerCase();
str2 = str2.toLowerCase();
for(var i = 0; i < str1.length; i++)
{
if (str2[i] && str2[i] !== str1[i])
{
dist += Math.abs(str1.charCodeAt(i) - str2.charCodeAt(i)) + Math.abs(str2.indexOf( str1[i] )) * 2;
}
else if (!str2[i])
{
// If there's no letter in the comparing string
dist += dist;
}
}
return dist;
};
function getBestWord(str1, words, useLevenshteinDist)
{
useLevenshteinDist = (useLevenshteinDist != null) ? useLevenshteinDist : true;
var bestWord = words[0];
var minDist = ((useLevenshteinDist) ? levenshteinDist(str1, bestWord) :
naiveHammingDistance(str1, bestWord));
for (var i = 1; i < words.length; i++)
{
var tmp = ((useLevenshteinDist) ? levenshteinDist(str1, words[i]) :
((str1 == words[i]) ? 0 : naiveHammingDistance(str1, words[i])));
if (tmp < minDist || (tmp == minDist &&
str1.length > bestWord.length &&
bestWord.length < words[i].length))
{
bestWord = words[i];
minDist = tmp;
}
if (bestWord == str1)
{
break;
}
}
return bestWord;
}
function mouseEvent(type, sx, sy, cx, cy, shift)
{
var evt;
var e = {
bubbles: true,
cancelable: (type != "mousemove"),
view: window,
detail: 0,
screenX: sx,
screenY: sy,
clientX: cx,
clientY: cy,
ctrlKey: false,
altKey: false,
shiftKey: (shift != null) ? shift : false,
metaKey: false,
button: 0,
relatedTarget: undefined
};
if (typeof( document.createEvent ) == "function")
{
evt = document.createEvent("MouseEvents");
evt.initMouseEvent(type,
e.bubbles, e.cancelable, e.view, e.detail,
e.screenX, e.screenY, e.clientX, e.clientY,
e.ctrlKey, e.altKey, e.shiftKey, e.metaKey,
e.button, document.body.parentNode);
}
else if (document.createEventObject)
{
evt = document.createEventObject();
for (prop in e)
{
evt[prop] = e[prop];
}
evt.button = { 0:1, 1:4, 2:2 }[evt.button] || evt.button;
}
return evt;
};
function dispatchEvent (el, evt)
{
if (el.dispatchEvent)
{
el.dispatchEvent(evt);
}
else if (el.fireEvent)
{
el.fireEvent('on' + type, evt);
}
return evt;
};
var keyHandlerEscape = ui.keyHandler.escape;
ui.keyHandler.escape = function(evt)
{
if ((!mxClient.IS_MAC && mxEvent.isAltDown(evt)) ||
((mxClient.IS_MAC || chromeOs) && mxEvent.isControlDown(evt)))
{
if (speechOutputEnabled)
{
App.say('Speech output disabled');
}
speechOutputEnabled = !speechOutputEnabled;
if (speechOutputEnabled)
{
App.say('Speech output enabled');
}
mxEvent.consume(evt);
}
else if (mxEvent.isShiftDown(evt))
{
App.sayHint();
mxEvent.consume(evt);
}
else
{
keyHandlerEscape.apply(this, arguments);
}
};
ui.keyHandler.bindAction(32, true, 'speechListen'); // Ctrl+SPACE
ui.keyHandler.bindAction(79, true, 'speechListenContinuous'); // Ctrl+O
/**
* Plays a beep.
*/
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=');
var beep2 = new Audio('data:audio/wav;base64,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');
App.beep = function(wav)
{
wav = (wav != null) ? wav : beep;
wav.play();
};
// Thread to reset the label
var resetStatus = null;
/**
*
* Static method for speech output.
*/
App.say = function(message, params)
{
if ('speechSynthesis' in window && message != null)
{
var text = mxResources.replacePlaceholders(message, params || []);
lastMessageTimestamp = null;
lastMessage = text;
if (text != null && text.length > 0)
{
if (td != null)
{
var tmp = text;
// Capitalize string
if (tmp != null && tmp.length > 1)
{
tmp = tmp.charAt(0).toUpperCase() + tmp.slice(1);
}
td.innerHTML = ((speechOutputEnabled) ? '<img style="margin-right:4px;" align="absmiddle" border="0" src="' +
outputImage + '"/>' : '') + ' ' + tmp;
td.style.color = '#235695';
if (resetStatus != null)
{
window.clearTimeout(resetStatus);
resetStatus = null;
}
resetStatus = window.setTimeout(function()
{
updateStatusMessage();
}, (recognizing != null) ? 1000 : 3000);
}
// Workaround for talking too much
if (speechOutputEnabled && (!speechSynthesis.speaking || !speechSynthesis.pending))
{
if (text.length < maxMessageLength)
{
var msg = new SpeechSynthesisUtterance();
// Picks random voice with same main locale
var voices = speechSynthesis.getVoices();
// Say "again" for same last message except more than 10 secs ago or shorter than again
if (lastMessageTimestamp != null && text == lastMessage && text != null &&
text.length > 5 && lastMessage != 'again')
{
if (lastMessageTimestamp != null &&
new Date().getTime() - lastMessageTimestamp.getTime() < 10000)
{
text = 'repeat';
}
}
msg.voice = voices[currentVoice];
msg.voiceURI = 'native';
//msg.lang = lang;
msg.text = text;
console.log('App.say speak:', msg.text);
speechSynthesis.speak(msg);
lastMessageTimestamp = new Date();
}
else
{
console.log('App.say ignored:', text);
}
}
else
{
console.log('App.say skipped:', message);
}
}
}
};
/**
* Static method for speech output.
*/
App.listen = function(continuous)
{
if ('webkitSpeechRecognition' in window && speechInputEnabled)
{
if (recognizing != null)
{
recognizing.stop();
}
else
{
var recognition = new webkitSpeechRecognition();
recognition.interimResults = true;
// TODO: Should use grammar instead of trying more alternatives
recognition.maxAlternatives = 5;
recognition.lang = lang;
if (continuous != null)
{
recognition.continuous = continuous;
}
recognition.onstart = function(event)
{
updateStatusMessage();
App.beep();
};
recognition.onresult = function(event)
{
for (var i = event.resultIndex; i < event.results.length; ++i)
{
if (td != null)
{
td.innerHTML = '<img style="margin-right:4px;" align="absmiddle" border="0" src="' + micImage +
'"/> ' + event.results[i][0].transcript;
td.style.color = (event.results[i].isFinal) ? '#235695' : 'darkGray';
}
if (event.results[i].isFinal)
{
var ok = false;
for (var j = 0; j < event.results[i].length; j++)
{
if (App.executeVoiceCommand(event.results[i][j].transcript, recognition))
{
ok = true;
break;
}
}
if (!ok)
{
App.say('{1} not found', [event.results[i][0].transcript]);
}
if (td != null)
{
if (resetStatus != null)
{
window.clearTimeout(resetStatus);
resetStatus = null;
}
resetStatus = window.setTimeout(function()
{
updateStatusMessage();
}, 1000);
}
}
}
};
recognition.onend = function(event)
{
// Overrides footer
recognizing = null;
updateStatusMessage();
App.beep(beep2);
};
recognition.start();
recognizing = recognition;
}
}
else
{
speechOutputEnabled = !speechOutputEnabled;
lastMessageTimestamp = null;
App.say(lastMessage || 'Ready');
lastMessageTimestamp = null;
}
};
/**
* Executes the given voice command.
*/
App.executeVoiceCommand = function(command, recognition)
{
console.log('App.execute:', mxUtils.trim(command));
var tokens = mxUtils.trim(command).split(' ');
if (tokens.length > 0 && graph.isEnabled())
{
// Ask for Mic permissions
// FIXME: Dialog seems to be hidden until tab is changed
function resolveStylename(token, styles)
{
var tmp = token.toLowerCase().replace(/ /g, '');
var style = null;
for (var i = 0; i < styles.length; i++)
{
if (styles[i].toLowerCase() == tmp)
{
style = styles[i];
break;
}
}
return style;
};
// Main command
tokens[0] = tokens[0].toLowerCase();
// TODO: Use hamming distance for best match command but include all possible actions
// which might be too slow
// console.log('connect', naiveHammingDistance(tokens[0], 'connect'), naiveHammingDistance('disable', 'connect'), naiveHammingDistance('change', 'connect'));
if (graph.isEditing())
{
if (tokens.length == 1 && tokens[0] == 'apply')
{
graph.stopEditing();
}
else if (tokens.length == 1 && tokens[0] == 'undo')
{
document.execCommand('undo', false, null);
}
else if (tokens.length == 1 && tokens[0] == 'redo')
{
document.execCommand('redo', false, null);
}
else
{
document.execCommand('insertHTML', false, command);
}
return true;
}
else if (tokens[0] == 'edit' && tokens[1] == 'text')
{
var cells = graph.getSelectionCells();
if (cells.length == 1)
{
graph.startEditingAtCell(cells[0]);
}
return true;
}
else if (tokens[0] == 'hello' || tokens[0] == 'hi')
{
App.say('Hello! Try "Help", "Help Topic" or "Quick Start".');
return true;
}
else if (tokens[0] == 'help')
{
var wnd = window.open('https://support.draw.io/questions/9666655/how-to-use-the-voice-plugin');
if (wnd == null)
{
App.say('Popup blocked');
}
else if (tokens.length > 1)
{
// Just used to check if popup windows are allowed
wnd.close();
var searchTerm = mxUtils.trim(command.substring(tokens[0].length));
if (searchTerm != null && searchTerm.length > 0)
{
var form = document.createElement('div');
form.style.display = 'inline';
form.innerHTML = ':<form style="display:inline;margin-left:8px;" id="rw_search_form"' +
'target="_blank" method="get" action="https://support.draw.io/dosearchsite.action">' +
'<input id="rw_search_query" type="text" name="queryString" size="25"></form>';
var realForm = form.getElementsByTagName('form')[0]
var input = form.getElementsByTagName('input')[0];
input.setAttribute('value', searchTerm);
realForm.submit();
App.say(command);
}
}
else
{
App.say('help');
}
return true;
}
else if ((tokens[0] == 'info' && tokens.length == 1) || mxUtils.trim(command).toLowerCase() == 'what\'s this')
{
App.sayHint();
return true;
}
else if (tokens[0] == 'install' && tokens.length == 1)
{
setPluginInstalled(true);
App.say('Installed');
return true;
}
else if (tokens[0] == 'uninstall' && tokens.length == 1)
{
setPluginInstalled(false);
App.say('Uninstalled');
return true;
}
else if (tokens[0] == 'quick' && tokens.length > 0 && tokens[1].toLowerCase() == 'start')
{
var wnd = window.open('https://youtu.be/8OaMWa4R1SE?t=1');
if (wnd == null)
{
App.say('Popup blocked');
}
return true;
}
else if (tokens[0] == 'search' && tokens.length > 1)
{
var searchToken = tokens.slice(1, tokens.length).join(' ').toLowerCase();
var wnd = window.open('https://www.google.ch/search?q=' + encodeURIComponent(searchToken) + '&tbm=isch', 'voicePluginSearchResult');
if (wnd == null)
{
App.say('Popup blocked');
}
return true;
}
else if (tokens[0] == 'shrink')
{
var cell = graph.getSelectionCell();
var geo = graph.getCellGeometry(cell);
if (graph.getModel().isVertex(cell) && geo != null)
{
geo = geo.clone();
if (tokens.length == 1 || tokens[1] == 'height')
{
geo.height = graph.snap(Math.round(geo.height * 0.5));
}
if (tokens.length == 1 || tokens[1] != 'height')
{
geo.width = graph.snap(Math.round(geo.width * 0.5));
}
if (geo.width > graph.tolerance && geo.height > graph.tolerance)
{
graph.getModel().setGeometry(cell, geo);
App.say('Resized');
}
else
{
App.say('Too small');
}
}
else
{
App.say('No cell to resize');
}
return true;
}
else if (tokens[0] == 'double')
{
var cell = graph.getSelectionCell();
var geo = graph.getCellGeometry(cell);
if (graph.getModel().isVertex(cell) && geo != null)
{
geo = geo.clone();
if (tokens.length == 1 || tokens[1] == 'height')
{
geo.height *= 2;
}
if (tokens.length == 1 || tokens[1] != 'height')
{
geo.width *= 2;
}
if (geo.width > graph.tolerance && geo.height > graph.tolerance)
{
graph.getModel().setGeometry(cell, geo);
App.say('Resized');
}
else
{
App.say('Too small');
}
}
else
{
App.say('No cell to resize');
}
return true;
}
else if (tokens[0] == 'width' || tokens[0] == 'with' || tokens[0] == 'height')
{
// Try numeric stylename if last token is numeric
var lastToken = tokens[tokens.length - 1].toLowerCase();
// Fixes some special cases for recoginition of short phrases
// like "stroke width 2" where it tries to be clever
if (lastToken == 'tool' || lastToken == 'tune' || lastToken == 'tomb' || lastToken == 'tube')
{
lastToken = '2';
}
else if (lastToken == 'd3')
{
lastToken = '3';
}
else if (lastToken == 'v')
{
lastToken = '5';
}
else
{
// Converts numeric words to numbers (system only seems to return written numbers for <= 4)
var numbers = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine'];
var tmpIndex = mxUtils.indexOf(numbers, lastToken);
if (tmpIndex >= 0)
{
lastToken = tmpIndex;
}
}
var lastValue = parseFloat(tokens[1].toLowerCase());
if (tokens.length >= 2 && !isNaN(lastValue))
{
var cell = graph.getSelectionCell();
var geo = graph.getCellGeometry(cell);
if (graph.getModel().isVertex(cell) && geo != null)
{
geo = geo.clone();
if (tokens[0] == 'height')
{
geo.height = lastValue;
}
else
{
geo.width = lastValue;
}
graph.getModel().setGeometry(cell, geo);
App.say('Resized');
}
else
{
App.say('No cell to resize');
}
}
return true;
}
else if (tokens[0] == 'insert' && tokens.length > 1)
{
// Fixes some common mistakes
if (tokens[1].toLowerCase() == 'edits')
{
tokens[1] = 'ellipse';
}
var searchTerm = mxUtils.trim(tokens.slice(1, tokens.length).join(' '));
var current = graph.getSelectionCell();
// Clears selection to disable built-in connecting
graph.clearSelection();
insertShape(searchTerm, function()
{
var cell = graph.getSelectionCell();
// Connects dangling edge of previously selected edge and moves cell
if (graph.model.isVertex(cell) && graph.model.isEdge(current) &&
(graph.model.getTerminal(current, true) == null ||
graph.model.getTerminal(current, false) == null))
{
var edgeState = graph.view.getState(current);
var vertexState = graph.view.getState(cell);
if (vertexState != null && edgeState != null && edgeState.absolutePoints != null &&
edgeState.absolutePoints.length > 1)
{
var source = graph.model.getTerminal(current, true) == null;
var pts = edgeState.absolutePoints;
var pt = pts[(source) ? 0 : pts.length - 1];
var loc = new mxPoint(vertexState.getCenterX(), vertexState.getCenterY());
if (loc != null && edgeState != null)
{
var s = graph.view.scale;
var dx = pt.x - loc.x;
var dy = pt.y - loc.y;
// TODO: Should add insert to transaction but need absolute position
graph.model.beginUpdate();
try
{
graph.moveCells(graph.getSelectionCells(), dx / s, dy / s);
graph.model.setTerminal(current, cell, source);
}
finally
{
graph.model.endUpdate();
}
}
}
}
});
return true;
}
else if (tokens[0] == 'connect' || tokens[0] == 'kinect' || (tokens[0] == 'clone' && tokens.length > 1))
{
var cell = graph.getSelectionCell();
if (graph.getModel().isVertex(cell))
{
// Uses east direction if token not understood
var direction = mxConstants.DIRECTION_EAST;
if (tokens.length > 1)
{
tokens[1] = tokens[1].toLowerCase();
// Guessing direction based on minimum hamming distance for given set
var guess = getBestWord(tokens[1], ['up', 'left', 'down', 'right', 'north', 'south', 'east', 'west']);
if (guess == 'up' || guess == mxConstants.DIRECTION_NORTH)
{
direction = mxConstants.DIRECTION_NORTH;
}
else if (guess == 'left' || guess == mxConstants.DIRECTION_WEST)
{
direction = mxConstants.DIRECTION_WEST;
}
else if (guess == 'down' || guess == mxConstants.DIRECTION_SOUTH)
{
direction = mxConstants.DIRECTION_SOUTH;
}
}
var length = graph.defaultEdgeLength;
var cloneSource = tokens[0] == 'clone';
var evt = mouseEvent('click', 1, 50, 1, 50, !cloneSource);
var cells = graph.connectVertex(cell, direction, length, evt);
graph.selectCellsForConnectVertex(cells, evt, ui.hoverIcons);
}
return true;
}
// Fixes drive and ride common mistakes for right
else if (mxUtils.indexOf(['and', 'drive', 'ride', 'move', 'up', 'left', 'down', 'right', 'north', 'south', 'east', 'west', 'downright'], tokens[0]) >= 0)
{
var cell = graph.getSelectionCell();
if (!graph.isSelectionEmpty())
{
// Uses east direction if token not understood
var dx = 0;
var dy = 0;
for (var i = 0; i < tokens.length; i++)
{
tokens[i] = tokens[i].toLowerCase();
var direction = null;
// Downright is a single word needs special handling
if (tokens[i].toLowerCase() == 'downright')
{
dx += graph.defaultEdgeLength;
dy += graph.defaultEdgeLength;
}
else
{
var guess = null;
// Guessing direction based on minimum hamming distance for given set
// Handle some common cases
if (tokens[i] == 'drive' || tokens[i] == 'ride')
{
guess = 'right';
}
else
{
guess = getBestWord(tokens[i], ['move', 'and', 'up', 'left', 'right', 'north', 'south', 'east', 'west', 'down', 'downright']);
}
if (guess == 'up' || guess == mxConstants.DIRECTION_NORTH)
{
direction = mxConstants.DIRECTION_NORTH;
}
else if (guess == 'left' || guess == mxConstants.DIRECTION_WEST)
{
direction = mxConstants.DIRECTION_WEST;
}
else if (guess == 'down' || guess == mxConstants.DIRECTION_SOUTH)
{
direction = mxConstants.DIRECTION_SOUTH;
}
else if (guess == 'right' || guess == mxConstants.DIRECTION_EAST)
{
direction = mxConstants.DIRECTION_EAST;
}
if (direction != null)
{
if (direction == mxConstants.DIRECTION_NORTH)
{
dy += -graph.defaultEdgeLength;
}
else if (direction == mxConstants.DIRECTION_WEST)
{
dx += -graph.defaultEdgeLength;
}
else if (direction == mxConstants.DIRECTION_SOUTH)
{
dy += graph.defaultEdgeLength;
}
else if (direction == mxConstants.DIRECTION_EAST)
{
dx += graph.defaultEdgeLength;
}
}
}
}
if (dx != 0 || dy != null)
{
graph.moveCells(graph.getSelectionCells(), dx, dy);
App.say('Moved');
}
return true;
}
}
else if (tokens[0] == 'text')
{
var cells = graph.getSelectionCells();
if (cells.length == 0 && graph.model.contains(lastInserted))
{
cells = [lastInserted];
}
if (cells.length == 0)
{
App.say('No cell for text');
}
else if (tokens[0].length >= 2)
{
var value = tokens.slice(1, tokens.length).join(' ');
if (value.length > 0)
{
// Capitalize string
if (value.length > 1)
{
value = value.charAt(0).toUpperCase() + value.slice(1);
}
graph.labelChanged(cells[0], value);
}
}
return true;
}
else if (tokens[0] == 'disconnect' && tokens.length == 1)
{
var cells = graph.getAllEdges(graph.getSelectionCells());
if (cells.length == 0)
{
App.say('No cell to disconnect');
}
else
{
graph.removeCells(cells);
}
return true;
}
else if (tokens[0] == 'deselect' && tokens.length == 1)
{
graph.clearSelection();
return true;
}
else if (tokens[0] === 'select' && (tokens.length == 1 || mxUtils.indexOf(
['vertices', 'edges', 'none', 'all'], tokens[1].toLowerCase()) < 0))
{
// Handles some other shortcuts
if (tokens.length == 1 || mxUtils.indexOf(['last'], tokens[1].toLowerCase()) >= 0)
{
if (graph.model.contains(lastInserted))
{
graph.setSelectionCell(lastInserted);
App.say('{1} selected', [graph.getWordForCell(lastInserted).replace(/([A-Z])/g, ' $1')]);
return true;
}
}
// Handles alternative cases of edges
else if (mxUtils.indexOf(['connection', 'connections', 'inches'], tokens[1].toLowerCase()) >= 0)
{
var edges = graph.getAllEdges(graph.getSelectionCells());
if (edges.length > 0)
{
graph.setSelectionCells(edges);
}
else
{
ui.actions.get('selectEdges').funct();
}
App.sayHint();
return true;
}
// Handles alternative version of vertices
else if (mxUtils.indexOf(['shapes'], tokens[1].toLowerCase()) >= 0)
{
ui.actions.get('selectVertices').funct();
App.sayHint();
return true;
}
else if (mxUtils.indexOf(['previous'], tokens[1].toLowerCase()) >= 0)
{
if (graph.isSelectionEmpty())
{
// Selects the first vertex
var parent = graph.getDefaultParent();
var childCount = graph.model.getChildCount(parent);
for (var i = childCount - 1; i >= 0; i--)
{
var child = graph.model.getChildAt(parent, i);
if (graph.model.isVertex(child))
{
graph.setSelectionCell(child);
App.say('{1} selected', [graph.getWordForCell(child).replace(/([A-Z])/g, ' $1')]);
return true;
}
}
}
else
{
var cell = graph.getSelectionCell();
var model = graph.getModel();
var index = model.getParent(cell).getIndex(cell);
var childCount = model.getChildCount(model.getParent(cell));
if (index >= 0)
{
var next = model.getParent(cell).getChildAt(((index == 0) ? childCount : index) - 1);
if (next != null)
{
graph.setSelectionCell(next);
App.say('{1} selected', [graph.getWordForCell(next)]);
return true;
}
}
App.say('Previous not found');
}
}
else if (mxUtils.indexOf(['source', 'target'], tokens[1].toLowerCase()) >= 0)
{
var cell = graph.getSelectionCell();
if (graph.model.isEdge(cell))
{
var terminal = graph.model.getTerminal(cell, tokens[1].toLowerCase() == 'source');
if (terminal != null)
{
graph.setSelectionCell(terminal);
return true;
}
}
}
else if (mxUtils.indexOf(['next'], tokens[1].toLowerCase()) >= 0)
{
if (graph.isSelectionEmpty())
{
// Selects the first vertex
var parent = graph.getDefaultParent();
var childCount = graph.model.getChildCount(parent);
for (var i = 0; i < childCount; i++)
{
var child = graph.model.getChildAt(parent, i);
if (graph.model.isVertex(child))
{
graph.setSelectionCell(child);
App.say('{1} selected', [graph.getWordForCell(child).replace(/([A-Z])/g, ' $1')]);
return true;
}
}
}
else
{
var cell = graph.getSelectionCell();
var model = graph.getModel();
var index = model.getParent(cell).getIndex(cell);
var childCount = model.getChildCount(model.getParent(cell));
if (index < childCount)
{
var next = model.getParent(cell).getChildAt(((index == childCount - 1) ? 0 : index + 1));
if (next != null)
{
graph.setSelectionCell(next);
App.say('{1} selected', [graph.getWordForCell(next).replace(/([A-Z])/g, ' $1')]);
return true;
}
}
}
}
else
{
var states = graph.view.states.getValues();
var token = tokens[1].toLowerCase();
var searchToken = tokens.slice(1, tokens.length).join('').toLowerCase();
for (var i = states.length - 1; i > 0; i--)
{
// Simple matching for shape name or label
if (!graph.isCellSelected(states[i].cell))
{
// Tries label search first
var lab = graph.getLabel(states[i].cell);
if (lab != null && lab.length > 0)
{
lab = lab.toLowerCase().replace(/ /g, '');
// Some common mistakes
if (lab.charAt(lab.length - 1) == '2' && tokens[tokens.length - 1].toLowerCase() == 'to')
{
lab = lab.substring(0, lab.length - 1) + 'to';
}
var min = Math.min(searchToken.length, lab.length);
if (searchToken.substring(0, min) == lab.substring(0, min))
{
graph.setSelectionCell(states[i].cell);
App.say('{1} selected', [graph.getWordForCell(states[i].cell)]);
return true;
}
}
// Then tries shapename search
if (tokens.length == 2)
{
// Some common names
if (mxUtils.indexOf(['circle'], token) >= 0)
{
searchToken = 'ellipse';
}
else if (mxUtils.indexOf(['condition', 'decision'], token) >= 0)
{
searchToken = 'rhombus';
}
else if (mxUtils.indexOf(['preparation'], token) >= 0)
{
searchToken = 'hexagon';
}
else if (mxUtils.indexOf(['trapez'], token) >= 0)
{
searchToken = 'parallelogram';
}
}
var wrd = graph.getWordForCell(states[i].cell, true);
var min = Math.min(wrd.length, searchToken.length);
if (searchToken.substring(0, min) == wrd.substring(0, min))
{
graph.setSelectionCell(states[i].cell);
App.say('{1} selected', [graph.getWordForCell(states[i].cell)]);
return true;
}
}
}
}
}
else if (tokens[0] === 'remove')
{
var cells = graph.getSelectionCells();
if (cells.length == 0 && graph.model.contains(lastInserted))
{
cells = [lastInserted];
}
if (cells.length == 0)
{
App.say('No cells');
}
else if (tokens[1].toLowerCase() == 'text')
{
graph.labelChanged(cells[0], '');
}
else
{
// Remove style
var styleToken = mxUtils.trim(tokens.slice(1, tokens.length).join(' ').toLowerCase());
// Fixes common recognition errors
styleToken = styleToken.replace(/a line/g, 'align')
// Merges to single word
styleToken = styleToken.replace(/ /g, '')
// Fixes common speech errors that make no sense in the context
var keys = [mxConstants.STYLE_FILLCOLOR, mxConstants.STYLE_GRADIENTCOLOR,
mxConstants.STYLE_STROKECOLOR, mxConstants.STYLE_FONTCOLOR,
mxConstants.STYLE_LABEL_BACKGROUNDCOLOR, mxConstants.STYLE_LABEL_BORDERCOLOR,
mxConstants.STYLE_STROKEWIDTH, mxConstants.STYLE_FONTSIZE, mxConstants.STYLE_SPACING,
mxConstants.STYLE_SPACING_TOP, mxConstants.STYLE_SPACING_LEFT, mxConstants.STYLE_SPACING_RIGHT,
mxConstants.STYLE_SPACING_BOTTOM, mxConstants.STYLE_PERIMETER_SPACING,
mxConstants.STYLE_STARTSIZE, mxConstants.STYLE_ENDSIZE, mxConstants.STYLE_OPACITY,
mxConstants.STYLE_TEXT_OPACITY, mxConstants.STYLE_ROTATION,
mxConstants.STYLE_ALIGN, mxConstants.STYLE_VERTICAL_ALIGN, mxConstants.STYLE_FONTFAMILY,
mxConstants.STYLE_LABEL_POSITION, mxConstants.STYLE_VERTICAL_LABEL_POSITION,
mxConstants.STYLE_GRADIENT_DIRECTION];
// Guesses best word
var style = getBestWord(styleToken, keys);
if (style != null)
{
graph.setCellStyles(style, null, cells);
ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [null], 'cells', cells));
}
else
{
App.say('Unknown style {1}', [changeTokens[0]]);
}
}
return true;
}
else
{
var cells = graph.getSelectionCells();
if (cells.length == 0 && graph.model.contains(lastInserted))
{
cells = [lastInserted];
}
// Try numeric stylename if last token is numeric
var lastToken = tokens[tokens.length - 1].toLowerCase();
// Fixes some special cases for recoginition of short phrases
// like "stroke width 2" where it tries to be clever
if (lastToken == 'tool' || lastToken == 'tune' || lastToken == 'tomb' || lastToken == 'tube')
{
lastToken = '2';
}
else if (lastToken == 'd3')
{
lastToken = '3';
}
else if (lastToken == 'v')
{
lastToken = '5';
}
else
{
// Converts numeric words to numbers (system only seems to return written numbers for <= 4)
var numbers = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine'];
var tmpIndex = mxUtils.indexOf(numbers, lastToken);
if (tmpIndex >= 0)
{
lastToken = tmpIndex;
}
}
// If last token is numeric
var lastValue = parseFloat(lastToken);
if (tokens.length >= 2 && !isNaN(lastValue))
{
var styleToken = tokens.slice(0, tokens.length - 1).join('').toLowerCase();
// Numeric styles
var keys = [mxConstants.STYLE_STROKEWIDTH, mxConstants.STYLE_FONTSIZE, mxConstants.STYLE_SPACING,
mxConstants.STYLE_SPACING_TOP, mxConstants.STYLE_SPACING_LEFT, mxConstants.STYLE_SPACING_RIGHT,
mxConstants.STYLE_SPACING_BOTTOM, mxConstants.STYLE_PERIMETER_SPACING,
mxConstants.STYLE_STARTSIZE, mxConstants.STYLE_ENDSIZE, mxConstants.STYLE_OPACITY,
mxConstants.STYLE_TEXT_OPACITY, mxConstants.STYLE_ROTATION]
var style = null;
// Workaround for problem with "font size" is to say "size" ("font" is near "phone")
if (styleToken == 'size')
{
style = mxConstants.STYLE_FONTSIZE;
}
else
{
// Guesses best word
style = getBestWord(styleToken, keys);
}
if (style != null)
{
if (cells.length == 0)
{
App.say('No cell for {1}', [style]);
}
// Plausibility check
else if (lastValue <= 400)
{
graph.setCellStyles(style, lastValue, cells);
ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [lastValue], 'cells', cells));
}
else
{
App.say('{1} ignored', [lastValue]);
}
// Terminate
return true;
}
}
// If "color" appears in the command
if (tokens.length >= 2)
{
// Replaces GB spelling with US spelling for internal lookups
var colorCommand = mxUtils.trim(command.toLowerCase().replace(/colour/g, 'color'));
var colorIndex = colorCommand.indexOf('color');
// "Color" (word alone) matches to fill color
if (colorIndex >= 0)
{
// Text from first space after color* word
var colorToken = mxUtils.trim(colorCommand.substring(
colorCommand.indexOf(' ', colorIndex))).replace(/ /g, '');
var color = null;
// Checks for color code of token using ntc
for (var i = 0; i < ntc.names.length; i++)
{
if (ntc.names[i][1].toLowerCase().replace(/ /g, '') == colorToken)
{
color = '#' + ntc.names[i][0];
break;
}
}
// Color is known
if (color != null)
{
var styleToken = colorCommand.substring(0, colorIndex + 'color'.length).replace(/ /g, '');
keys = [mxConstants.STYLE_GRADIENTCOLOR,
mxConstants.STYLE_STROKECOLOR, mxConstants.STYLE_FONTCOLOR,
mxConstants.STYLE_LABEL_BACKGROUNDCOLOR, mxConstants.STYLE_LABEL_BORDERCOLOR,
mxConstants.STYLE_FILLCOLOR];
var style = null;
// Handles some special cases
if (styleToken == 'thecolor' || styleToken == 'color')
{
if (graph.model.isEdge(graph.getSelectionCell()))
{
style = mxConstants.STYLE_STROKECOLOR;
}
else
{
style = mxConstants.STYLE_FILLCOLOR;
}
}
else
{
// Guesses best word
style = getBestWord(styleToken, keys);
}
if (style != null)
{
if (cells.length == 0)
{
App.say('No cell for {1}', [style]);
}
else
{
graph.setCellStyles(style, color, cells);
ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [color], 'cells', cells));
}
// Terminate
return true;
}
}
}
}
// If the first token is a general stylename
if (tokens.length >= 2)
{
var keys = [mxConstants.STYLE_ALIGN, mxConstants.STYLE_VERTICAL_ALIGN,
mxConstants.STYLE_FONTFAMILY, mxConstants.STYLE_LABEL_POSITION,
mxConstants.STYLE_VERTICAL_LABEL_POSITION,
mxConstants.STYLE_GRADIENT_DIRECTION];
var styleToken = mxUtils.trim(tokens.slice(0, tokens.length - 1).join(' ').toLowerCase());
// Fixes common recognition errors
styleToken = styleToken.replace(/a line/g, 'align')
// Merges to single word
styleToken = styleToken.replace(/ /g, '')
var style = resolveStylename(styleToken, keys);
var value = mxUtils.trim(tokens[tokens.length - 1].toLowerCase());
if (style != null && value != null && value.length > 0)
{
if (cells.length == 0)
{
App.say('No cell for {1}', [style]);
}
else
{
graph.setCellStyles(style, value, cells);
ui.fireEvent(new mxEventObject('styleChanged', 'keys', [style], 'values', [value], 'cells', cells));
}
// Terminate
return true;
}
}
// Checks if the command is a shape name
var vertices = [];
for (var i = 0; i < cells.length; i++)
{
var cell = cells[i];
if (graph.model.isVertex(cell))
{
vertices.push(cell);
}
}
var shapenameToken = mxUtils.trim(tokens.join('')).toLowerCase();
// Searches for registered shape names
// LATER: Using search like insert requires access to
// the cells of the search result items
if (shapeList == null)
{
shapeList = [];
for (var tmp in mxCellRenderer.prototype.defaultShapes)
{
shapeList.push(tmp.toLowerCase());
}
}
// Only exact matches allowed here
// LATER: Support rounded style
var shape = null;
// Some common mappings
if (mxUtils.indexOf(['cirlce', 'event', 'start', 'end'], shapenameToken) >= 0)
{
shapenameToken = shape = 'ellipse';
}
else if (mxUtils.indexOf(['condition', 'decision'], shapenameToken) >= 0)
{
shapenameToken = shape = 'rhombus';
}
else if (mxUtils.indexOf(['preparation'], shapenameToken) >= 0)
{
shapenameToken = shape = 'hexagon';
}
else if (mxUtils.indexOf(['trapez'], shapenameToken) >= 0)
{
shapenameToken = shape = 'parallelogram';
}
else
{
shape = getBestWord(shapenameToken, shapeList);
}
// LATER: Ignore all edge shapes
if (shape != null && shape.toLowerCase() == shapenameToken && shape != 'connector' &&
shape != 'arrow' && shape != 'flexarrow' && shape != 'arrowconnector' &&
shape != 'link')
{
if (cells.length == 0)
{
App.say('No cell for {1}', [shape]);
}
else if (vertices.length == 0)
{
App.say('Connections ignored');
}
else
{
// LATER: Add perimeter style etc
graph.setCellStyles(mxConstants.STYLE_SHAPE, shapenameToken, vertices);
ui.fireEvent(new mxEventObject('styleChanged', 'keys', [mxConstants.STYLE_SHAPE],
'values', [shapenameToken], 'cells', vertices));
// Terminate
return true;
}
}
// Lazy creation of cache for action names
if (actions == null)
{
actions = {};
actionList = [];
for (var name in ui.actions.actions)
{
var shortname = mxUtils.trim(name).toLowerCase().replace(/ /g, '');
actions[shortname] = ui.actions.actions[name];
actionList.push(shortname);
}
}
var name = mxUtils.trim(command).toLowerCase().replace(/ /g, '');
if (urlParams['dev'] == '1')
{
var guess = getBestWord(name, actionList, true);
console.log('App.say guess:', name, '=>', guess, '(' +
naiveHammingDistance(name, guess) + ', ' +
levenshteinDist(name, guess) +
')');
}
// Some common redirects and mistakes
if (tokens[0] == 'back')
{
command = name = 'undo';
}
else if (tokens[0] == 'restore')
{
command = name = 'redo';
}
else if (tokens[0] == 'clone')
{
command = name = 'duplicate';
}
else if (mxUtils.indexOf(['all', 'small', 'wall', 'call'], name) >= 0)
{
command = name = 'selectall';
}
else if (mxUtils.indexOf(['both', 'boat', 'phone', 'don\'t', 'food', 'bolt', 'bull'], tokens[0]) >= 0)
{
command = name = 'bold';
}
var action = actions[name];
if (action != null)
{
if (action.isEnabled())
{
App.say(name);
action.funct();
}
else
{
// Cannot use name here as it has no spaces
App.say('{1} disabled', [command]);
}
return true;
}
return false;
}
}
};
if ('speechSynthesis' in window)
{
// Shows dialog for mic access
if (navigator.webkitGetUserMedia)
{
window.addEventListener('load', function()
{
navigator.webkitGetUserMedia({audio:true}, function()
{
//console.log('access to mic ok');
}, function(e)
{
//alert('Error getting audio');
console.log(e);
});
});
}
// Installs helper methods and listeners for speech output
var graph = ui.editor.graph;
App.reloadVoicePlugin = function()
{
// Shows UI
speechOutputEnabled = true;
menu.style.display = '';
td.style.display = 'inline-block';
};
App.sayHint = function()
{
if (graph.getSelectionCount() == 0)
{
App.say('No cell selected.' + ((ui.isDiagramEmpty()) ? ' Diagram is empty.' : ''));
}
else if (graph.getSelectionCount() == 1)
{
var cell = graph.getSelectionCell();
if (graph.getModel().isVertex(cell))
{
var geo = graph.getCellGeometry(cell)
if (geo != null)
{
App.say('{1} at {2} and {3} is {4} times {5} pixels', [graph.getWordForCell(cell, true),
geo.x, geo.y, geo.width, geo.height]);
}
}
else
{
App.say('One connection selected');
}
var lab = mxUtils.trim(graph.getLabel(cell));
if (lab != null && lab.length > 0 && lab.length < 20)
{
App.say('Label is {1}', [lab]);
}
}
else
{
var cells = graph.getSelectionCells();
var vertexCount = 0;
var edgeCount = 0;
for (var i = 0; i < cells.length; i++)
{
var cell = cells[i];
if (graph.model.isEdge(cell))
{
edgeCount++;
}
else if (graph.model.isVertex(cell))
{
vertexCount++;
}
}
if (vertexCount > 0 && edgeCount > 0)
{
var tmp = '';
if (vertexCount == 1)
{
tmp += 'One shape';
}
else
{
tmp += '{1} shapes';
}
tmp += ' and ';
if (edgeCount == 1)
{
tmp += 'one connection';
}
else
{
tmp += '{2} connections';
}
App.say(tmp + ' selected', [vertexCount, edgeCount]);
}
else if (vertexCount > 0)
{
App.say('{1} shapes selected', [vertexCount]);
}
else
{
App.say('{1} connections selected', [edgeCount]);
}
}
};
// Can return more than first word
function firstWord(value)
{
// TODO Use regex
if (value != null && value.length > maxLabelLength)
{
var space = value.indexOf(' ');
// Add second word if label is short
var tmp = value.indexOf(' ', space + 1);
if (tmp >= 0 && tmp <= maxLabelLength)
{
space = tmp;
// Add third word if label is short
var tmp = value.indexOf(' ', space + 1);
if (tmp >= 0 && tmp <= maxLabelLength)
{
space = tmp;
}
}
if (space >= 0)
{
value = value.substring(0, space);
}
}
return value;
};
graph.getWordForCell = function(cell, ignoreLabel)
{
var label = 'no';
if (cell != null)
{
label = (ignoreLabel) ? null : firstWord(this.getLabel(cell));
if (label == null || label.length == 0 || label.length > maxLabelLength || mxUtils.isNumeric(label))
{
var state = this.view.getState(cell);
var style = (state != null) ? state.style : this.getCellStyle(cell);
var tmp = style[mxConstants.STYLE_SHAPE];
if (tmp == 'label')
{
label = 'rectangle';
}
else if (tmp != null)
{
var dot = tmp.lastIndexOf('.');
if (dot >= 0)
{
tmp = tmp.substring(dot + 1);
}
label = tmp
}
}
var div = document.createElement('div');
div.innerHTML = label;
label = div.innerText;
}
return label;
};
ui.addListener('styleChanged', function(sender, evt)
{
//console.log('styleChanged', evt, evt.getProperty('keys'));
var cells = evt.getProperty('cells');
var keys = evt.getProperty('keys');
var values = evt.getProperty('values');
if (cells != null && keys != null && keys.length == 1 && values.length == 1)
{
var tmp = values[0];
if (typeof tmp === 'string' && tmp.charAt(0) == '#')
{
var n_match = ntc.name(tmp);
//if (n_match[2]) /* exact match */
{
tmp = n_match[1];
}
}
if (tmp == mxConstants.NONE || tmp == null)
{
App.say('Removed {1}', [keys[0]]);
}
else
{
// Replaces camel case notation with spaces
var key = keys[0].replace(/([A-Z])/g, ' $1')
App.say('{1} {2}', [key, tmp]);
}
}
});
graph.addListener(mxEvent.LABEL_CHANGED, function(sender, evt)
{
//console.log('CELL_CONNECTED', evt);
var cell = evt.getProperty('cell');
var old = evt.getProperty('old');
// Resolves placeholders in new label
var value = this.getLabel(cell);
if ((value != null && value.length > 20) ||
(old != null && old.length > 20))
{
App.say('Label changed');
}
else if (value != null && value.length == 0)
{
App.say('Label removed');
}
else if (value != null)
{
App.say('{1}', [value]);
}
});
graph.addListener(mxEvent.CELL_CONNECTED, function(sender, evt)
{
// console.log('CELL_CONNECTED', evt);
// var edge = evt.getProperty('edge');
// var terminal = evt.getProperty('terminal');
// var other = graph.model.getTerminal(edge, !evt.getProperty('source'));
//
// if (other != null && terminal != null)
// {
// App.say('connected {1} to {2}', [getWordForCell(other), getWordForCell(terminal)]);
// }
});
graph.addListener(mxEvent.CELLS_ADDED, function(sender, evt)
{
//console.log('CELLS_ADDED', evt);
var cells = evt.getProperty('cells');
if (cells.length > 1)
{
var vertexCount = 0;
var edgeCount = 0;
for (var i = 0; i < cells.length; i++)
{
var cell = cells[i];
if (graph.model.isEdge(cell))
{
edgeCount++;
}
else if (graph.model.isVertex(cell))
{
vertexCount++;
}
}
if (vertexCount > 0 && edgeCount > 0)
{
var tmp = '';
if (vertexCount == 1)
{
tmp += 'One shape';
}
else
{
tmp += '{1} shapes';
}
tmp += ' and ';
if (edgeCount == 1)
{
tmp += 'one connection';
}
else
{
tmp += '{2} connections';
}
App.say(tmp + ' inserted', [vertexCount, edgeCount]);
}
else if (vertexCount > 0)
{
App.say('{1} shapes', [vertexCount]);
}
else
{
App.say('{1} connections', [edgeCount]);
}
}
else
{
cell = cells[0];
if (graph.model.isEdge(cell) && graph.model.getTerminal(cell, true) != null &&
graph.model.getTerminal(cell, false) != null)
{
if (graph.getWordForCell(graph.model.getTerminal(cell, true)) ==
graph.getWordForCell(graph.model.getTerminal(cell, false)))
{
App.say('Connected');
}
else
{
App.say('{1} connected to {2}', [graph.getWordForCell(graph.model.getTerminal(cell, true)),
graph.getWordForCell(graph.model.getTerminal(cell, false))]);
}
}
else if (graph.model.isVertex(cell))
{
lastInserted = cell;
// Replaces camel case notation with spaces
var word = graph.getWordForCell(cell).replace(/([A-Z])/g, ' $1');
App.say('{1}', [word]);
}
}
});
graph.addListener(mxEvent.CELLS_REMOVED, function(sender, evt)
{
//console.log('CELLS_REMOVED', evt);
var cells = evt.getProperty('cells');
if (cells.length > 1)
{
App.say('{1} cells deleted', [cells.length]);
}
else
{
cell = cells[0];
App.say('{1} deleted', [graph.getWordForCell(cell)]);
}
});
}
});
/** Code for color to voice based on ntc (name that color)
/*
+-----------------------------------------------------------------+
| Created by Chirag Mehta - http://chir.ag/projects/ntc |
|-----------------------------------------------------------------|
| ntc js (Name that Color JavaScript) |
+-----------------------------------------------------------------+
All the functions, code, lists etc. have been written specifically
for the Name that Color JavaScript by Chirag Mehta unless otherwise
specified.
This script is released under the: Creative Commons License:
Attribution 2.5 http://creativecommons.org/licenses/by/2.5/
Sample Usage:
<script type="text/javascript" src="ntc.js"></script>
<script type="text/javascript">
var n_match = ntc.name("#6195ED");
n_rgb = n_match[0]; // This is the RGB value of the closest matching color
n_name = n_match[1]; // This is the text string for the name of the match
n_exactmatch = n_match[2]; // True if exact color match, False if close-match
alert(n_match);
</script>
*/
var ntc = {
init: function() {
var color, rgb, hsl;
for(var i = 0; i < ntc.names.length; i++)
{
color = "#" + ntc.names[i][0];
rgb = ntc.rgb(color);
hsl = ntc.hsl(color);
ntc.names[i].push(rgb[0], rgb[1], rgb[2], hsl[0], hsl[1], hsl[2]);
}
},
name: function(color) {
color = color.toUpperCase();
if(color.length < 3 || color.length > 7)
return ["#000000", "Invalid Color: " + color, false];
if(color.length % 3 == 0)
color = "#" + color;
if(color.length == 4)
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);
var rgb = ntc.rgb(color);
var r = rgb[0], g = rgb[1], b = rgb[2];
var hsl = ntc.hsl(color);
var h = hsl[0], s = hsl[1], l = hsl[2];
var ndf1 = 0; ndf2 = 0; ndf = 0;
var cl = -1, df = -1;
for(var i = 0; i < ntc.names.length; i++)
{
if(color == "#" + ntc.names[i][0])
return ["#" + ntc.names[i][0], ntc.names[i][1], true];
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);
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);
ndf = ndf1 + ndf2 * 2;
if(df < 0 || df > ndf)
{
df = ndf;
cl = i;
}
}
return (cl < 0 ? ["#000000", "Invalid Color: " + color, false] : ["#" + ntc.names[cl][0], ntc.names[cl][1], false]);
},
// adopted from: Farbtastic 1.2
// http://acko.net/dev/farbtastic
hsl: function (color) {
var rgb = [parseInt('0x' + color.substring(1, 3)) / 255, parseInt('0x' + color.substring(3, 5)) / 255, parseInt('0x' + color.substring(5, 7)) / 255];
var min, max, delta, h, s, l;
var r = rgb[0], g = rgb[1], b = rgb[2];
min = Math.min(r, Math.min(g, b));
max = Math.max(r, Math.max(g, b));
delta = max - min;
l = (min + max) / 2;
s = 0;
if(l > 0 && l < 1)
s = delta / (l < 0.5 ? (2 * l) : (2 - 2 * l));
h = 0;
if(delta > 0)
{
if (max == r && max != g) h += (g - b) / delta;
if (max == g && max != b) h += (2 + (b - r) / delta);
if (max == b && max != r) h += (4 + (r - g) / delta);
h /= 6;
}
return [parseInt(h * 255), parseInt(s * 255), parseInt(l * 255)];
},
// adopted from: Farbtastic 1.2
// http://acko.net/dev/farbtastic
rgb: function(color) {
return [parseInt('0x' + color.substring(1, 3)), parseInt('0x' + color.substring(3, 5)), parseInt('0x' + color.substring(5, 7))];
},
names: [
// Added all standard HTML color names / gaudenz, oct-2015
["F0F8FF", "aliceblue"],
["FAEBD7", "antiquewhite"],
["00FFFF", "aqua"],
["7FFFD4", "aquamarine"],
["F0FFFF", "azure"],
["F5F5DC", "beige"],
["FFE4C4", "bisque"],
["000000", "black"],
["FFEBCD", "blanchedalmond"],
["0000FF", "blue"],
["8A2BE2", "blueviolet"],
["A52A2A", "brown"],
["DEB887", "burlywood"],
["5F9EA0", "cadetblue"],
["7FFF00", "chartreuse"],
["D2691E", "chocolate"],
["FF7F50", "coral"],
["6495ED", "cornflowerblue"],
["FFF8DC", "cornsilk"],
["DC143C", "crimson"],
["00FFFF", "cyan"],
["00008B", "darkblue"],
["008B8B", "darkcyan"],
["B8860B", "darkgoldenrod"],
["A9A9A9", "darkgray"],
["A9A9A9", "darkgrey"],
["006400", "darkgreen"],
["BDB76B", "darkkhaki"],
["8B008B", "darkmagenta"],
["556B2F", "darkolivegreen"],
["FF8C00", "darkorange"],
["9932CC", "darkorchid"],
["8B0000", "darkred"],
["E9967A", "darksalmon"],
["8FBC8F", "darkseagreen"],
["483D8B", "darkslateblue"],
["2F4F4F", "darkslategray"],
["2F4F4F", "darkslategrey"],
["00CED1", "darkturquoise"],
["9400D3", "darkviolet"],
["FF1493", "deeppink"],
["00BFFF", "deepskyblue"],
["696969", "dimgray"],
["696969", "dimgrey"],
["1E90FF", "dodgerblue"],
["B22222", "firebrick"],
["FFFAF0", "floralwhite"],
["228B22", "forestgreen"],
["FF00FF", "fuchsia"],
["DCDCDC", "gainsboro"],
["F8F8FF", "ghostwhite"],
["FFD700", "gold"],
["DAA520", "goldenrod"],
["808080", "gray"],
["808080", "grey"],
["008000", "green"],
["ADFF2F", "greenyellow"],
["F0FFF0", "honeydew"],
["FF69B4", "hotpink"],
["CD5C5C", "indianred "],
["4B0082", "indigo "],
["FFFFF0", "ivory"],
["F0E68C", "khaki"],
["E6E6FA", "lavender"],
["FFF0F5", "lavenderblush"],
["7CFC00", "lawngreen"],
["FFFACD", "lemonchiffon"],
["ADD8E6", "lightblue"],
["F08080", "lightcoral"],
["E0FFFF", "lightcyan"],
["FAFAD2", "lightgoldenrodyellow"],
["D3D3D3", "lightgray"],
["D3D3D3", "lightgrey"],
["90EE90", "lightgreen"],
["FFB6C1", "lightpink"],
["FFA07A", "lightsalmon"],
["20B2AA", "lightseagreen"],
["87CEFA", "lightskyblue"],
["778899", "lightslategray"],
["778899", "lightslategrey"],
["B0C4DE", "lightsteelblue"],
["FFFFE0", "lightyellow"],
["00FF00", "lime"],
["32CD32", "limegreen"],
["FAF0E6", "linen"],
["FF00FF", "magenta"],
["800000", "maroon"],
["66CDAA", "mediumaquamarine"],
["0000CD", "mediumblue"],
["BA55D3", "mediumorchid"],
["9370DB", "mediumpurple"],
["3CB371", "mediumseagreen"],
["7B68EE", "mediumslateblue"],
["00FA9A", "mediumspringgreen"],
["48D1CC", "mediumturquoise"],
["C71585", "mediumvioletred"],
["191970", "midnightblue"],
["F5FFFA", "mintcream"],
["FFE4E1", "mistyrose"],
["FFE4B5", "moccasin"],
["FFDEAD", "navajowhite"],
["000080", "navy"],
["FDF5E6", "oldlace"],
["808000", "olive"],
["6B8E23", "olivedrab"],
["FFA500", "orange"],
["FF4500", "orangered"],
["DA70D6", "orchid"],
["EEE8AA", "palegoldenrod"],
["98FB98", "palegreen"],
["AFEEEE", "paleturquoise"],
["DB7093", "palevioletred"],
["FFEFD5", "papayawhip"],
["FFDAB9", "peachpuff"],
["CD853F", "peru"],
["FFC0CB", "pink"],
["DDA0DD", "plum"],
["B0E0E6", "powderblue"],
["800080", "purple"],
["FF0000", "red"],
["BC8F8F", "rosybrown"],
["4169E1", "royalblue"],
["8B4513", "saddlebrown"],
["FA8072", "salmon"],
["F4A460", "sandybrown"],
["2E8B57", "seagreen"],
["FFF5EE", "seashell"],
["A0522D", "sienna"],
["C0C0C0", "silver"],
["87CEEB", "skyblue"],
["6A5ACD", "slateblue"],
["708090", "slategray"],
["708090", "slategrey"],
["FFFAFA", "snow"],
["00FF7F", "springgreen"],
["4682B4", "steelblue"],
["D2B48C", "tan"],
["008080", "teal"],
["D8BFD8", "thistle"],
["FF6347", "tomato"],
["40E0D0", "turquoise"],
["EE82EE", "violet"],
["F5DEB3", "wheat"],
["FFFFFF", "white"],
["F5F5F5", "whitesmoke"],
["FFFF00", "yellow"],
["9ACD32", "yellowgreen"],
// Existing table
["000000", "Black"],
["000080", "Navy Blue"],
["0000C8", "Dark Blue"],
["0000FF", "Blue"],
["000741", "Stratos"],
["001B1C", "Swamp"],
["002387", "Resolution Blue"],
["002900", "Deep Fir"],
["002E20", "Burnham"],
["002FA7", "International Klein Blue"],
["003153", "Prussian Blue"],
["003366", "Midnight Blue"],
["003399", "Smalt"],
["003532", "Deep Teal"],
["003E40", "Cyprus"],
["004620", "Kaitoke Green"],
["0047AB", "Cobalt"],
["004816", "Crusoe"],
["004950", "Sherpa Blue"],
["0056A7", "Endeavour"],
["00581A", "Camarone"],
["0066CC", "Science Blue"],
["0066FF", "Blue Ribbon"],
["00755E", "Tropical Rain Forest"],
["0076A3", "Allports"],
["007BA7", "Deep Cerulean"],
["007EC7", "Lochmara"],
["007FFF", "Azure Radiance"],
["008080", "Teal"],
["0095B6", "Bondi Blue"],
["009DC4", "Pacific Blue"],
["00A693", "Persian Green"],
["00A86B", "Jade"],
["00CC99", "Caribbean Green"],
["00CCCC", "Robin's Egg Blue"],
["00FF00", "Green"],
["00FF7F", "Spring Green"],
["00FFFF", "Cyan / Aqua"],
["010D1A", "Blue Charcoal"],
["011635", "Midnight"],
["011D13", "Holly"],
["012731", "Daintree"],
["01361C", "Cardin Green"],
["01371A", "County Green"],
["013E62", "Astronaut Blue"],
["013F6A", "Regal Blue"],
["014B43", "Aqua Deep"],
["015E85", "Orient"],
["016162", "Blue Stone"],
["016D39", "Fun Green"],
["01796F", "Pine Green"],
["017987", "Blue Lagoon"],
["01826B", "Deep Sea"],
["01A368", "Green Haze"],
["022D15", "English Holly"],
["02402C", "Sherwood Green"],
["02478E", "Congress Blue"],
["024E46", "Evening Sea"],
["026395", "Bahama Blue"],
["02866F", "Observatory"],
["02A4D3", "Cerulean"],
["03163C", "Tangaroa"],
["032B52", "Green Vogue"],
["036A6E", "Mosque"],
["041004", "Midnight Moss"],
["041322", "Black Pearl"],
["042E4C", "Blue Whale"],
["044022", "Zuccini"],
["044259", "Teal Blue"],
["051040", "Deep Cove"],
["051657", "Gulf Blue"],
["055989", "Venice Blue"],
["056F57", "Watercourse"],
["062A78", "Catalina Blue"],
["063537", "Tiber"],
["069B81", "Gossamer"],
["06A189", "Niagara"],
["073A50", "Tarawera"],
["080110", "Jaguar"],
["081910", "Black Bean"],
["082567", "Deep Sapphire"],
["088370", "Elf Green"],
["08E8DE", "Bright Turquoise"],
["092256", "Downriver"],
["09230F", "Palm Green"],
["09255D", "Madison"],
["093624", "Bottle Green"],
["095859", "Deep Sea Green"],
["097F4B", "Salem"],
["0A001C", "Black Russian"],
["0A480D", "Dark Fern"],
["0A6906", "Japanese Laurel"],
["0A6F75", "Atoll"],
["0B0B0B", "Cod Gray"],
["0B0F08", "Marshland"],
["0B1107", "Gordons Green"],
["0B1304", "Black Forest"],
["0B6207", "San Felix"],
["0BDA51", "Malachite"],
["0C0B1D", "Ebony"],
["0C0D0F", "Woodsmoke"],
["0C1911", "Racing Green"],
["0C7A79", "Surfie Green"],
["0C8990", "Blue Chill"],
["0D0332", "Black Rock"],
["0D1117", "Bunker"],
["0D1C19", "Aztec"],
["0D2E1C", "Bush"],
["0E0E18", "Cinder"],
["0E2A30", "Firefly"],
["0F2D9E", "Torea Bay"],
["10121D", "Vulcan"],
["101405", "Green Waterloo"],
["105852", "Eden"],
["110C6C", "Arapawa"],
["120A8F", "Ultramarine"],
["123447", "Elephant"],
["126B40", "Jewel"],
["130000", "Diesel"],
["130A06", "Asphalt"],
["13264D", "Blue Zodiac"],
["134F19", "Parsley"],
["140600", "Nero"],
["1450AA", "Tory Blue"],
["151F4C", "Bunting"],
["1560BD", "Denim"],
["15736B", "Genoa"],
["161928", "Mirage"],
["161D10", "Hunter Green"],
["162A40", "Big Stone"],
["163222", "Celtic"],
["16322C", "Timber Green"],
["163531", "Gable Green"],
["171F04", "Pine Tree"],
["175579", "Chathams Blue"],
["182D09", "Deep Forest Green"],
["18587A", "Blumine"],
["19330E", "Palm Leaf"],
["193751", "Nile Blue"],
["1959A8", "Fun Blue"],
["1A1A68", "Lucky Point"],
["1AB385", "Mountain Meadow"],
["1B0245", "Tolopea"],
["1B1035", "Haiti"],
["1B127B", "Deep Koamaru"],
["1B1404", "Acadia"],
["1B2F11", "Seaweed"],
["1B3162", "Biscay"],
["1B659D", "Matisse"],
["1C1208", "Crowshead"],
["1C1E13", "Rangoon Green"],
["1C39BB", "Persian Blue"],
["1C402E", "Everglade"],
["1C7C7D", "Elm"],
["1D6142", "Green Pea"],
["1E0F04", "Creole"],
["1E1609", "Karaka"],
["1E1708", "El Paso"],
["1E385B", "Cello"],
["1E433C", "Te Papa Green"],
["1E90FF", "Dodger Blue"],
["1E9AB0", "Eastern Blue"],
["1F120F", "Night Rider"],
["1FC2C2", "Java"],
["20208D", "Jacksons Purple"],
["202E54", "Cloud Burst"],
["204852", "Blue Dianne"],
["211A0E", "Eternity"],
["220878", "Deep Blue"],
["228B22", "Forest Green"],
["233418", "Mallard"],
["240A40", "Violet"],
["240C02", "Kilamanjaro"],
["242A1D", "Log Cabin"],
["242E16", "Black Olive"],
["24500F", "Green House"],
["251607", "Graphite"],
["251706", "Cannon Black"],
["251F4F", "Port Gore"],
["25272C", "Shark"],
["25311C", "Green Kelp"],
["2596D1", "Curious Blue"],
["260368", "Paua"],
["26056A", "Paris M"],
["261105", "Wood Bark"],
["261414", "Gondola"],
["262335", "Steel Gray"],
["26283B", "Ebony Clay"],
["273A81", "Bay of Many"],
["27504B", "Plantation"],
["278A5B", "Eucalyptus"],
["281E15", "Oil"],
["283A77", "Astronaut"],
["286ACD", "Mariner"],
["290C5E", "Violent Violet"],
["292130", "Bastille"],
["292319", "Zeus"],
["292937", "Charade"],
["297B9A", "Jelly Bean"],
["29AB87", "Jungle Green"],
["2A0359", "Cherry Pie"],
["2A140E", "Coffee Bean"],
["2A2630", "Baltic Sea"],
["2A380B", "Turtle Green"],
["2A52BE", "Cerulean Blue"],
["2B0202", "Sepia Black"],
["2B194F", "Valhalla"],
["2B3228", "Heavy Metal"],
["2C0E8C", "Blue Gem"],
["2C1632", "Revolver"],
["2C2133", "Bleached Cedar"],
["2C8C84", "Lochinvar"],
["2D2510", "Mikado"],
["2D383A", "Outer Space"],
["2D569B", "St Tropaz"],
["2E0329", "Jacaranda"],
["2E1905", "Jacko Bean"],
["2E3222", "Rangitoto"],
["2E3F62", "Rhino"],
["2E8B57", "Sea Green"],
["2EBFD4", "Scooter"],
["2F270E", "Onion"],
["2F3CB3", "Governor Bay"],
["2F519E", "Sapphire"],
["2F5A57", "Spectra"],
["2F6168", "Casal"],
["300529", "Melanzane"],
["301F1E", "Cocoa Brown"],
["302A0F", "Woodrush"],
["304B6A", "San Juan"],
["30D5C8", "Turquoise"],
["311C17", "Eclipse"],
["314459", "Pickled Bluewood"],
["315BA1", "Azure"],
["31728D", "Calypso"],
["317D82", "Paradiso"],
["32127A", "Persian Indigo"],
["32293A", "Blackcurrant"],
["323232", "Mine Shaft"],
["325D52", "Stromboli"],
["327C14", "Bilbao"],
["327DA0", "Astral"],
["33036B", "Christalle"],
["33292F", "Thunder"],
["33CC99", "Shamrock"],
["341515", "Tamarind"],
["350036", "Mardi Gras"],
["350E42", "Valentino"],
["350E57", "Jagger"],
["353542", "Tuna"],
["354E8C", "Chambray"],
["363050", "Martinique"],
["363534", "Tuatara"],
["363C0D", "Waiouru"],
["36747D", "Ming"],
["368716", "La Palma"],
["370202", "Chocolate"],
["371D09", "Clinker"],
["37290E", "Brown Tumbleweed"],
["373021", "Birch"],
["377475", "Oracle"],
["380474", "Blue Diamond"],
["381A51", "Grape"],
["383533", "Dune"],
["384555", "Oxford Blue"],
["384910", "Clover"],
["394851", "Limed Spruce"],
["396413", "Dell"],
["3A0020", "Toledo"],
["3A2010", "Sambuca"],
["3A2A6A", "Jacarta"],
["3A686C", "William"],
["3A6A47", "Killarney"],
["3AB09E", "Keppel"],
["3B000B", "Temptress"],
["3B0910", "Aubergine"],
["3B1F1F", "Jon"],
["3B2820", "Treehouse"],
["3B7A57", "Amazon"],
["3B91B4", "Boston Blue"],
["3C0878", "Windsor"],
["3C1206", "Rebel"],
["3C1F76", "Meteorite"],
["3C2005", "Dark Ebony"],
["3C3910", "Camouflage"],
["3C4151", "Bright Gray"],
["3C4443", "Cape Cod"],
["3C493A", "Lunar Green"],
["3D0C02", "Bean "],
["3D2B1F", "Bistre"],
["3D7D52", "Goblin"],
["3E0480", "Kingfisher Daisy"],
["3E1C14", "Cedar"],
["3E2B23", "English Walnut"],
["3E2C1C", "Black Marlin"],
["3E3A44", "Ship Gray"],
["3EABBF", "Pelorous"],
["3F2109", "Bronze"],
["3F2500", "Cola"],
["3F3002", "Madras"],
["3F307F", "Minsk"],
["3F4C3A", "Cabbage Pont"],
["3F583B", "Tom Thumb"],
["3F5D53", "Mineral Green"],
["3FC1AA", "Puerto Rico"],
["3FFF00", "Harlequin"],
["401801", "Brown Pod"],
["40291D", "Cork"],
["403B38", "Masala"],
["403D19", "Thatch Green"],
["405169", "Fiord"],
["40826D", "Viridian"],
["40A860", "Chateau Green"],
["410056", "Ripe Plum"],
["411F10", "Paco"],
["412010", "Deep Oak"],
["413C37", "Merlin"],
["414257", "Gun Powder"],
["414C7D", "East Bay"],
["4169E1", "Royal Blue"],
["41AA78", "Ocean Green"],
["420303", "Burnt Maroon"],
["423921", "Lisbon Brown"],
["427977", "Faded Jade"],
["431560", "Scarlet Gum"],
["433120", "Iroko"],
["433E37", "Armadillo"],
["434C59", "River Bed"],
["436A0D", "Green Leaf"],
["44012D", "Barossa"],
["441D00", "Morocco Brown"],
["444954", "Mako"],
["454936", "Kelp"],
["456CAC", "San Marino"],
["45B1E8", "Picton Blue"],
["460B41", "Loulou"],
["462425", "Crater Brown"],
["465945", "Gray Asparagus"],
["4682B4", "Steel Blue"],
["480404", "Rustic Red"],
["480607", "Bulgarian Rose"],
["480656", "Clairvoyant"],
["481C1C", "Cocoa Bean"],
["483131", "Woody Brown"],
["483C32", "Taupe"],
["49170C", "Van Cleef"],
["492615", "Brown Derby"],
["49371B", "Metallic Bronze"],
["495400", "Verdun Green"],
["496679", "Blue Bayoux"],
["497183", "Bismark"],
["4A2A04", "Bracken"],
["4A3004", "Deep Bronze"],
["4A3C30", "Mondo"],
["4A4244", "Tundora"],
["4A444B", "Gravel"],
["4A4E5A", "Trout"],
["4B0082", "Pigment Indigo"],
["4B5D52", "Nandor"],
["4C3024", "Saddle"],
["4C4F56", "Abbey"],
["4D0135", "Blackberry"],
["4D0A18", "Cab Sav"],
["4D1E01", "Indian Tan"],
["4D282D", "Cowboy"],
["4D282E", "Livid Brown"],
["4D3833", "Rock"],
["4D3D14", "Punga"],
["4D400F", "Bronzetone"],
["4D5328", "Woodland"],
["4E0606", "Mahogany"],
["4E2A5A", "Bossanova"],
["4E3B41", "Matterhorn"],
["4E420C", "Bronze Olive"],
["4E4562", "Mulled Wine"],
["4E6649", "Axolotl"],
["4E7F9E", "Wedgewood"],
["4EABD1", "Shakespeare"],
["4F1C70", "Honey Flower"],
["4F2398", "Daisy Bush"],
["4F69C6", "Indigo"],
["4F7942", "Fern Green"],
["4F9D5D", "Fruit Salad"],
["4FA83D", "Apple"],
["504351", "Mortar"],
["507096", "Kashmir Blue"],
["507672", "Cutty Sark"],
["50C878", "Emerald"],
["514649", "Emperor"],
["516E3D", "Chalet Green"],
["517C66", "Como"],
["51808F", "Smalt Blue"],
["52001F", "Castro"],
["520C17", "Maroon Oak"],
["523C94", "Gigas"],
["533455", "Voodoo"],
["534491", "Victoria"],
["53824B", "Hippie Green"],
["541012", "Heath"],
["544333", "Judge Gray"],
["54534D", "Fuscous Gray"],
["549019", "Vida Loca"],
["55280C", "Cioccolato"],
["555B10", "Saratoga"],
["556D56", "Finlandia"],
["5590D9", "Havelock Blue"],
["56B4BE", "Fountain Blue"],
["578363", "Spring Leaves"],
["583401", "Saddle Brown"],
["585562", "Scarpa Flow"],
["587156", "Cactus"],
["589AAF", "Hippie Blue"],
["591D35", "Wine Berry"],
["592804", "Brown Bramble"],
["593737", "Congo Brown"],
["594433", "Millbrook"],
["5A6E9C", "Waikawa Gray"],
["5A87A0", "Horizon"],
["5B3013", "Jambalaya"],
["5C0120", "Bordeaux"],
["5C0536", "Mulberry Wood"],
["5C2E01", "Carnaby Tan"],
["5C5D75", "Comet"],
["5D1E0F", "Redwood"],
["5D4C51", "Don Juan"],
["5D5C58", "Chicago"],
["5D5E37", "Verdigris"],
["5D7747", "Dingley"],
["5DA19F", "Breaker Bay"],
["5E483E", "Kabul"],
["5E5D3B", "Hemlock"],
["5F3D26", "Irish Coffee"],
["5F5F6E", "Mid Gray"],
["5F6672", "Shuttle Gray"],
["5FA777", "Aqua Forest"],
["5FB3AC", "Tradewind"],
["604913", "Horses Neck"],
["605B73", "Smoky"],
["606E68", "Corduroy"],
["6093D1", "Danube"],
["612718", "Espresso"],
["614051", "Eggplant"],
["615D30", "Costa Del Sol"],
["61845F", "Glade Green"],
["622F30", "Buccaneer"],
["623F2D", "Quincy"],
["624E9A", "Butterfly Bush"],
["625119", "West Coast"],
["626649", "Finch"],
["639A8F", "Patina"],
["63B76C", "Fern"],
["6456B7", "Blue Violet"],
["646077", "Dolphin"],
["646463", "Storm Dust"],
["646A54", "Siam"],
["646E75", "Nevada"],
["6495ED", "Cornflower Blue"],
["64CCDB", "Viking"],
["65000B", "Rosewood"],
["651A14", "Cherrywood"],
["652DC1", "Purple Heart"],
["657220", "Fern Frond"],
["65745D", "Willow Grove"],
["65869F", "Hoki"],
["660045", "Pompadour"],
["660099", "Purple"],
["66023C", "Tyrian Purple"],
["661010", "Dark Tan"],
["66B58F", "Silver Tree"],
["66FF00", "Bright Green"],
["66FF66", "Screamin' Green"],
["67032D", "Black Rose"],
["675FA6", "Scampi"],
["676662", "Ironside Gray"],
["678975", "Viridian Green"],
["67A712", "Christi"],
["683600", "Nutmeg Wood Finish"],
["685558", "Zambezi"],
["685E6E", "Salt Box"],
["692545", "Tawny Port"],
["692D54", "Finn"],
["695F62", "Scorpion"],
["697E9A", "Lynch"],
["6A442E", "Spice"],
["6A5D1B", "Himalaya"],
["6A6051", "Soya Bean"],
["6B2A14", "Hairy Heath"],
["6B3FA0", "Royal Purple"],
["6B4E31", "Shingle Fawn"],
["6B5755", "Dorado"],
["6B8BA2", "Bermuda Gray"],
["6B8E23", "Olive Drab"],
["6C3082", "Eminence"],
["6CDAE7", "Turquoise Blue"],
["6D0101", "Lonestar"],
["6D5E54", "Pine Cone"],
["6D6C6C", "Dove Gray"],
["6D9292", "Juniper"],
["6D92A1", "Gothic"],
["6E0902", "Red Oxide"],
["6E1D14", "Moccaccino"],
["6E4826", "Pickled Bean"],
["6E4B26", "Dallas"],
["6E6D57", "Kokoda"],
["6E7783", "Pale Sky"],
["6F440C", "Cafe Royale"],
["6F6A61", "Flint"],
["6F8E63", "Highland"],
["6F9D02", "Limeade"],
["6FD0C5", "Downy"],
["701C1C", "Persian Plum"],
["704214", "Sepia"],
["704A07", "Antique Bronze"],
["704F50", "Ferra"],
["706555", "Coffee"],
["708090", "Slate Gray"],
["711A00", "Cedar Wood Finish"],
["71291D", "Metallic Copper"],
["714693", "Affair"],
["714AB2", "Studio"],
["715D47", "Tobacco Brown"],
["716338", "Yellow Metal"],
["716B56", "Peat"],
["716E10", "Olivetone"],
["717486", "Storm Gray"],
["718080", "Sirocco"],
["71D9E2", "Aquamarine Blue"],
["72010F", "Venetian Red"],
["724A2F", "Old Copper"],
["726D4E", "Go Ben"],
["727B89", "Raven"],
["731E8F", "Seance"],
["734A12", "Raw Umber"],
["736C9F", "Kimberly"],
["736D58", "Crocodile"],
["737829", "Crete"],
["738678", "Xanadu"],
["74640D", "Spicy Mustard"],
["747D63", "Limed Ash"],
["747D83", "Rolling Stone"],
["748881", "Blue Smoke"],
["749378", "Laurel"],
["74C365", "Mantis"],
["755A57", "Russett"],
["7563A8", "Deluge"],
["76395D", "Cosmic"],
["7666C6", "Blue Marguerite"],
["76BD17", "Lima"],
["76D7EA", "Sky Blue"],
["770F05", "Dark Burgundy"],
["771F1F", "Crown of Thorns"],
["773F1A", "Walnut"],
["776F61", "Pablo"],
["778120", "Pacifika"],
["779E86", "Oxley"],
["77DD77", "Pastel Green"],
["780109", "Japanese Maple"],
["782D19", "Mocha"],
["782F16", "Peanut"],
["78866B", "Camouflage Green"],
["788A25", "Wasabi"],
["788BBA", "Ship Cove"],
["78A39C", "Sea Nymph"],
["795D4C", "Roman Coffee"],
["796878", "Old Lavender"],
["796989", "Rum"],
["796A78", "Fedora"],
["796D62", "Sandstone"],
["79DEEC", "Spray"],
["7A013A", "Siren"],
["7A58C1", "Fuchsia Blue"],
["7A7A7A", "Boulder"],
["7A89B8", "Wild Blue Yonder"],
["7AC488", "De York"],
["7B3801", "Red Beech"],
["7B3F00", "Cinnamon"],
["7B6608", "Yukon Gold"],
["7B7874", "Tapa"],
["7B7C94", "Waterloo "],
["7B8265", "Flax Smoke"],
["7B9F80", "Amulet"],
["7BA05B", "Asparagus"],
["7C1C05", "Kenyan Copper"],
["7C7631", "Pesto"],
["7C778A", "Topaz"],
["7C7B7A", "Concord"],
["7C7B82", "Jumbo"],
["7C881A", "Trendy Green"],
["7CA1A6", "Gumbo"],
["7CB0A1", "Acapulco"],
["7CB7BB", "Neptune"],
["7D2C14", "Pueblo"],
["7DA98D", "Bay Leaf"],
["7DC8F7", "Malibu"],
["7DD8C6", "Bermuda"],
["7E3A15", "Copper Canyon"],
["7F1734", "Claret"],
["7F3A02", "Peru Tan"],
["7F626D", "Falcon"],
["7F7589", "Mobster"],
["7F76D3", "Moody Blue"],
["7FFF00", "Chartreuse"],
["7FFFD4", "Aquamarine"],
["800000", "Maroon"],
["800B47", "Rose Bud Cherry"],
["801818", "Falu Red"],
["80341F", "Red Robin"],
["803790", "Vivid Violet"],
["80461B", "Russet"],
["807E79", "Friar Gray"],
["808000", "Olive"],
["808080", "Gray"],
["80B3AE", "Gulf Stream"],
["80B3C4", "Glacier"],
["80CCEA", "Seagull"],
["81422C", "Nutmeg"],
["816E71", "Spicy Pink"],
["817377", "Empress"],
["819885", "Spanish Green"],
["826F65", "Sand Dune"],
["828685", "Gunsmoke"],
["828F72", "Battleship Gray"],
["831923", "Merlot"],
["837050", "Shadow"],
["83AA5D", "Chelsea Cucumber"],
["83D0C6", "Monte Carlo"],
["843179", "Plum"],
["84A0A0", "Granny Smith"],
["8581D9", "Chetwode Blue"],
["858470", "Bandicoot"],
["859FAF", "Bali Hai"],
["85C4CC", "Half Baked"],
["860111", "Red Devil"],
["863C3C", "Lotus"],
["86483C", "Ironstone"],
["864D1E", "Bull Shot"],
["86560A", "Rusty Nail"],
["868974", "Bitter"],
["86949F", "Regent Gray"],
["871550", "Disco"],
["87756E", "Americano"],
["877C7B", "Hurricane"],
["878D91", "Oslo Gray"],
["87AB39", "Sushi"],
["885342", "Spicy Mix"],
["886221", "Kumera"],
["888387", "Suva Gray"],
["888D65", "Avocado"],
["893456", "Camelot"],
["893843", "Solid Pink"],
["894367", "Cannon Pink"],
["897D6D", "Makara"],
["8A3324", "Burnt Umber"],
["8A73D6", "True V"],
["8A8360", "Clay Creek"],
["8A8389", "Monsoon"],
["8A8F8A", "Stack"],
["8AB9F1", "Jordy Blue"],
["8B00FF", "Electric Violet"],
["8B0723", "Monarch"],
["8B6B0B", "Corn Harvest"],
["8B8470", "Olive Haze"],
["8B847E", "Schooner"],
["8B8680", "Natural Gray"],
["8B9C90", "Mantle"],
["8B9FEE", "Portage"],
["8BA690", "Envy"],
["8BA9A5", "Cascade"],
["8BE6D8", "Riptide"],
["8C055E", "Cardinal Pink"],
["8C472F", "Mule Fawn"],
["8C5738", "Potters Clay"],
["8C6495", "Trendy Pink"],
["8D0226", "Paprika"],
["8D3D38", "Sanguine Brown"],
["8D3F3F", "Tosca"],
["8D7662", "Cement"],
["8D8974", "Granite Green"],
["8D90A1", "Manatee"],
["8DA8CC", "Polo Blue"],
["8E0000", "Red Berry"],
["8E4D1E", "Rope"],
["8E6F70", "Opium"],
["8E775E", "Domino"],
["8E8190", "Mamba"],
["8EABC1", "Nepal"],
["8F021C", "Pohutukawa"],
["8F3E33", "El Salva"],
["8F4B0E", "Korma"],
["8F8176", "Squirrel"],
["8FD6B4", "Vista Blue"],
["900020", "Burgundy"],
["901E1E", "Old Brick"],
["907874", "Hemp"],
["907B71", "Almond Frost"],
["908D39", "Sycamore"],
["92000A", "Sangria"],
["924321", "Cumin"],
["926F5B", "Beaver"],
["928573", "Stonewall"],
["928590", "Venus"],
["9370DB", "Medium Purple"],
["93CCEA", "Cornflower"],
["93DFB8", "Algae Green"],
["944747", "Copper Rust"],
["948771", "Arrowtown"],
["950015", "Scarlett"],
["956387", "Strikemaster"],
["959396", "Mountain Mist"],
["960018", "Carmine"],
["964B00", "Brown"],
["967059", "Leather"],
["9678B6", "Purple Mountain's Majesty"],
["967BB6", "Lavender Purple"],
["96A8A1", "Pewter"],
["96BBAB", "Summer Green"],
["97605D", "Au Chico"],
["9771B5", "Wisteria"],
["97CD2D", "Atlantis"],
["983D61", "Vin Rouge"],
["9874D3", "Lilac Bush"],
["98777B", "Bazaar"],
["98811B", "Hacienda"],
["988D77", "Pale Oyster"],
["98FF98", "Mint Green"],
["990066", "Fresh Eggplant"],
["991199", "Violet Eggplant"],
["991613", "Tamarillo"],
["991B07", "Totem Pole"],
["996666", "Copper Rose"],
["9966CC", "Amethyst"],
["997A8D", "Mountbatten Pink"],
["9999CC", "Blue Bell"],
["9A3820", "Prairie Sand"],
["9A6E61", "Toast"],
["9A9577", "Gurkha"],
["9AB973", "Olivine"],
["9AC2B8", "Shadow Green"],
["9B4703", "Oregon"],
["9B9E8F", "Lemon Grass"],
["9C3336", "Stiletto"],
["9D5616", "Hawaiian Tan"],
["9DACB7", "Gull Gray"],
["9DC209", "Pistachio"],
["9DE093", "Granny Smith Apple"],
["9DE5FF", "Anakiwa"],
["9E5302", "Chelsea Gem"],
["9E5B40", "Sepia Skin"],
["9EA587", "Sage"],
["9EA91F", "Citron"],
["9EB1CD", "Rock Blue"],
["9EDEE0", "Morning Glory"],
["9F381D", "Cognac"],
["9F821C", "Reef Gold"],
["9F9F9C", "Star Dust"],
["9FA0B1", "Santas Gray"],
["9FD7D3", "Sinbad"],
["9FDD8C", "Feijoa"],
["A02712", "Tabasco"],
["A1750D", "Buttered Rum"],
["A1ADB5", "Hit Gray"],
["A1C50A", "Citrus"],
["A1DAD7", "Aqua Island"],
["A1E9DE", "Water Leaf"],
["A2006D", "Flirt"],
["A23B6C", "Rouge"],
["A26645", "Cape Palliser"],
["A2AAB3", "Gray Chateau"],
["A2AEAB", "Edward"],
["A3807B", "Pharlap"],
["A397B4", "Amethyst Smoke"],
["A3E3ED", "Blizzard Blue"],
["A4A49D", "Delta"],
["A4A6D3", "Wistful"],
["A4AF6E", "Green Smoke"],
["A50B5E", "Jazzberry Jam"],
["A59B91", "Zorba"],
["A5CB0C", "Bahia"],
["A62F20", "Roof Terracotta"],
["A65529", "Paarl"],
["A68B5B", "Barley Corn"],
["A69279", "Donkey Brown"],
["A6A29A", "Dawn"],
["A72525", "Mexican Red"],
["A7882C", "Luxor Gold"],
["A85307", "Rich Gold"],
["A86515", "Reno Sand"],
["A86B6B", "Coral Tree"],
["A8989B", "Dusty Gray"],
["A899E6", "Dull Lavender"],
["A8A589", "Tallow"],
["A8AE9C", "Bud"],
["A8AF8E", "Locust"],
["A8BD9F", "Norway"],
["A8E3BD", "Chinook"],
["A9A491", "Gray Olive"],
["A9ACB6", "Aluminium"],
["A9B2C3", "Cadet Blue"],
["A9B497", "Schist"],
["A9BDBF", "Tower Gray"],
["A9BEF2", "Perano"],
["A9C6C2", "Opal"],
["AA375A", "Night Shadz"],
["AA4203", "Fire"],
["AA8B5B", "Muesli"],
["AA8D6F", "Sandal"],
["AAA5A9", "Shady Lady"],
["AAA9CD", "Logan"],
["AAABB7", "Spun Pearl"],
["AAD6E6", "Regent St Blue"],
["AAF0D1", "Magic Mint"],
["AB0563", "Lipstick"],
["AB3472", "Royal Heath"],
["AB917A", "Sandrift"],
["ABA0D9", "Cold Purple"],
["ABA196", "Bronco"],
["AC8A56", "Limed Oak"],
["AC91CE", "East Side"],
["AC9E22", "Lemon Ginger"],
["ACA494", "Napa"],
["ACA586", "Hillary"],
["ACA59F", "Cloudy"],
["ACACAC", "Silver Chalice"],
["ACB78E", "Swamp Green"],
["ACCBB1", "Spring Rain"],
["ACDD4D", "Conifer"],
["ACE1AF", "Celadon"],
["AD781B", "Mandalay"],
["ADBED1", "Casper"],
["ADDFAD", "Moss Green"],
["ADE6C4", "Padua"],
["ADFF2F", "Green Yellow"],
["AE4560", "Hippie Pink"],
["AE6020", "Desert"],
["AE809E", "Bouquet"],
["AF4035", "Medium Carmine"],
["AF4D43", "Apple Blossom"],
["AF593E", "Brown Rust"],
["AF8751", "Driftwood"],
["AF8F2C", "Alpine"],
["AF9F1C", "Lucky"],
["AFA09E", "Martini"],
["AFB1B8", "Bombay"],
["AFBDD9", "Pigeon Post"],
["B04C6A", "Cadillac"],
["B05D54", "Matrix"],
["B05E81", "Tapestry"],
["B06608", "Mai Tai"],
["B09A95", "Del Rio"],
["B0E0E6", "Powder Blue"],
["B0E313", "Inch Worm"],
["B10000", "Bright Red"],
["B14A0B", "Vesuvius"],
["B1610B", "Pumpkin Skin"],
["B16D52", "Santa Fe"],
["B19461", "Teak"],
["B1E2C1", "Fringy Flower"],
["B1F4E7", "Ice Cold"],
["B20931", "Shiraz"],
["B2A1EA", "Biloba Flower"],
["B32D29", "Tall Poppy"],
["B35213", "Fiery Orange"],
["B38007", "Hot Toddy"],
["B3AF95", "Taupe Gray"],
["B3C110", "La Rioja"],
["B43332", "Well Read"],
["B44668", "Blush"],
["B4CFD3", "Jungle Mist"],
["B57281", "Turkish Rose"],
["B57EDC", "Lavender"],
["B5A27F", "Mongoose"],
["B5B35C", "Olive Green"],
["B5D2CE", "Jet Stream"],
["B5ECDF", "Cruise"],
["B6316C", "Hibiscus"],
["B69D98", "Thatch"],
["B6B095", "Heathered Gray"],
["B6BAA4", "Eagle"],
["B6D1EA", "Spindle"],
["B6D3BF", "Gum Leaf"],
["B7410E", "Rust"],
["B78E5C", "Muddy Waters"],
["B7A214", "Sahara"],
["B7A458", "Husk"],
["B7B1B1", "Nobel"],
["B7C3D0", "Heather"],
["B7F0BE", "Madang"],
["B81104", "Milano Red"],
["B87333", "Copper"],
["B8B56A", "Gimblet"],
["B8C1B1", "Green Spring"],
["B8C25D", "Celery"],
["B8E0F9", "Sail"],
["B94E48", "Chestnut"],
["B95140", "Crail"],
["B98D28", "Marigold"],
["B9C46A", "Wild Willow"],
["B9C8AC", "Rainee"],
["BA0101", "Guardsman Red"],
["BA450C", "Rock Spray"],
["BA6F1E", "Bourbon"],
["BA7F03", "Pirate Gold"],
["BAB1A2", "Nomad"],
["BAC7C9", "Submarine"],
["BAEEF9", "Charlotte"],
["BB3385", "Medium Red Violet"],
["BB8983", "Brandy Rose"],
["BBD009", "Rio Grande"],
["BBD7C1", "Surf"],
["BCC9C2", "Powder Ash"],
["BD5E2E", "Tuscany"],
["BD978E", "Quicksand"],
["BDB1A8", "Silk"],
["BDB2A1", "Malta"],
["BDB3C7", "Chatelle"],
["BDBBD7", "Lavender Gray"],
["BDBDC6", "French Gray"],
["BDC8B3", "Clay Ash"],
["BDC9CE", "Loblolly"],
["BDEDFD", "French Pass"],
["BEA6C3", "London Hue"],
["BEB5B7", "Pink Swan"],
["BEDE0D", "Fuego"],
["BF5500", "Rose of Sharon"],
["BFB8B0", "Tide"],
["BFBED8", "Blue Haze"],
["BFC1C2", "Silver Sand"],
["BFC921", "Key Lime Pie"],
["BFDBE2", "Ziggurat"],
["BFFF00", "Lime"],
["C02B18", "Thunderbird"],
["C04737", "Mojo"],
["C08081", "Old Rose"],
["C0C0C0", "Silver"],
["C0D3B9", "Pale Leaf"],
["C0D8B6", "Pixie Green"],
["C1440E", "Tia Maria"],
["C154C1", "Fuchsia Pink"],
["C1A004", "Buddha Gold"],
["C1B7A4", "Bison Hide"],
["C1BAB0", "Tea"],
["C1BECD", "Gray Suit"],
["C1D7B0", "Sprout"],
["C1F07C", "Sulu"],
["C26B03", "Indochine"],
["C2955D", "Twine"],
["C2BDB6", "Cotton Seed"],
["C2CAC4", "Pumice"],
["C2E8E5", "Jagged Ice"],
["C32148", "Maroon Flush"],
["C3B091", "Indian Khaki"],
["C3BFC1", "Pale Slate"],
["C3C3BD", "Gray Nickel"],
["C3CDE6", "Periwinkle Gray"],
["C3D1D1", "Tiara"],
["C3DDF9", "Tropical Blue"],
["C41E3A", "Cardinal"],
["C45655", "Fuzzy Wuzzy Brown"],
["C45719", "Orange Roughy"],
["C4C4BC", "Mist Gray"],
["C4D0B0", "Coriander"],
["C4F4EB", "Mint Tulip"],
["C54B8C", "Mulberry"],
["C59922", "Nugget"],
["C5994B", "Tussock"],
["C5DBCA", "Sea Mist"],
["C5E17A", "Yellow Green"],
["C62D42", "Brick Red"],
["C6726B", "Contessa"],
["C69191", "Oriental Pink"],
["C6A84B", "Roti"],
["C6C3B5", "Ash"],
["C6C8BD", "Kangaroo"],
["C6E610", "Las Palmas"],
["C7031E", "Monza"],
["C71585", "Red Violet"],
["C7BCA2", "Coral Reef"],
["C7C1FF", "Melrose"],
["C7C4BF", "Cloud"],
["C7C9D5", "Ghost"],
["C7CD90", "Pine Glade"],
["C7DDE5", "Botticelli"],
["C88A65", "Antique Brass"],
["C8A2C8", "Lilac"],
["C8A528", "Hokey Pokey"],
["C8AABF", "Lily"],
["C8B568", "Laser"],
["C8E3D7", "Edgewater"],
["C96323", "Piper"],
["C99415", "Pizza"],
["C9A0DC", "Light Wisteria"],
["C9B29B", "Rodeo Dust"],
["C9B35B", "Sundance"],
["C9B93B", "Earls Green"],
["C9C0BB", "Silver Rust"],
["C9D9D2", "Conch"],
["C9FFA2", "Reef"],
["C9FFE5", "Aero Blue"],
["CA3435", "Flush Mahogany"],
["CABB48", "Turmeric"],
["CADCD4", "Paris White"],
["CAE00D", "Bitter Lemon"],
["CAE6DA", "Skeptic"],
["CB8FA9", "Viola"],
["CBCAB6", "Foggy Gray"],
["CBD3B0", "Green Mist"],
["CBDBD6", "Nebula"],
["CC3333", "Persian Red"],
["CC5500", "Burnt Orange"],
["CC7722", "Ochre"],
["CC8899", "Puce"],
["CCCAA8", "Thistle Green"],
["CCCCFF", "Periwinkle"],
["CCFF00", "Electric Lime"],
["CD5700", "Tenn"],
["CD5C5C", "Chestnut Rose"],
["CD8429", "Brandy Punch"],
["CDF4FF", "Onahau"],
["CEB98F", "Sorrell Brown"],
["CEBABA", "Cold Turkey"],
["CEC291", "Yuma"],
["CEC7A7", "Chino"],
["CFA39D", "Eunry"],
["CFB53B", "Old Gold"],
["CFDCCF", "Tasman"],
["CFE5D2", "Surf Crest"],
["CFF9F3", "Humming Bird"],
["CFFAF4", "Scandal"],
["D05F04", "Red Stage"],
["D06DA1", "Hopbush"],
["D07D12", "Meteor"],
["D0BEF8", "Perfume"],
["D0C0E5", "Prelude"],
["D0F0C0", "Tea Green"],
["D18F1B", "Geebung"],
["D1BEA8", "Vanilla"],
["D1C6B4", "Soft Amber"],
["D1D2CA", "Celeste"],
["D1D2DD", "Mischka"],
["D1E231", "Pear"],
["D2691E", "Hot Cinnamon"],
["D27D46", "Raw Sienna"],
["D29EAA", "Careys Pink"],
["D2B48C", "Tan"],
["D2DA97", "Deco"],
["D2F6DE", "Blue Romance"],
["D2F8B0", "Gossip"],
["D3CBBA", "Sisal"],
["D3CDC5", "Swirl"],
["D47494", "Charm"],
["D4B6AF", "Clam Shell"],
["D4BF8D", "Straw"],
["D4C4A8", "Akaroa"],
["D4CD16", "Bird Flower"],
["D4D7D9", "Iron"],
["D4DFE2", "Geyser"],
["D4E2FC", "Hawkes Blue"],
["D54600", "Grenadier"],
["D591A4", "Can Can"],
["D59A6F", "Whiskey"],
["D5D195", "Winter Hazel"],
["D5F6E3", "Granny Apple"],
["D69188", "My Pink"],
["D6C562", "Tacha"],
["D6CEF6", "Moon Raker"],
["D6D6D1", "Quill Gray"],
["D6FFDB", "Snowy Mint"],
["D7837F", "New York Pink"],
["D7C498", "Pavlova"],
["D7D0FF", "Fog"],
["D84437", "Valencia"],
["D87C63", "Japonica"],
["D8BFD8", "Thistle"],
["D8C2D5", "Maverick"],
["D8FCFA", "Foam"],
["D94972", "Cabaret"],
["D99376", "Burning Sand"],
["D9B99B", "Cameo"],
["D9D6CF", "Timberwolf"],
["D9DCC1", "Tana"],
["D9E4F5", "Link Water"],
["D9F7FF", "Mabel"],
["DA3287", "Cerise"],
["DA5B38", "Flame Pea"],
["DA6304", "Bamboo"],
["DA6A41", "Red Damask"],
["DA70D6", "Orchid"],
["DA8A67", "Copperfield"],
["DAA520", "Golden Grass"],
["DAECD6", "Zanah"],
["DAF4F0", "Iceberg"],
["DAFAFF", "Oyster Bay"],
["DB5079", "Cranberry"],
["DB9690", "Petite Orchid"],
["DB995E", "Di Serria"],
["DBDBDB", "Alto"],
["DBFFF8", "Frosted Mint"],
["DC143C", "Crimson"],
["DC4333", "Punch"],
["DCB20C", "Galliano"],
["DCB4BC", "Blossom"],
["DCD747", "Wattle"],
["DCD9D2", "Westar"],
["DCDDCC", "Moon Mist"],
["DCEDB4", "Caper"],
["DCF0EA", "Swans Down"],
["DDD6D5", "Swiss Coffee"],
["DDF9F1", "White Ice"],
["DE3163", "Cerise Red"],
["DE6360", "Roman"],
["DEA681", "Tumbleweed"],
["DEBA13", "Gold Tips"],
["DEC196", "Brandy"],
["DECBC6", "Wafer"],
["DED4A4", "Sapling"],
["DED717", "Barberry"],
["DEE5C0", "Beryl Green"],
["DEF5FF", "Pattens Blue"],
["DF73FF", "Heliotrope"],
["DFBE6F", "Apache"],
["DFCD6F", "Chenin"],
["DFCFDB", "Lola"],
["DFECDA", "Willow Brook"],
["DFFF00", "Chartreuse Yellow"],
["E0B0FF", "Mauve"],
["E0B646", "Anzac"],
["E0B974", "Harvest Gold"],
["E0C095", "Calico"],
["E0FFFF", "Baby Blue"],
["E16865", "Sunglo"],
["E1BC64", "Equator"],
["E1C0C8", "Pink Flare"],
["E1E6D6", "Periglacial Blue"],
["E1EAD4", "Kidnapper"],
["E1F6E8", "Tara"],
["E25465", "Mandy"],
["E2725B", "Terracotta"],
["E28913", "Golden Bell"],
["E292C0", "Shocking"],
["E29418", "Dixie"],
["E29CD2", "Light Orchid"],
["E2D8ED", "Snuff"],
["E2EBED", "Mystic"],
["E2F3EC", "Apple Green"],
["E30B5C", "Razzmatazz"],
["E32636", "Alizarin Crimson"],
["E34234", "Cinnabar"],
["E3BEBE", "Cavern Pink"],
["E3F5E1", "Peppermint"],
["E3F988", "Mindaro"],
["E47698", "Deep Blush"],
["E49B0F", "Gamboge"],
["E4C2D5", "Melanie"],
["E4CFDE", "Twilight"],
["E4D1C0", "Bone"],
["E4D422", "Sunflower"],
["E4D5B7", "Grain Brown"],
["E4D69B", "Zombie"],
["E4F6E7", "Frostee"],
["E4FFD1", "Snow Flurry"],
["E52B50", "Amaranth"],
["E5841B", "Zest"],
["E5CCC9", "Dust Storm"],
["E5D7BD", "Stark White"],
["E5D8AF", "Hampton"],
["E5E0E1", "Bon Jour"],
["E5E5E5", "Mercury"],
["E5F9F6", "Polar"],
["E64E03", "Trinidad"],
["E6BE8A", "Gold Sand"],
["E6BEA5", "Cashmere"],
["E6D7B9", "Double Spanish White"],
["E6E4D4", "Satin Linen"],
["E6F2EA", "Harp"],
["E6F8F3", "Off Green"],
["E6FFE9", "Hint of Green"],
["E6FFFF", "Tranquil"],
["E77200", "Mango Tango"],
["E7730A", "Christine"],
["E79F8C", "Tonys Pink"],
["E79FC4", "Kobi"],
["E7BCB4", "Rose Fog"],
["E7BF05", "Corn"],
["E7CD8C", "Putty"],
["E7ECE6", "Gray Nurse"],
["E7F8FF", "Lily White"],
["E7FEFF", "Bubbles"],
["E89928", "Fire Bush"],
["E8B9B3", "Shilo"],
["E8E0D5", "Pearl Bush"],
["E8EBE0", "Green White"],
["E8F1D4", "Chrome White"],
["E8F2EB", "Gin"],
["E8F5F2", "Aqua Squeeze"],
["E96E00", "Clementine"],
["E97451", "Burnt Sienna"],
["E97C07", "Tahiti Gold"],
["E9CECD", "Oyster Pink"],
["E9D75A", "Confetti"],
["E9E3E3", "Ebb"],
["E9F8ED", "Ottoman"],
["E9FFFD", "Clear Day"],
["EA88A8", "Carissma"],
["EAAE69", "Porsche"],
["EAB33B", "Tulip Tree"],
["EAC674", "Rob Roy"],
["EADAB8", "Raffia"],
["EAE8D4", "White Rock"],
["EAF6EE", "Panache"],
["EAF6FF", "Solitude"],
["EAF9F5", "Aqua Spring"],
["EAFFFE", "Dew"],
["EB9373", "Apricot"],
["EBC2AF", "Zinnwaldite"],
["ECA927", "Fuel Yellow"],
["ECC54E", "Ronchi"],
["ECC7EE", "French Lilac"],
["ECCDB9", "Just Right"],
["ECE090", "Wild Rice"],
["ECEBBD", "Fall Green"],
["ECEBCE", "Aths Special"],
["ECF245", "Starship"],
["ED0A3F", "Red Ribbon"],
["ED7A1C", "Tango"],
["ED9121", "Carrot Orange"],
["ED989E", "Sea Pink"],
["EDB381", "Tacao"],
["EDC9AF", "Desert Sand"],
["EDCDAB", "Pancho"],
["EDDCB1", "Chamois"],
["EDEA99", "Primrose"],
["EDF5DD", "Frost"],
["EDF5F5", "Aqua Haze"],
["EDF6FF", "Zumthor"],
["EDF9F1", "Narvik"],
["EDFC84", "Honeysuckle"],
["EE82EE", "Lavender Magenta"],
["EEC1BE", "Beauty Bush"],
["EED794", "Chalky"],
["EED9C4", "Almond"],
["EEDC82", "Flax"],
["EEDEDA", "Bizarre"],
["EEE3AD", "Double Colonial White"],
["EEEEE8", "Cararra"],
["EEEF78", "Manz"],
["EEF0C8", "Tahuna Sands"],
["EEF0F3", "Athens Gray"],
["EEF3C3", "Tusk"],
["EEF4DE", "Loafer"],
["EEF6F7", "Catskill White"],
["EEFDFF", "Twilight Blue"],
["EEFF9A", "Jonquil"],
["EEFFE2", "Rice Flower"],
["EF863F", "Jaffa"],
["EFEFEF", "Gallery"],
["EFF2F3", "Porcelain"],
["F091A9", "Mauvelous"],
["F0D52D", "Golden Dream"],
["F0DB7D", "Golden Sand"],
["F0DC82", "Buff"],
["F0E2EC", "Prim"],
["F0E68C", "Khaki"],
["F0EEFD", "Selago"],
["F0EEFF", "Titan White"],
["F0F8FF", "Alice Blue"],
["F0FCEA", "Feta"],
["F18200", "Gold Drop"],
["F19BAB", "Wewak"],
["F1E788", "Sahara Sand"],
["F1E9D2", "Parchment"],
["F1E9FF", "Blue Chalk"],
["F1EEC1", "Mint Julep"],
["F1F1F1", "Seashell"],
["F1F7F2", "Saltpan"],
["F1FFAD", "Tidal"],
["F1FFC8", "Chiffon"],
["F2552A", "Flamingo"],
["F28500", "Tangerine"],
["F2C3B2", "Mandys Pink"],
["F2F2F2", "Concrete"],
["F2FAFA", "Black Squeeze"],
["F34723", "Pomegranate"],
["F3AD16", "Buttercup"],
["F3D69D", "New Orleans"],
["F3D9DF", "Vanilla Ice"],
["F3E7BB", "Sidecar"],
["F3E9E5", "Dawn Pink"],
["F3EDCF", "Wheatfield"],
["F3FB62", "Canary"],
["F3FBD4", "Orinoco"],
["F3FFD8", "Carla"],
["F400A1", "Hollywood Cerise"],
["F4A460", "Sandy brown"],
["F4C430", "Saffron"],
["F4D81C", "Ripe Lemon"],
["F4EBD3", "Janna"],
["F4F2EE", "Pampas"],
["F4F4F4", "Wild Sand"],
["F4F8FF", "Zircon"],
["F57584", "Froly"],
["F5C85C", "Cream Can"],
["F5C999", "Manhattan"],
["F5D5A0", "Maize"],
["F5DEB3", "Wheat"],
["F5E7A2", "Sandwisp"],
["F5E7E2", "Pot Pourri"],
["F5E9D3", "Albescent White"],
["F5EDEF", "Soft Peach"],
["F5F3E5", "Ecru White"],
["F5F5DC", "Beige"],
["F5FB3D", "Golden Fizz"],
["F5FFBE", "Australian Mint"],
["F64A8A", "French Rose"],
["F653A6", "Brilliant Rose"],
["F6A4C9", "Illusion"],
["F6F0E6", "Merino"],
["F6F7F7", "Black Haze"],
["F6FFDC", "Spring Sun"],
["F7468A", "Violet Red"],
["F77703", "Chilean Fire"],
["F77FBE", "Persian Pink"],
["F7B668", "Rajah"],
["F7C8DA", "Azalea"],
["F7DBE6", "We Peep"],
["F7F2E1", "Quarter Spanish White"],
["F7F5FA", "Whisper"],
["F7FAF7", "Snow Drift"],
["F8B853", "Casablanca"],
["F8C3DF", "Chantilly"],
["F8D9E9", "Cherub"],
["F8DB9D", "Marzipan"],
["F8DD5C", "Energy Yellow"],
["F8E4BF", "Givry"],
["F8F0E8", "White Linen"],
["F8F4FF", "Magnolia"],
["F8F6F1", "Spring Wood"],
["F8F7DC", "Coconut Cream"],
["F8F7FC", "White Lilac"],
["F8F8F7", "Desert Storm"],
["F8F99C", "Texas"],
["F8FACD", "Corn Field"],
["F8FDD3", "Mimosa"],
["F95A61", "Carnation"],
["F9BF58", "Saffron Mango"],
["F9E0ED", "Carousel Pink"],
["F9E4BC", "Dairy Cream"],
["F9E663", "Portica"],
["F9EAF3", "Amour"],
["F9F8E4", "Rum Swizzle"],
["F9FF8B", "Dolly"],
["F9FFF6", "Sugar Cane"],
["FA7814", "Ecstasy"],
["FA9D5A", "Tan Hide"],
["FAD3A2", "Corvette"],
["FADFAD", "Peach Yellow"],
["FAE600", "Turbo"],
["FAEAB9", "Astra"],
["FAECCC", "Champagne"],
["FAF0E6", "Linen"],
["FAF3F0", "Fantasy"],
["FAF7D6", "Citrine White"],
["FAFAFA", "Alabaster"],
["FAFDE4", "Hint of Yellow"],
["FAFFA4", "Milan"],
["FB607F", "Brink Pink"],
["FB8989", "Geraldine"],
["FBA0E3", "Lavender Rose"],
["FBA129", "Sea Buckthorn"],
["FBAC13", "Sun"],
["FBAED2", "Lavender Pink"],
["FBB2A3", "Rose Bud"],
["FBBEDA", "Cupid"],
["FBCCE7", "Classic Rose"],
["FBCEB1", "Apricot Peach"],
["FBE7B2", "Banana Mania"],
["FBE870", "Marigold Yellow"],
["FBE96C", "Festival"],
["FBEA8C", "Sweet Corn"],
["FBEC5D", "Candy Corn"],
["FBF9F9", "Hint of Red"],
["FBFFBA", "Shalimar"],
["FC0FC0", "Shocking Pink"],
["FC80A5", "Tickle Me Pink"],
["FC9C1D", "Tree Poppy"],
["FCC01E", "Lightning Yellow"],
["FCD667", "Goldenrod"],
["FCD917", "Candlelight"],
["FCDA98", "Cherokee"],
["FCF4D0", "Double Pearl Lusta"],
["FCF4DC", "Pearl Lusta"],
["FCF8F7", "Vista White"],
["FCFBF3", "Bianca"],
["FCFEDA", "Moon Glow"],
["FCFFE7", "China Ivory"],
["FCFFF9", "Ceramic"],
["FD0E35", "Torch Red"],
["FD5B78", "Wild Watermelon"],
["FD7B33", "Crusta"],
["FD7C07", "Sorbus"],
["FD9FA2", "Sweet Pink"],
["FDD5B1", "Light Apricot"],
["FDD7E4", "Pig Pink"],
["FDE1DC", "Cinderella"],
["FDE295", "Golden Glow"],
["FDE910", "Lemon"],
["FDF5E6", "Old Lace"],
["FDF6D3", "Half Colonial White"],
["FDF7AD", "Drover"],
["FDFEB8", "Pale Prim"],
["FDFFD5", "Cumulus"],
["FE28A2", "Persian Rose"],
["FE4C40", "Sunset Orange"],
["FE6F5E", "Bittersweet"],
["FE9D04", "California"],
["FEA904", "Yellow Sea"],
["FEBAAD", "Melon"],
["FED33C", "Bright Sun"],
["FED85D", "Dandelion"],
["FEDB8D", "Salomie"],
["FEE5AC", "Cape Honey"],
["FEEBF3", "Remy"],
["FEEFCE", "Oasis"],
["FEF0EC", "Bridesmaid"],
["FEF2C7", "Beeswax"],
["FEF3D8", "Bleach White"],
["FEF4CC", "Pipi"],
["FEF4DB", "Half Spanish White"],
["FEF4F8", "Wisp Pink"],
["FEF5F1", "Provincial Pink"],
["FEF7DE", "Half Dutch White"],
["FEF8E2", "Solitaire"],
["FEF8FF", "White Pointer"],
["FEF9E3", "Off Yellow"],
["FEFCED", "Orange White"],
["FF0000", "Red"],
["FF007F", "Rose"],
["FF00CC", "Purple Pizzazz"],
["FF00FF", "Magenta / Fuchsia"],
["FF2400", "Scarlet"],
["FF3399", "Wild Strawberry"],
["FF33CC", "Razzle Dazzle Rose"],
["FF355E", "Radical Red"],
["FF3F34", "Red Orange"],
["FF4040", "Coral Red"],
["FF4D00", "Vermilion"],
["FF4F00", "International Orange"],
["FF6037", "Outrageous Orange"],
["FF6600", "Blaze Orange"],
["FF66FF", "Pink Flamingo"],
["FF681F", "Orange"],
["FF69B4", "Hot Pink"],
["FF6B53", "Persimmon"],
["FF6FFF", "Blush Pink"],
["FF7034", "Burning Orange"],
["FF7518", "Pumpkin"],
["FF7D07", "Flamenco"],
["FF7F00", "Flush Orange"],
["FF7F50", "Coral"],
["FF8C69", "Salmon"],
["FF9000", "Pizazz"],
["FF910F", "West Side"],
["FF91A4", "Pink Salmon"],
["FF9933", "Neon Carrot"],
["FF9966", "Atomic Tangerine"],
["FF9980", "Vivid Tangerine"],
["FF9E2C", "Sunshade"],
["FFA000", "Orange Peel"],
["FFA194", "Mona Lisa"],
["FFA500", "Web Orange"],
["FFA6C9", "Carnation Pink"],
["FFAB81", "Hit Pink"],
["FFAE42", "Yellow Orange"],
["FFB0AC", "Cornflower Lilac"],
["FFB1B3", "Sundown"],
["FFB31F", "My Sin"],
["FFB555", "Texas Rose"],
["FFB7D5", "Cotton Candy"],
["FFB97B", "Macaroni and Cheese"],
["FFBA00", "Selective Yellow"],
["FFBD5F", "Koromiko"],
["FFBF00", "Amber"],
["FFC0A8", "Wax Flower"],
["FFC0CB", "Pink"],
["FFC3C0", "Your Pink"],
["FFC901", "Supernova"],
["FFCBA4", "Flesh"],
["FFCC33", "Sunglow"],
["FFCC5C", "Golden Tainoi"],
["FFCC99", "Peach Orange"],
["FFCD8C", "Chardonnay"],
["FFD1DC", "Pastel Pink"],
["FFD2B7", "Romantic"],
["FFD38C", "Grandis"],
["FFD700", "Gold"],
["FFD800", "School bus Yellow"],
["FFD8D9", "Cosmos"],
["FFDB58", "Mustard"],
["FFDCD6", "Peach Schnapps"],
["FFDDAF", "Caramel"],
["FFDDCD", "Tuft Bush"],
["FFDDCF", "Watusi"],
["FFDDF4", "Pink Lace"],
["FFDEAD", "Navajo White"],
["FFDEB3", "Frangipani"],
["FFE1DF", "Pippin"],
["FFE1F2", "Pale Rose"],
["FFE2C5", "Negroni"],
["FFE5A0", "Cream Brulee"],
["FFE5B4", "Peach"],
["FFE6C7", "Tequila"],
["FFE772", "Kournikova"],
["FFEAC8", "Sandy Beach"],
["FFEAD4", "Karry"],
["FFEC13", "Broom"],
["FFEDBC", "Colonial White"],
["FFEED8", "Derby"],
["FFEFA1", "Vis Vis"],
["FFEFC1", "Egg White"],
["FFEFD5", "Papaya Whip"],
["FFEFEC", "Fair Pink"],
["FFF0DB", "Peach Cream"],
["FFF0F5", "Lavender blush"],
["FFF14F", "Gorse"],
["FFF1B5", "Buttermilk"],
["FFF1D8", "Pink Lady"],
["FFF1EE", "Forget Me Not"],
["FFF1F9", "Tutu"],
["FFF39D", "Picasso"],
["FFF3F1", "Chardon"],
["FFF46E", "Paris Daisy"],
["FFF4CE", "Barley White"],
["FFF4DD", "Egg Sour"],
["FFF4E0", "Sazerac"],
["FFF4E8", "Serenade"],
["FFF4F3", "Chablis"],
["FFF5EE", "Seashell Peach"],
["FFF5F3", "Sauvignon"],
["FFF6D4", "Milk Punch"],
["FFF6DF", "Varden"],
["FFF6F5", "Rose White"],
["FFF8D1", "Baja White"],
["FFF9E2", "Gin Fizz"],
["FFF9E6", "Early Dawn"],
["FFFACD", "Lemon Chiffon"],
["FFFAF4", "Bridal Heath"],
["FFFBDC", "Scotch Mist"],
["FFFBF9", "Soapstone"],
["FFFC99", "Witch Haze"],
["FFFCEA", "Buttery White"],
["FFFCEE", "Island Spice"],
["FFFDD0", "Cream"],
["FFFDE6", "Chilean Heath"],
["FFFDE8", "Travertine"],
["FFFDF3", "Orchid White"],
["FFFDF4", "Quarter Pearl Lusta"],
["FFFEE1", "Half and Half"],
["FFFEEC", "Apricot White"],
["FFFEF0", "Rice Cake"],
["FFFEF6", "Black White"],
["FFFEFD", "Romance"],
["FFFF00", "Yellow"],
["FFFF66", "Laser Lemon"],
["FFFF99", "Pale Canary"],
["FFFFB4", "Portafino"],
["FFFFF0", "Ivory"],
["FFFFFF", "White"]
]
}
ntc.init();