What we have
Lets see what options we have when we want to get color palettes out of images and platform is React-Native.
- We can use native 3rd party libraries out there in community.
- We can build our own.
When we say build our own then lets see do we have any easy way out as far as React-Native is concern. I can see that we can do something with Canvas
and now there is a new doubt that “how to use Canvas
in React-Native”? And seems React-Native have solution for this also which is React-Native WebView
component. Let see how we can use this.
React-Native WebView
First create a
WebView
<WebView
source=
javaScriptEnabled={true}
onMessage={imageResponseCallback}
/>
-
In source we will provide canvas html
-
JavaScriptEnabled: true
as we will be creatingCanvas
using JS for that JavaScript need to be enabled. -
onMessage
will be used for response callback
Lets provide source as
html
which will haveCanvas
let canvasHtml = `
<!DOCTYPE html>
<html>
<script>
const interval = setInterval(() => {
const img = document.getElementById('__colorPickerCanvasImage');
if (img.src.length != 0) {
getColorPalettes();
clearInterval(interval);
}
}, 10);
</script>
<body>
<img src=''
id='__colorPickerCanvasImage'
onload='getColorPalettes()' />
</body>
</html>
`;
};
- So, basically we are planning to render base64 image inside
Canvas
. - Get color
rgba
values - Sending message with
rgbaValues
to callback handler
Now lets render image, for getting color palettes out of image we need to provide base64 value of image. You can use any online original image to base64 image. I will be using base64-image.de.
function getColorPalettes() {
const imageElement = document.getElementById('__colorPickerCanvasImage');
const canvas = document.createElement('canvas');
const canvasContext = canvas.getContext && canvas.getContext('2d');
if (!canvasContext) return [0, 0, 0, 1];
let imageWidth = canvas.width = imageElement.naturalWidth
|| imageElement.offsetWidth
|| imageElement.width;
const imageHeight = canvas.height = imageElement.naturalHeight
|| imageElement.offsetHeight
|| imageElement.height;
canvasContext.drawImage(imageElement, 0, 0);
....
}
Get Color Palettes out of image
function getAllPalettes(width, height, context) {
let distinctPalettes = [];
// loop through each and every pixels of image
for (let i=0; i<=height; i++) {
for (let j=0; j<=width; j++) {
try {
data = context.getImageData(i, j, 1, 1);
if (data.data.toString().trim() !== '0,0,0,0') {
distinctPalettes.push(data.data);
}
} catch(e) {
console.log(e);
}
}
}
return distinctPalettes;
}
// get dominant color palettes
function getDominantPalettes(allPalettes, distinctCount, colorType) {
const combinations = getPaletteOccurrences(allPalettes);
let palettes = combinations[0];
let occurrences = combinations[1];
const dominantPalettes = [];
while (distinctCount) {
let dominant = 0, dominantKey = 0;
occurrences.forEach((v, k) => {
if (v > dominant) {
dominant = v;
dominantKey = k;
}
});
dominantPalettes.push(palettes[dominantKey]);
palettes.splice(dominantKey, 1);
occurrences.splice(dominantKey, 1);
distinctCount--;
}
return dominantPalettes;
}
// get palette occurrences
function getPaletteOccurrences(palettes) {
let paletteList = [], occurrenceList = [], previousPalette;
palettes.sort();
palettes.forEach((palette, key) => {
if (palette.toString() !== previousPalette) {
paletteList.push(palette);
occurrenceList.push(1);
} else {
occurrenceList[occurrenceList.length-1]++;
}
previousPalette = palettes[key].toString();
});
return [paletteList, occurrenceList];
}
In last we want to notify UI with retrieved color palettes. We will be using
window.postMessage
for sending data to callback method.
let palette = getDominantPalettes(getAllPalettes(imageWidth, imageHeight, canvasContext), 3, 'rgba');
window.postMessage(JSON.stringify({'message':'imageColorPicker','payload':palette}));
Complete project
- Fork complete project here
- Project demo