drawio/war/plugins/voice.js

4031 lines
115 KiB
JavaScript
Raw Normal View History

2016-09-06 14:07:11 +00:00
/**
* 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) {
// 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 = 50;
// 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
var td = ui.getOrCreateVoiceButton();
if (td != null)
{
// Installs listener for start/stop listen
mxEvent.addListener(td, 'click', function(evt)
{
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="' + ui.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="' + ui.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)
{
ui.menus.addCheckmark(item);
}
})(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/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////VEFHAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAU291bmRib3kuZGUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAMjAwNGh0dHA6Ly93d3cuc291bmRib3kuZG
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="' + ui.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();