PK creative Spring Festival, I am participating in the “Spring Festival creative submission contest”, please see: Spring Festival creative submission Contest

preface

Today is the third day of the Chinese New Year, the Spring Festival holiday is almost half way! On the street, looking ahead, in front of the couplets, lanterns hung high, firecrackers on the ground…… They are all Chinese red! Write code looked down, this year I also wore a red outfit. It seems that red is the theme color of Spring Festival, so what color is your Spring Festival?

Today we are going to explore how to extract the main color of an image! Let’s take a look at the final result.

(Thanks to Visual China for the photo)

As we all know, an image is composed of blocks of pixels, and each pixel is represented by a set of data that can represent its color. There are many kinds of color space, such as RGB, CIELab, YUV and other methods we are familiar with. In front end development, we use RGB or RGBa to represent a color most often.

A picture’s main hue, also known as theme color, average color, color palette, they can be used as image features, features can be used for further analysis of the image. In the design industry, the choice and collocation of colors often affect the most intuitive visual feelings. For example, when a music platform plays music, the background color of the song is determined by its cover, which is more harmonious.

Here is a small advertisement, I wrote a small project before also used this scheme, interested friends can refer to this article. 👉 Vue family barrel based online music player (with online demo)

Color extraction algorithm

Some common algorithms

Median segmentation

The median segmentation algorithm firstly maps all pixels to RGB space and repeatedly slices subspace in this three-dimensional space. Finally, the mean value of pixels in the segmented space is taken as the extraction result. The largest block (the longest side has the largest length, or the largest volume, or the largest number of pixels) in all blocks are selected when the block is divided. The cutting point should be located in the edge direction, so that the pixels of the two blocks after segmentation are half of each position.

  1. Pixels are mapped to RGB space

  1. Block calculation

  1. The median segmentation

  1. Repeatedly segmentation

  1. Calculates the average color of blocks

Octree algorithm

The core idea of the octree algorithm is to divide the color space with the octree, and then merge the leaf nodes to gradually gather the color (quantization). For the explanation of the octree, please refer to “What is the Octree Algorithm for Game Scene Management?”

K-means clustering algorithm

The idea of k-means clustering is very simple and can be divided into the following steps:

  1. Take the initial K centers of mass;

  2. Classify all samples according to their distance from the center of mass;

  3. Recalculate the center of mass to determine whether to exit the condition:

    • The distance between the two centroids is small enough to satisfy the exit condition;
    • If you do not exit, return to Step 2.

A general idea

The algorithms above are daunting.

When I see such a requirement, the first thing I think of is to get the RGB value of all the pixels of the whole image, and then carry out a sort according to the number of RGB combinations, and the most times is the main color of the image!

const img = this.$refs.img
const that = this
img.onload = function(){
    const w = this.width
    const h = this.height
    // Create canvas
    const canvas = document.createElement('canvas')
    canvas.width = w
    canvas.height = h
    // Draw the picture on the canvas
    const context = canvas.getContext('2d')
    context.drawImage(this.0.0)
    // Get pixel RGB data
    let pxArr = context.getImageData(0.0,w,h).data
    pxArr = [...pxArr]
    // Count pixel colors
    const colorList = {}
    let i = 0
    while(i < pxArr.length){
        const r = pxArr[i]
        const g = pxArr[i+1]
        const b = pxArr[i+2]
        i = i + 4
        const key = [r,g,b].join(",")
        key in colorList ? ++colorList[key] : (colorList[key] = 1)}// Sort the statistics
    let arr = []
    for(let key in colorList){
        arr.push({
            rgb: `rgb(${key}) `.num: colorList[key]
        })
    }
    arr = arr.sort((a,b) = >b.num - a.num)
Copy the code

After the above operations, we get the RGB value array in descending order according to the number of occurrences. We select the first ten as our palette data, and the results are as follows:

We can see that, as seen by the naked eye, the largest color in this image is indeed this red color, which is consistent with the output of our algorithm. However, it is difficult to distinguish the differences between them with naked eyes, and the results of the algorithm can not represent the overall characteristics of the image. Therefore, this method can only get the most densely distributed color set in the image.

A good plug-in

ColorThief is a plugin for extracting Dominant Color and Palette from images, developed using JavaScript and Canvas. It is also very simple to use, as shown below.

GetColor ()

getColor(sourceImage[, quality])
returns {r: num, g: num, b: num}
Copy the code
let colorThief = new ColorThief();
colorThief.getColor(sourceImage);
Copy the code

Extract the image palette getPalette()

getPalette(sourceImage[, colorCount, quality])
returns [ [num, num, num], [num, num, num], ... ]
Copy the code
// Here we extract 8 colors
let colorThief = new ColorThief();
colorThief.getPalette(sourceImage, 8);
Copy the code

It is reported that the principle of this plug-in is based on the median segmentation method.

Project address: github.com/lokesh/colo…

System implementation

The core part of our project — color extraction algorithm has been implemented, the next is to build the start page.

Images are uploaded and displayed

Use the Input tag to select the local image, use FileReader to read the selected image, decode it as Base64, and finally display it on the page.

<input type="file" @change="getFile">

<img :src="imgSrc" class="img" ref="img">
Copy the code
getFile(e) {
    let that = this
    let files = e.target.files[0]

    if(! e || !window.FileReader) return
    let reader = new FileReader()
    reader.readAsDataURL(files)
    reader.onloadend = function(){
        that.imgSrc = this.result
    }
}
Copy the code

Color to heavy

If the source imageVery few colorsAnd less than 10, which we set artificially, then the final result will have repeated colors with the same RGB value, as shown in the figure below.What we’re thinking here is going to begetPalette()The return result of this function is de-duplicated. But the elements in this array are also arrays, usingSetIt’s hard to do a de-duplication of an array element, and since you can’t determine an array element, you can always determine a string. weWe start by converting the array elements to strings, get an array of strings, and then delete, finally will be deletedRestores a string to an arrayIt’ll be ok.

// Extract 10 colors
let colors = [...colorThief.getPalette(img,10)]
let colorsStrArr = []

/ / RGB to heavy
colors.forEach(item= > {
    colorsStrArr.push(item.join("-"))
})
colorsStrArr = [...new Set(colorsStrArr)]
colorsStrArr.forEach(item= > {
    this.colors.push(this.toRGB(item.split("-")))})Copy the code

There are no duplicate cards in the palette.

The HTML is saved as a picture

At first, I thought of converting the final generated results directly into pictures and saving them locally, but in the process of surfing online, I found a plug-in: HTML2Canvas, which can convert HTML pages into canvas and save pictures with canvas!

Project address: github.com/niklasvh/ht…

It is also very, very simple to use!

html2canvas(document.querySelector('.card-wrap')).then(canvas= > {
    document.appendChild(canvas)
})
Copy the code

So we have a canvas, but the resolution is a little bit personal.

So with canvas how do we save a canvas as a picture?

  • Right click on the canvas and you’ll have one saved as an image.

But I don’t want to insert the canvas on the page, so I’ll use the A tag.

  • Dynamic settingaOf the labelhrefProperties forcanvas.toDataURL("image/png")And then simulate clicking.
<a href="#" ref="downloadA" style="opacity:0" download="My Spring Festival Color"></a>
Copy the code
downloadResult(){
    html2canvas(document.querySelector('.card-wrap')).then(canvas= > {
        this.$refs.downloadA.href = canvas.toDataURL("image/png");
        this.$refs.downloadA.click()
    })
},
Copy the code

Online experience 😀

Experience the address

The resources

  1. Image color extraction – Square spring is very sweet
  2. JavaScript extract image theme color – Wang_Yi
  3. Theme color extraction – Gesangs
  4. Color Thief