If only the Canvas tag is left

If only the Canvas tag is left

[[420999]]

1. Background

If only the canvas tag is left, how can we draw the content on the page? This may be a false proposition, but using canvas can indeed help accomplish many things. Today, we will use canvas+AST syntax tree to build an information flow style.

2. Drawing Process

The entire drawing process is divided into three parts: basic elements, AST syntax tree, and main function class. Basic elements refer to pictures, text, rectangles, circles, etc.; the AST syntax tree here is a js object containing some properties; the main function class refers to the externally exposed interface, which is called to achieve the final drawing.

2.1 Basic Elements

No matter how complex something is, it is definitely made up of a series of simple elements. For example, a car is definitely made up of some simple mechanical parts; a computer is also made up of resistors, capacitors and other parts. Web pages are no exception, and are also made up of text, pictures, rectangles, etc.

2.1.1 Loading images

Pictures are the soul element of a page and occupy most of the space on the page.

  1. class DrawImage {
  2. constructor(ctx, imageObj) {
  3. this.ctx = ctx;
  4. this.imageObj = imageObj;
  5. }
  6.  
  7. draw() {
  8. const {centerX, centerY, src, sx = 1, sy = 1} = this.imageObj;
  9. const img = new Image();
  10. img.onload = () => {
  11. const imgWidth = img.width;
  12. const imgHeight = img.height;
  13. this.ctx.save();
  14. this.ctx.scale(sx, sy);
  15. this.ctx.drawImage(img, centerX - imgWidth * sx / 2, centerY - imgHeight * sy / 2);
  16. this.ctx.restore();
  17. };
  18. img.src = src;
  19. }
  20. }

2.1.2 Drawing Text

Text can improve the readability of the page, allowing everyone who observes the page to quickly understand the idea of ​​the page.

  1. class DrawText {
  2. constructor(ctx, textObj) {
  3. this.ctx = ctx;
  4. this.textObj = textObj;
  5. }
  6.  
  7. draw() {
  8. const {x, y, font, content, lineHeight = 20, width, fillStyle = '#000000' , textAlign = 'start' , textBaseline = 'middle' } = this.textObj;
  9. const branchesContent = this.getBranchsContent(content, width);
  10. this.ctx.save();
  11. this.ctx.fillStyle = fillStyle;
  12. this.ctx.textAlign = textAlign;
  13. this.ctx.textBaseline = textBaseline;
  14. this.ctx.font = font;
  15. branchesContent.forEach((branchContent, index ) => {
  16. this.ctx.fillText(branchContent, x, y + index * lineHeight);
  17. });
  18. this.ctx.restore();
  19. }
  20.  
  21. getBranchsContent(content, width) {
  22. if (!width) {
  23. return [content];
  24. }
  25. const charArr = content.split( '' );
  26. const branchesContent = [];
  27. let tempContent = '' ;
  28. charArr.forEach( char => {
  29. if (this.ctx.measureText(tempContent).width < width && this.ctx.measureText(tempContent + char ).width <= width) {
  30. tempContent += char ;
  31. }
  32. else {
  33. branchesContent.push(tempContent);
  34. tempContent = '' ;
  35. }
  36. });
  37. branchesContent.push(tempContent);
  38. return branchesContent;
  39. }
  40. }

2.1.3 Drawing a rectangle

Rectangular elements can be combined with text and other elements to achieve unexpected effects.

  1. class DrawRect {
  2. constructor(ctx, rectObj) {
  3. this.ctx = ctx;
  4. this.rectObj = rectObj;
  5. }
  6.  
  7. draw() {
  8. const {x, y, width, height, fillStyle, lineWidth = 1} = this.rectObj;
  9. this.ctx.save();
  10. this.ctx.fillStyle = fillStyle;
  11. this.ctx.lineWidth = lineWidth;
  12. this.ctx.fillRect(x, y, width, height);
  13. this.ctx.restore();
  14. }
  15. }

2.1.4 Drawing a circle

The circle and rectangle play the same role and are also relatively important roles on the page.

  1. class DrawCircle {
  2. constructor(ctx, circleObj) {
  3. this.ctx = ctx;
  4. this.circleObj = circleObj;
  5. }
  6.  
  7. draw() {
  8. const {x, y, R, startAngle = 0, endAngle = Math.PI * 2, lineWidth = 1, fillStyle} = this.circleObj;
  9. this.ctx.save();
  10. this.ctx.lineWidth = lineWidth;
  11. this.ctx.fillStyle = fillStyle;
  12. this.ctx.beginPath();
  13. this.ctx.arc(x, y, R, startAngle, endAngle);
  14. this.ctx.closePath();
  15. this.ctx.fill();
  16. this.ctx.restore();
  17. }
  18. }

2.2 AST Tree

AST is an abstract representation of the syntax structure of the source code. It represents the syntax structure of the programming language in a tree-like form, and each node on the tree represents a structure in the source code. For example, in Vue, the template syntax is converted to AST, and then the AST is converted to HTML structure. When we use canvas to draw a page, we also use AST to represent the content of the page. The types implemented are rect, img, text, and circle.

The content to be drawn this time includes the static page part and the animation part, so two canvases will be used to implement it. Each canvas will correspond to an AST tree, namely the static part AST tree and the dynamic part AST tree.

2.2.1 Static Part AST Tree

The AST tree of the static part of the page drawn this time is as follows, including rectangles, pictures, and text.

  1. const graphicAst = [
  2. {
  3. type: 'rect' ,
  4. x: 0,
  5. y: 0,
  6. width: 1400,
  7. height: 400,
  8. fillStyle: '#cec9ae'  
  9. },
  10. {
  11. type: 'img' ,
  12. centerX: 290,
  13. centerY: 200,
  14. sx: 0.9,
  15. sy: 0.9,
  16. src: 'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Finews.gtimg.com%2Fnewsapp_match%2F0%2F11858683821%2F0.jpg&refer=http% 3A%2F%2Finews.gtimg.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=jpeg?sec=1622015341&t=cc1bd95777dfa37d88c48bb6e179778e'  
  17. },
  18. {
  19. type: 'text' ,
  20. x: 600,
  21. y: 60,
  22. textAlign: 'start' ,
  23. textBaseline: 'middle' ,
  24. font: 'normal 40px serif' ,
  25. lineHeight: 50,
  26. width: 180,
  27. fillStyle: '#000000' ,
  28. content: 'Grey Wolf is the best wolf. He dreams of eating sheep every day, but his dream has never come true. However, he never gives up.'  
  29. },
  30. {
  31. type: 'text' ,
  32. x: 600,
  33. y: 170,
  34. textAlign: 'start' ,
  35. textBaseline: 'middle' ,
  36. font: 'normal 30px serif' ,
  37. lineHeight: 50,
  38. width: 180,
  39. fillStyle: '#7F7F7F' ,
  40. content: 'Come on for Big Bad Wolf, cheer for Big Bad Wolf, 😄'  
  41. },
  42. {
  43. type: 'text' ,
  44. x: 1200,
  45. y: 360,
  46. textAlign: 'start' ,
  47. textBaseline: 'ideographic' ,
  48. font: 'normal 30px serif' ,
  49. lineHeight: 50,
  50. width: 180,
  51. fillStyle: '#949494' ,
  52. content: 'Reading'  
  53. },
  54. {
  55. type: 'text' ,
  56. x: 1260,
  57. y: 363,
  58. textAlign: 'start' ,
  59. textBaseline: 'ideographic' ,
  60. font: 'normal 30px serif' ,
  61. lineHeight: 50,
  62. width: 180,
  63. fillStyle: '#949494' ,
  64. content: '520'  
  65. }
  66. ];

2.2.2 Dynamic Partial AST Tree

The AST tree of the animation part of the page drawn this time is dynamically generated and consists of a series of circles with dynamic colors.

  1. function getMarqueeAst(startX, endX, count , options = {}) {
  2. const {y = 15, R = 15} = options;
  3. if (!(endX >= startX && count > 0)) {
  4. return [];
  5. }
  6. const interval = (endX - startX) / count ;
  7. const marqueeAstArr = [];
  8. for (let i = 0; i < count ; i++) {
  9. const RValue = Math.random() * 255;
  10. const GValue = Math.random() * 255;
  11. const BValue = Math.random() * 255;
  12. const fillStyle = `rgb(${RValue}, ${GValue}, ${BValue})`;
  13. marqueeAstArr.push({
  14. type: 'circle' ,
  15. x: startX + i * interval,
  16. y,
  17. R,
  18. fillStyle
  19. });
  20. }
  21.  
  22. return marqueeAstArr;
  23. }

2.3 Main function class

In addition to the above basic element classes, they will be exposed to the outside through a main function class.

  1. class Draw {
  2. constructor(canvasDom) {
  3. this._canvasDom = canvasDom;
  4. this.ctx = this._canvasDom.getContext( '2d' );
  5. this.width = this._canvasDom.width;
  6. this.height = this._canvasDom.height;
  7. }
  8.  
  9. //Drawing function
  10. draw(ast) {
  11. ast.forEach(elementObj => {
  12. this.drawFactory(elementObj);
  13. const {children} = elementObj;
  14. // Recursive call
  15. if (children && Array.isArray(children)) {
  16. this.draw(children);
  17. }
  18. });
  19. }
  20.  
  21. // Factory model drawing corresponding basic elements
  22. drawFactory(elementObj) {
  23. const {type} = elementObj;
  24. switch(type) {
  25. case   'img' : {
  26. this.drawImage(elementObj);
  27. break;
  28. }
  29. case   'text' : {
  30. this.drawText(elementObj);
  31. break;
  32. }
  33. case   'rect' : {
  34. this.drawRect(elementObj);
  35. break;
  36. }
  37. case   'circle' : {
  38. this.drawCircle(elementObj);
  39. break;
  40. }
  41. }
  42. }
  43.  
  44. drawImage(imageObj) {
  45. const drawImage = new DrawImage(this.ctx, imageObj);
  46. drawImage.draw();
  47. }
  48.  
  49. drawText(textObj) {
  50. const drawText = new DrawText(this.ctx, textObj);
  51. drawText.draw();
  52. }
  53.  
  54. drawRect(rectObj) {
  55. const drawRect = new DrawRect(this.ctx, rectObj);
  56. drawRect.draw();
  57. }
  58.  
  59. drawCircle(circleObj) {
  60. const drawCircle = new DrawCircle(this.ctx, circleObj);
  61. drawCircle.draw();
  62. }
  63.  
  64. clearCanvas() {
  65. this.ctx.clearRect(0, 0, this.width, this.height);
  66. }
  67. }

2.4 Content drawing

The previous preparations have been completed. Now we will link the various functions with the AST tree to achieve the desired effect.

2.4.1 Static content drawing

First draw the static content as the cornerstone of the page.

  1. const basicCanvasDom = document.getElementById( 'basicCanvas' );
  2. const drawBasicInstance = new Draw(basicCanvasDom);
  3. drawBasicInstance.draw(graphicAst);

Static content.png

2.4.2 Drawing an animated marquee

Add some animation effects to this part to make it more exciting.

  1. const animationCanvasDom = document.getElementById( 'animationCanvas' );
  2. const drawAnimationInstance = new Draw(animationCanvasDom);
  3.  
  4. let renderCount = 0;
  5. function animate() {
  6. if (renderCount % 5 === 0) {
  7. drawAnimationInstance.clearCanvas();
  8. drawAnimationInstance.draw(getMarqueeAst(20, 1440, 22));
  9. drawAnimationInstance.draw(getMarqueeAst(20, 1440, 22, {
  10. y: 380
  11. }));
  12. }
  13. window.requestAnimationFrame(animate);
  14. renderCount++;
  15. }
  16. animate();

This article is reprinted from the WeChat public account "Zhiyuanzhe", which can be followed through the following QR code. To reprint this article, please contact Zhiyuanzhe's public account.

<<:  Ready to use right out of the box? StreamNative Platform 1.0 is now available

>>:  A brief analysis of the importance of service gateways to enterprise core application architecture

Recommend

What is the difficulty in porting your number?

Number portability was once considered an importa...

JustVPS New London VPS 30% off, $3.08/month - 1GB/20GB/300M unlimited traffic

JustVPS.pro recently launched a new VPS in London...

Mercury enables remote procedure calls (RPC) for high performance computing

summary Remote Procedure Call (RPC) is a widely u...

How to deal with the four major challenges of edge computing

Edge computing use cases are broad and its early ...

The secrets of Netty network programming, just read this one

Netty version: 4.1.55.Final Traditional IO model ...

Out-of-the-box infrastructure connectivity options

When it comes to connecting network devices acros...

5G brings precise positioning to the Internet of Things

Cellular has ‘all the ingredients’ to enhance pre...

The global 5G base station market will reach US$236.98 billion in 2026

According to the new research report "Applic...

TCP three-way handshake: in-depth understanding and C# example implementation

In computer network communications, TCP (Transmis...