/** ColorModule holds the functions used to implement the basic-mode palette. * */ const ColorModule = (() => { // Array containing the colours of the current palette let currentPalette = []; // Reference to the HTML palette const coloursList = document.getElementById("palette-list"); // Binding events to callbacks console.info("Initialized Color Module.."); document.getElementById('jscolor-hex-input').addEventListener('change',colorChanged, false); document.getElementById('jscolor-hex-input').addEventListener('input', colorChanged, false); document.getElementById('add-color-button').addEventListener('click', addColorButtonEvent, false); // Making the colours in the HTML menu sortable new Sortable(document.getElementById("colors-menu"), { animation:100, filter: ".noshrink", draggable: ".draggable-colour", onEnd: function() {dragging = false} }); /** Changes all of one color to another after being changed from the color picker * * @param {*} colorHexElement The element that has been changed * @returns */ function colorChanged(colorHexElement) { // Get old and new colors from the element const hexElement = colorHexElement.target; const hexElementValue = hexElement.value; const newColor = Color.hexToRgb(hexElementValue); const oldColor = hexElement.oldColor; //if the color is not a valid hex color, exit this function and do nothing const newColorHex = hexElementValue.toLowerCase(); if (/^[0-9a-f]{6}$/i.test(newColorHex) == false) return; currentPalette.splice(currentPalette.indexOf("#" + newColor), 1); newColor.a = 255; //save undo state new HistoryState().EditColor(hexElementValue.toLowerCase(), Color.rgbToHex(oldColor)); //get the currently selected color const currentlyEditedColor = document.getElementsByClassName('jscolor-active')[0]; const duplicateColorWarning = document.getElementById('duplicate-color-warning'); //check if selected color already matches another color colors = document.getElementsByClassName('color-button'); //loop through all colors in palette for (var i = 0; i < colors.length; i++) { //if generated color matches this color if (newColorHex == colors[i].jscolor.toString()) { //if the color isnt the one that has the picker currently open if (!colors[i].parentElement.classList.contains('jscolor-active')) { //console.log('%cColor is duplicate', colorCheckingStyle); //show the duplicate color warning duplicateColorWarning.style.visibility = 'visible'; //shake warning icon duplicateColorWarning.classList.remove('shake'); void duplicateColorWarning.offsetWidth; duplicateColorWarning.classList.add('shake'); //exit function without updating color return; } } } //if the color being edited has a duplicate color warning, remove it duplicateColorWarning.style.visibility = 'hidden'; currentlyEditedColor.firstChild.jscolor.fromString(newColorHex); ColorModule.replaceAllOfColor(oldColor, newColor); //set new old color to changed color hexElement.oldColor = newColor; currentPalette.push('#' + newColorHex); //if this is the current color, update the drawing color if (hexElement.colorElement.parentElement.classList.contains('selected')) { for (let i=1; i { //hide edit button event.target.parentElement.lastChild.classList.add('hidden'); //show jscolor picker, if basic mode is enabled if (pixelEditorMode == 'Basic') event.target.parentElement.firstChild.jscolor.show(); else Dialogue.showDialogue("palette-block", false); }); return listItem; } /** Deletes a color from the palette * * @param {*} color A string in hex format or the HTML element corresponding to the color * that should be removed. */ function deleteColor (color) { const logStyle = 'background: #913939; color: white; padding: 5px;'; //if color is a string, then find the corresponding button if (typeof color === 'string') { //get all colors in palette colors = document.getElementsByClassName('color-button'); //loop through colors for (var i = 0; i < colors.length; i++) { //console.log(color,'=',colors[i].jscolor.toString()); if (color == colors[i].jscolor.toString()) { //set color to the color button color = colors[i]; break; } } //if the color wasn't found if (typeof color === 'string') { //exit function return; } } //hide color picker color.jscolor.hide(); //find lightest color in palette var colors = document.getElementsByClassName('color-button'); var lightestColor = [0,null]; for (var i = 0; i < colors.length; i++) { //get colors lightness var lightness = Color.rgbToHsl(colors[i].jscolor.toRgb()).l; //if not the color we're deleting if (colors[i] != color) { //if lighter than the current lightest, set as the new lightest if (lightness > lightestColor[0]) { lightestColor[0] = lightness; lightestColor[1] = colors[i]; } } } //replace deleted color with lightest color ColorModule.replaceAllOfColor(color.jscolor.toString(),lightestColor[1].jscolor.toString()); //if the color you are deleting is the currently selected color if (color.parentElement.classList.contains('selected')) { //set current color TO LIGHTEST COLOR lightestColor[1].parentElement.classList.add('selected'); currentLayer.context.fillStyle = '#'+lightestColor[1].jscolor.toString(); } //delete the element colorsMenu.removeChild(color.parentElement); } /** Replaces all of a single color on the canvas with a different color * * @param {*} oldColor Old colour in {r,g,b} object format * @param {*} newColor New colour in {r,g,b} object format */ function replaceAllOfColor (oldColor, newColor) { //convert strings to objects if nessesary if (typeof oldColor === 'string') oldColor = Color.hexToRgb(oldColor); if (typeof newColor === 'string') newColor = Color.hexToRgb(newColor); //create temporary image from canvas to search through var tempImage = currentLayer.context.getImageData(0, 0, canvasSize[0], canvasSize[1]); //loop through all pixels for (var i=0;i 0) colors[0].parentElement.remove(); var lightestColor = new Color("hex", '#000000'); var darkestColor = new Color("hex", '#ffffff'); // Adding all the colours in the array for (var i = 0; i < paletteColors.length; i++) { var newColor = new Color("hex", paletteColors[i]); var newColorElement = ColorModule.addColor(newColor.hex); var newColRgb = newColor.rgb; var lightestColorRgb = lightestColor.rgb; if (newColRgb.r + newColRgb.g + newColRgb.b > lightestColorRgb.r + lightestColorRgb.g + lightestColorRgb.b) lightestColor = newColor; var darkestColorRgb = darkestColor.rgb; if (newColRgb.r + newColRgb.g + newColRgb.b < darkestColorRgb.r + darkestColorRgb.g + darkestColorRgb.b) { //remove current color selection var selectedColor = document.querySelector('#colors-menu li.selected'); if (selectedColor) selectedColor.classList.remove('selected'); //set as current color newColorElement.classList.add('selected'); darkestColor = newColor; } } //prepend # if not present if (!darkestColor.hex.includes('#')) darkestColor.hex = '#' + darkestColor.hex; //set as current color currentLayer.context.fillStyle = darkestColor.hex; } /** Creates the palette with the colours used in all the layers * */ function createPaletteFromLayers() { let colors = {}; for (let i=0; i= settings.maxColorsOnImportedImage) { alert('The image loaded seems to have more than '+settings.maxColorsOnImportedImage+' colors.'); break; } } } } } } //create array out of colors object let colorPaletteArray = []; for (let color in colors) { if (colors.hasOwnProperty(color)) { colorPaletteArray.push('#'+Color.rgbToHex(colors[color])); } } //create palette from colors array createColorPalette(colorPaletteArray); } return { getCurrentPalette, addColor, deleteColor, replaceAllOfColor, addToSimplePalette, resetPalette, createColorPalette, createPaletteFromLayers } })();