CSS Paint API: A CSS-like Drawing Board

CSS Paint API: A CSS-like Drawing Board

1. Use Canvas images as CSS background images

The CSS Paint API can be simply understood as (not actually equivalent to) using the Canvas canvas as the background image of a normal element.

That is, background-image of CSS is a Canvas. We can use most of the Canvas APIs to draw various complex and interesting graphic effects to enrich the visual presentation of web page elements in a more efficient way. For example, the blue button is not just a blue background, it also has a white cloud effect on it, which is great!

2. A simple case to understand CSS Paint API

For example, we want to create a transparent image background. Something like this:

The complete CSS code and JS code are as follows:

.box {
    width: 180px; height: 180px;
    /* transparent-grid name yourself*/
    background-image: paint(transparent-grid);
}

Then the JS for drawing graphics must be imported as a module. For example, create a file named paint-grid.js and import it on the page:

if (window.CSS) {
    CSS.paintWorklet.addModule('paint-grid.js');
}

The paint-grid.js file code is as follows:

// transparent-grid naming and CSS correspondence registerPaint('transparent-grid', class {
    paint(context, size) {
       // Here is the drawing code....
    }
});

The above is the fixed routine used by CSS Paint API:

  1. paint(abc) in CSS;
  2. JS add module CSS.paintWorklet.addModule('xxx.js');
  3. The code routine in xxx.js is fixed, just write the drawing code in the comment position below;
registerPaint('abc', class {
    paint(context, size, properties) {
       // Drawing code goes here....
    }
});

The two parameters in paint(context, size) can be briefly introduced:

context

For drawing context, the full name is PaintRenderingContext2D, which is a close relative of Canvas's CanvasRenderingContext2D. All APIs are from Canvas and are exactly the same. However, due to security restrictions, some APIs in Canvas cannot be used. The available and unavailable APIs are shown in the following table:

Paint Available APIs Paint Unavailable API
CanvasState CanvasImageData
CanvasTransform CanvasUserInterface
Canvas Compositing CanvasText
CanvasImageSmoothing CanvasTextDrawingStyles
CanvasFillStrokeStyles
CanvasShadowStyles
CanvasRect
CanvasDrawPath
CanvasDrawImage
CanvasPathDrawingStyles
CanvasPath

size

size is an object containing the drawing size. The data structure is as follows:

{
    width: 180,
    height: 180
}

The size of size is affected by the size of background-size property. Therefore, for repeated backgrounds, you can use background-repeat to perform a tile loop without having to loop in the drawn JS code. For example, the demo effect that is about to be shown below can also be achieved in this way, CSS part:

.box {
    width: 180px; height: 180px;
    background-image: paint(transparent-grid);
    background-size: 16px 16px;
}

Then, in paint-grid.js, you only need to fill white-gray-gray-white, four grids are enough, no loop is needed. properties can be used to obtain the obtained CSS properties and property values, including CSS variable values; and some other parameters.

Seeing is believing, you can click here: CSS Paint API draws a transparent grid as a background demo (currently only works on Chrome)

The effect is as follows:

The complete drawing code in paint-grid.js is as follows:

registerPaint('transparent-grid', class {
    paint(context, size) {
        // Two grid colors var color1 = '#fff', color2 = '#eee';
        // Grid size var units = 8;
        // Loop through the horizontal axis for (var x = 0; x < size.width; x += units) {
            for (var y = 0; y < size.height; y += units) {
                context.fillStyle = (x + y) % (units * 2) === 0 ? color1 : color2;
                context.fillRect(x, y, units, units);
            }
        }
    }
});

Replenish:
For repeating backgrounds like grid, you can use background-repeat to make them tiled and looped. You don’t have to loop in the drawn JS code, but you need to use background-size property to change the drawing size. For example, the demo effect above can also be achieved like this, CSS part:

.box {
    width: 180px; height: 180px;
    background-image: paint(transparent-grid);
    background-size: 16px 16px;
}

Then, in paint-grid.js, you only need to fill white-gray-gray-white, four grids are enough, no loop is needed.

registerPaint('transparent-grid', class {
    paint(context, size) {
        // Two grid colors var color1 = '#fff', color2 = '#eee';
        // Two white grids context.fillStyle = color1;
        context.fillRect(0, 0, 8, 8);
        context.fillRect(8, 8, 8, 8);
        // Two gray grids context.fillStyle = color1;
        context.fillRect(0, 4, 8, 8);
        context.fillRect(4, 0, 8, 8);
    }
});

Be more understandable.

3. CSS variables make Paint API shine

The above example shows the basic use of CSS Paint API. However, although it looks trendy, it does not reflect the outstanding features of CSS Paint API.

Think about it, if I directly use JS plus Canvas API to draw a grid pattern, convert it to Base64, and directly display it as the background image of the element, wouldn’t it be the same effect, and the compatibility is better (IE9+ supported), and all Canvas APIs can be used without restrictions. In comparison, there is absolutely no reason to use the CSS Paint API!

That’s right! If we just need a static background, it is better to draw it directly on Canvas and then convert it into a Base64 image (
toDataURL() method) or Blob image (toBlob() method).

The advantage of CSS Paint API is that as a CSS property value, the rendering is real-time and automatically redrawn with the browser. Therefore, as long as our drawing is associated with CSS variables, all rendering effects will be refreshed and redrawn in real time, which is awesome!

Still using the transparent grid example above, we can extract the grid color and grid size as CSS variables as follows:

.box {
    width: 180px; height: 180px;
    --color1: #fff;
    --color2: #eee;
    --units: 8;
    background: paint(custom-grid);
}

We can obtain these defined variables when drawing, as shown below:

registerPaint('custom-grid', class {
    // Get 3 variables static get inputProperties() {
        return [
            '--color1',
            '--color2',
            '--units'
        ]
    }
    paint(context, size, properties) {
        // Two grid colors var color1 = properties.get('--color1').toString();
        var color2 = properties.get('--color2').toString();
        // Grid size var units = Number(properties.get('--units'));
        // Drawing code, same as before...
    }
});

The static effect is the same:

However, if we modify the variable value defined in the CSS code, we can see the real-time change effect of the Paint background image (see the GIF below):

Seeing is believing. You can click here: CSS variables plus Paint API to draw transparent grid demo

No additional timer is required, real-time rendering, and control is very convenient.

With the CSS Properties & Values ​​API, we can register all variables such as --units as legal CSS properties. Then, we can use transition and animation properties to control the movement and animation of the Paint background image with pure CSS, and the effect of floating clouds on the button is no problem at all.

This case will be introduced later.

Houdini, compatible with other

The CSS Paint API introduced in this article is part of CSS Houdini, and the CSS Properties & Values ​​API mentioned at the end is also part of the API currently supported by Chrome. CSS Houdini can customize CSS properties, layout, etc., and its future potential is limitless.

Due to compatibility issues, if you want to use the CSS Paint API in an actual project, you also need to do compatibility processing, for example:

.box {
    width: 180px; height: 180px;
    background: url();
    background: paint(transparent-grid, whatever);
}

CSS Paint API is more suitable for dynamic scenes and for meeting requirements that require real-time drawing and rendering. If it is a purely static display, it can be achieved directly using JS plus Canvas. There is no need to use technology for the sake of technology.

There are many things to play with CSS Houdini and its related new technologies. I will introduce more about them in the future. This article ends here. Thank you for reading!

Reference Documentation

https://www.w3.org/TR/css-paint-api-1/https://css-houdini.rocks/

Finally, finally, let me talk about something else that may be of interest to you. For the transparent grid effect in this article, the best way to achieve it is to draw the CSS background directly, using linear gradients and CSS3 multiple backgrounds.

The code is as follows:

.box {
    width: 180px; height: 180px;
    background-color: #fff;
    background-image: linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee), linear-gradient(45deg, #eee 25%, transparent 25%, transparent 75%, #eee 75%, #eee);
    background-size: 16px 16px;
    background-position: 0 0, 8px 8px;
}

The real-time effect is as follows:

Size control is very convenient and naturally supports animation .

This article is an original article, and knowledge points will be updated frequently and some errors will be corrected. Therefore, please keep the original source when reprinting to facilitate tracing, avoid being misled by outdated and erroneous knowledge, and have a better reading experience.
This article address: https://www.zhangxinxu.com/wordpress/?p=8204

This is the end of this article about the CSS Paint API, the drawing board of the CSS world. For more relevant CSS Paint API content, please search 123WORDPRESS.COM’s previous articles or continue to browse the related articles below. I hope everyone will support 123WORDPRESS.COM in the future!

<<:  Solve the problem of garbled data in MySQL database migration

>>:  Solve the error problem of warnings potentially fixable with the `--fix` option when running a Vue project

Recommend

Implementation of CSS3 button border animation

First look at the effect: html <a href="#...

How are Vue components parsed and rendered?

Preface This article will explain how Vue compone...

How to add and delete unique indexes for fields in MySQL

1. Add PRIMARY KEY (primary key index) mysql>A...

Install multiple versions of PHP for Nginx on Linux

When we install and configure the server LNPM env...

Vue implements dynamic circular percentage progress bar

Recently, when developing a small program, I enco...

Summary of Common Problems with Mysql Indexes

Q1: What indexes does the database have? What are...

A brief understanding of the three uses of standard SQL update statements

1. Environment: MySQL-5.0.41-win32 Windows XP Pro...

Detailed explanation of MySQL phantom reads and how to eliminate them

Table of contents Transaction Isolation Level Wha...

A brief discussion on the types of node.js middleware

Table of contents Overview 1. Application-level m...

Introduction to Jenkins and how to deploy Jenkins with Docker

1. Related concepts 1.1 Jenkins Concepts: Jenkins...

Detailed explanation of writing multiple conditions of CSS: not

The :not pseudo-class selector can filter element...

WebStorm cannot correctly identify the solution of Vue3 combined API

1 Problem Description Vue3's combined API can...