├── .devcontainer └── devcontainer.json ├── .github └── FUNDING.yml ├── .gitignore ├── LICENSE ├── README.md ├── build-layer.sh ├── docs ├── 00-setup.md ├── 01-use.md ├── 404.html ├── Gemfile ├── LICENSE.txt ├── _config.yml ├── _includes │ ├── custom-head.html │ ├── disqus_comments.html │ ├── footer.html │ ├── google-analytics.html │ ├── head.html │ ├── header.html │ ├── social-icons │ │ ├── devto.svg │ │ ├── dribbble.svg │ │ ├── facebook.svg │ │ ├── flickr.svg │ │ ├── github.svg │ │ ├── gitlab.svg │ │ ├── google_scholar.svg │ │ ├── instagram.svg │ │ ├── keybase.svg │ │ ├── linkedin.svg │ │ ├── mastodon.svg │ │ ├── microdotblog.svg │ │ ├── pinterest.svg │ │ ├── rss.svg │ │ ├── stackoverflow.svg │ │ ├── telegram.svg │ │ ├── twitter.svg │ │ └── youtube.svg │ ├── social-item.html │ ├── social.html │ └── svg_symbol.html ├── _layouts │ ├── base.html │ ├── home.html │ ├── page.html │ └── post.html ├── _sass │ └── minima │ │ ├── _base.scss │ │ ├── _layout.scss │ │ ├── custom-styles.scss │ │ ├── custom-variables.scss │ │ ├── initialize.scss │ │ └── skins │ │ ├── auto.scss │ │ ├── classic.scss │ │ ├── dark.scss │ │ ├── solarized-dark.scss │ │ ├── solarized-light.scss │ │ └── solarized.scss ├── assets │ ├── DejaVuSansCircle.png │ ├── css │ │ └── style.scss │ └── favicon.ico ├── index.md ├── minima.gemspec └── script │ ├── bootstrap │ ├── build │ ├── cibuild │ └── server ├── lib ├── libX11.so.6 ├── libXau.so.6 ├── libXext.so.6 ├── libXrender.so.1 ├── libblkid.so.1 ├── libbrotlicommon.so.1 ├── libbrotlidec.so.1 ├── libbz2.so.1 ├── libc.so.6 ├── libcairo-gobject.so.2 ├── libcairo.so.2 ├── libdatrie.so.1 ├── libffi.so.8 ├── libfontconfig.so.1 ├── libfreetype.so.6 ├── libfribidi.so.0 ├── libgcc_s.so.1 ├── libgdk_pixbuf-2.0.so.0 ├── libgif.so.7 ├── libgio-2.0.so.0 ├── libglib-2.0.so.0 ├── libgmodule-2.0.so.0 ├── libgobject-2.0.so.0 ├── libgraphite2.so.3 ├── libharfbuzz.so.0 ├── libjpeg.so.62 ├── liblzma.so.5 ├── libm.so.6 ├── libmount.so.1 ├── libpango-1.0.so.0 ├── libpangocairo-1.0.so.0 ├── libpangoft2-1.0.so.0 ├── libpcre2-8.so.0 ├── libpixman-1.so.0 ├── libpng16.so.16 ├── librsvg-2.so.2 ├── libselinux.so.1 ├── libstdc++.so.6 ├── libthai.so.0 ├── libxcb-render.so.0 ├── libxcb-shm.so.0 ├── libxcb.so.1 ├── libxml2.so.2 └── libz.so.1 ├── nodejs └── test │ ├── canvas-font.js │ ├── canvas-svg.js │ ├── chart-line.js │ ├── fabric-rectangle.js │ └── konva-circle.js └── package-sam.yml /.devcontainer/devcontainer.json: -------------------------------------------------------------------------------- 1 | { 2 | "image": "public.ecr.aws/amazonlinux/amazonlinux:2023", 3 | "features": { 4 | } 5 | } 6 | -------------------------------------------------------------------------------- /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | # These are supported funding model platforms 2 | 3 | # github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] 4 | # patreon: # Replace with a single Patreon username 5 | # open_collective: # Replace with a single Open Collective username 6 | # ko_fi: # Replace with a single Ko-fi username 7 | # tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel 8 | # community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry 9 | # liberapay: # Replace with a single Liberapay username 10 | # issuehunt: # Replace with a single IssueHunt username 11 | # otechie: # Replace with a single Otechie username 12 | # custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] 13 | 14 | custom: ['https://paypal.me/charoitel'] 15 | 16 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .idea/ 2 | docs/.jekyll-cache 3 | docs/_site 4 | docs/Gemfile.lock 5 | nodejs/node_modules 6 | nodejs/node*/node_modules 7 | nodejs/package*.json 8 | package*.json 9 | canvas-nodejs_v*.zip 10 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2020-2025 Charoite Lee 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Canvas Layer for AWS Lambda 2 | 3 | ![GitHub](https://img.shields.io/github/license/charoitel/lambda-layer-canvas-nodejs) 4 | 5 | Canvas Layer for AWS Lambda is published and available on [AWS Serverless Application Repository](https://serverlessrepo.aws.amazon.com/applications/arn:aws:serverlessrepo:us-east-1:990551184979:applications~lambda-layer-canvas-nodejs), and GitHub at [charoitel/lambda-layer-canvas-nodejs](https://github.com/charoitel/lambda-layer-canvas-nodejs). The layer aims to provide a Cairo backed Mozilla Web Canvas API implementation layer for AWS Lambda, powered by [node-canvas](https://github.com/Automattic/node-canvas). 6 | 7 | ## About node-canvas 8 | 9 | [node-canvas](https://github.com/Automattic/node-canvas) is a Cairo backed Canvas implementation for Node.js. It implements the [Mozilla Web Canvas API](https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API) as closely as possible. For the latest API compliance, you may check [Compatibility Status](https://github.com/Automattic/node-canvas/wiki/Compatibility-Status). 10 | 11 | ## How this layer is built? 12 | 13 | The layer is built from source of node-canvas npm package on [amazonlinux](https://hub.docker.com/_/amazonlinux) dev container instance, with following native dependencies installed. You may check the build layer script, ``` build-layer.sh ```, which is available in repository, for details. 14 | 15 | ```bash 16 | gcc-c++ cairo-devel pango-devel libjpeg-turbo-devel giflib-devel librsvg2-devel pango-devel bzip2-devel jq python3 17 | ``` 18 | 19 | Since AWS Lambda is a secure and isolated runtime and execution environment, the layer aims to target AWS Lambda compatible and native build. As there are canvas libraries and frameworks relying on node-canvas running on Node.js runtime, this layer may also try to include and support those libraries and frameworks. Currently, following libraries and frameworks are included when building and packaging the layer: 20 | 21 | - [Chart.js](#chartjs-support) 22 | - [Fabric.js](#fabricjs-support) 23 | - [Konva](#konva-support) 24 | 25 | ### Chart.js support 26 | 27 | [Chart.js](https://github.com/chartjs/chart.js) provides a set of frequently used chart types, plugins, and customization options. In addition to a reasonable set of built-in chart types, there are also community-maintained chart types. 28 | 29 | > Current supported version chart.js@4.4.9 with chartjs-node-canvas@5.0.0 30 | 31 | ### Fabric.js support 32 | 33 | [Fabric.js](https://github.com/fabricjs/fabric.js) provides a missing and interactive object model for canvas, as well as an SVG parser, layer of interactivity, and a whole suite of other indispensable tools. 34 | 35 | > Current supported version fabric@6.6.4 36 | 37 | ### Konva support 38 | 39 | [Konva](https://github.com/konvajs/konva) enables high performance animations, transitions, node nesting, layering, filtering, caching, event handling for desktop and mobile applications, and much more. 40 | 41 | > Current supported version konva@9.3.20 42 | -------------------------------------------------------------------------------- /build-layer.sh: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | # 3 | # Setup environment before build layer 4 | # yum update -y 5 | # yum groupinstall "Development Tools" -y 6 | # yum install gcc-c++ cairo-devel pango-devel libjpeg-turbo-devel giflib-devel librsvg2-devel pango-devel bzip2-devel jq python3 -y 7 | # 8 | # Setting Up Node.js, refer https://docs.aws.amazon.com/sdk-for-javascript/v2/developer-guide/setting-up-node-on-ec2-instance.html for details 9 | # curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.40.3/install.sh | bash 10 | # . ~/.nvm/nvm.sh 11 | # nvm install --lts 12 | # 13 | set -e 14 | 15 | LAYER_NAME=canvas-nodejs 16 | LAYER_DESCRIPTION="Cairo backed Mozilla Web Canvas API implementation layer for AWS Lambda" 17 | LAYER_VERSION=3.1.0 18 | LAYER_AUTHOR="Charoite Lee" 19 | 20 | DOT_CHAR="." 21 | NODE_VERSION=$(node -v) 22 | NODE_VERSION=${NODE_VERSION:1} 23 | SEMVER_VERSION=7.7.1 24 | 25 | # Remove packaged layer if exists 26 | if [ -n "$(find . -name 'canvas-nodejs_v*.zip')" ]; then 27 | rm canvas-nodejs_v*.zip 28 | fi 29 | 30 | # Clean and prepare Node.js modules and dependencies 31 | if [ "$(ls -A lib)" ]; then 32 | rm lib/* 33 | fi 34 | cd nodejs 35 | if [ "$(ls -A node*)" ]; then 36 | rm -rf node* 37 | fi 38 | rm -rf node_modules node${NODE_VERSION%%$DOT_CHAR*} package*.json ../package-lock.json 39 | npm init -y 40 | npm install canvas --build-from-source 41 | npm install fabric 42 | npm install konva 43 | npm install chartjs-node-canvas chart.js 44 | npm install mocha --save-dev 45 | jq --arg LAYER_NAME "$LAYER_NAME" --arg LAYER_DESCRIPTION "$LAYER_DESCRIPTION" --arg LAYER_VERSION "$LAYER_VERSION" --arg LAYER_AUTHOR "$LAYER_AUTHOR" --arg SEMVER_VERSION "$SEMVER_VERSION" '.name = $LAYER_NAME | .description = $LAYER_DESCRIPTION | .version = $LAYER_VERSION | .license = "MIT" | .author = $LAYER_AUTHOR | .scripts.test = "mocha" | .overrides.semver = $SEMVER_VERSION ' package.json > package-tmp.json 46 | mv -f package-tmp.json package.json 47 | 48 | # Test if installed modules and dependencies work fine 49 | npm test 50 | cp package-lock.json .. 51 | npm rm mocha 52 | 53 | # Prepare and package layer 54 | mkdir node${NODE_VERSION%%$DOT_CHAR*} 55 | mv node_modules node${NODE_VERSION%%$DOT_CHAR*} 56 | cd .. 57 | find nodejs/node${NODE_VERSION%%$DOT_CHAR*} -type f -name '*.node' 2>/dev/null | grep -v 'obj\.target' | xargs ldd | awk 'NF == 4 { system("cp " $3 " lib") }' 58 | zip -q -r canvas-nodejs_v$LAYER_VERSION-node${NODE_VERSION%%$DOT_CHAR*}.zip . -x LICENSE README.md .git/**\* .github/**\* .gitignore nodejs/test/**\* *.yml build-layer.sh 59 | -------------------------------------------------------------------------------- /docs/00-setup.md: -------------------------------------------------------------------------------- 1 | --- 2 | layout: page 3 | title: Setup 4 | permalink: /setup/ 5 | --- 6 | 7 | ## Prerequisites 8 | 9 | In order to start to use [lambda-layer-canvas-nodejs](https://github.com/charoitel/lambda-layer-canvas-nodejs) published on [AWS Serverless Application Repository](https://serverlessrepo.aws.amazon.com/applications/arn:aws:serverlessrepo:us-east-1:990551184979:applications~lambda-layer-canvas-nodejs), you must have your AWS account with following services available: 10 | 11 | - [AWS Serverless Application Repository](https://aws.amazon.com/serverless/serverlessrepo) 12 | - [AWS Lambda](https://aws.amazon.com/lambda) 13 | 14 | ## Serverless application deployment 15 | 16 | Once you have your AWS account ready, there are two ways to deploy [lambda-layer-canvas-nodejs](https://serverlessrepo.aws.amazon.com/applications/arn:aws:serverlessrepo:us-east-1:990551184979:applications~lambda-layer-canvas-nodejs) and make it available in your AWS Lambda console: 17 | 18 | - [Deploy through AWS Serverless Application Repository](#deploy-through-aws-serverless-application-repository) 19 | - [Deploy through AWS Lambda console](#deploy-through-aws-lambda-console) 20 | 21 | ### Deploy through AWS Serverless Application Repository 22 | 23 | 1. Open https://serverlessrepo.aws.amazon.com/applications/arn:aws:serverlessrepo:us-east-1:990551184979:applications~lambda-layer-canvas-nodejs 24 | 2. Click **Deploy** button 25 | 3. Login to your AWS account if you haven't login yet 26 | 4. Edit **_Name_** (Optional) 27 | 5. Click **Deploy** button 28 | 6. Deployment is started and in progress 29 | 7. Check your AWS Lambda console once the deployment is completed 30 | 31 | ### Deploy through AWS Lambda console 32 | 33 | 1. Login to your AWS account and open your AWS Lambda console 34 | 2. Click **Create application** button 35 | 3. Select **_Serverless application_** 36 | 4. Input `lambda-layer-canvas-nodejs` into search box and press _Enter_ key 37 | 5. Click on the title of `lambda-layer-canvas-nodejs` card 38 | 6. Edit **_Name_** (Optional) 39 | 7. Click **Deploy** button 40 | 8. Deployment is started and in progress 41 | 9. Check your AWS Lambda console once the deployment is completed 42 | 43 | ### Using canvas layer 44 | 45 | After the deployment is completed, you may refer [usage example](/lambda-layer-canvas-nodejs/{% link 01-use.md %}) and follow the example where a Lambda function uses the canvas layer to generate PNG graphic with colored text and circle rendered. 46 | 47 | ## Setup environment to build the layer 48 | 49 | Alternately, you may setup your own environment to build the layer according to your specific needs. When using the layer with nodejs-18.x in Amazon Linux, it requires glibc-2.28, meanwhile, compiling glibc-2.28 requires make-4.x or later[^1]. 50 | 51 | ```console 52 | $ wget https://ftp.gnu.org/gnu/make/make-4.3.tar.gz 53 | $ tar -xzvf make-4.3.tar.gz 54 | $ cd make-4.3/ 55 | $ ./configure --prefix=/usr 56 | $ make 57 | $ make install 58 | ``` 59 | Once make-4.x or later is ready, we may start to compiling glibc-2.28 on Amazon Linux. However, during `make install` there would be an error due to `cannot found -lnss_test2` which could be ignored[^2]. 60 | 61 | ```console 62 | $ wget https://ftp.gnu.org/gnu/glibc/glibc-2.28.tar.gz 63 | $ tar -xzvf glibc-2.28.tar.gz 64 | $ cd glibc-2.28 65 | $ mkdir build && cd build 66 | $ ../configure --prefix=/usr --enable-add-ons --with-headers=/usr/include --with-binutils=/usr/bin 67 | $ make 68 | $ make install 69 | ... 70 | /usr/bin/ld: cannot find -lnss_test2 71 | collect2: error: ld returned 1 exit status 72 | Execution of gcc -B/usr/bin/ failed! 73 | ... 74 | ``` 75 | 76 | To test if the installation of the compiled glibc-2.28 is success or not, we may check if `GLIBC_2.28` is enabled[^1] [^2]. 77 | 78 | ```console 79 | $ ls -l /lib64/libc.so.6 80 | lrwxrwxrwx 1 root root 12 Jul 3 15:14 /lib64/libc.so.6 -> libc-2.28.so 81 | $ strings /lib64/libc.so.6 | grep GLIBC 82 | ... 83 | GLIBC_2.22 84 | GLIBC_2.23 85 | GLIBC_2.24 86 | GLIBC_2.25 87 | GLIBC_2.26 88 | GLIBC_2.27 89 | GLIBC_2.28 90 | GLIBC_PRIVATE 91 | ... 92 | ``` 93 | 94 | Once your environment is ready, you may execute the build layer script, ``` build-layer.sh ```, to build the layer and deploy through the AWS Lambda console. 95 | 96 | --- 97 | 98 | [^1]: [Centos 7 升级 Glibc-2.28](https://cloud.tencent.com/developer/article/2021784) 99 | [^2]: [CentOS 7.6 编译安装最新版本glibc2.30 实录](https://www.jianshu.com/p/1070373a50f6) -------------------------------------------------------------------------------- /docs/01-use.md: -------------------------------------------------------------------------------- 1 | --- 2 | layout: page 3 | title: Use 4 | permalink: /use/ 5 | --- 6 | 7 | ## Using canvas layer in function 8 | 9 | Before starting to use the canvas layer, it is required to have the layer deployed through AWS Serverless Application Repository or AWS Lambda console, you may refer [setup information](/lambda-layer-canvas-nodejs/{% link 00-setup.md %}) for details. Then, you may create your Lambda function using: 10 | 11 | * AWS Lambda console editor; or 12 | * Local tool in your machine; or 13 | * Any way you may prefer 14 | 15 | Following is an example where a Lambda function uses the canvas layer to generate PNG graphic with colored text and circle rendered. There is no environment variable setting is required in this example. 16 | 17 | ## Function structure 18 | 19 | The setup of the function may look like the following structure. For font file, you may use any supported font file you like, in this example, we are going to use DejaVu Sans[^2] which could be originally found in Amazon Linux 2 EC2 instance. 20 | 21 | ``` 22 | /var/task/ 23 | . 24 | ├── fonts/ 25 | ├── DejaVuSans.ttf 26 | ├── fonts.conf 27 | ├── index.js 28 | ``` 29 | 30 | ## Font configuration using fonts.config 31 | 32 | Since AWS Lambda does not appear to support even the basic fonts on its own, so we create the fonts directory at the root directory for our Lambda function with following font configuration file[^1]. 33 | 34 | ```xml 35 | 36 | 37 | 38 | /var/task/fonts 39 | /tmp/fonts-cache/ 40 | 41 | 42 | ``` 43 | 44 | ## Build function code in index.js 45 | 46 | Once the setup is ready, we may put following function code and test it in AWS Lambda console. 47 | 48 | ```javascript 49 | let { registerFont, createCanvas } = require('canvas'); 50 | 51 | exports.handler = async (event) => { 52 | registerFont(__dirname.concat('/fonts/DejaVuSans.ttf'), { family: 'DejaVu Sans' }); 53 | 54 | let canvas = createCanvas(300, 300); 55 | var ctxCircle = canvas.getContext('2d'); 56 | var X = canvas.width / 2; 57 | var Y = canvas.height / 2; 58 | var R = 45; 59 | ctxCircle.beginPath(); 60 | ctxCircle.arc(X, Y, R, 0, 2 * Math.PI, false); 61 | ctxCircle.lineWidth = 3; 62 | ctxCircle.strokeStyle = '#FF9900'; 63 | ctxCircle.stroke(); 64 | 65 | var ctxText = canvas.getContext('2d'); 66 | ctxText.font = '40px "DejaVu Sans"'; 67 | ctxText.fillStyle = '#146EB4'; 68 | ctxText.fillText('DejaVu Sans', 15, 120); 69 | 70 | const response = { 71 | statusCode: 200, 72 | body: JSON.stringify(canvas.toDataURL()), 73 | }; 74 | 75 | return response; 76 | }; 77 | ``` 78 | Following is the sample response from the above example function code. 79 | 80 | ```JSON 81 | { 82 | "statusCode": 200, 83 | "body": "\"\"" 84 | } 85 | ``` 86 | As a result, the generated PNG graphic with colored text and circle rendered may look like as follow. 87 | 88 | ![](/assets/DejaVuSansCircle.png) 89 | 90 | --- 91 | 92 | [^1]: [AWS Lambda Fonts & libfontconfig](https://medium.com/creditorwatch/aws-lambda-fonts-libfontconfig-5e837281a4ce) 93 | [^2]: [DejaVu Fonts](https://dejavu-fonts.github.io/) 94 | -------------------------------------------------------------------------------- /docs/404.html: -------------------------------------------------------------------------------- 1 | --- 2 | permalink: /404.html 3 | layout: default 4 | --- 5 | 6 | 19 | 20 |
21 |

404

22 | 23 |

Page not found :(

24 |

The requested page could not be found.

25 |
26 | -------------------------------------------------------------------------------- /docs/Gemfile: -------------------------------------------------------------------------------- 1 | gem "minima" 2 | -------------------------------------------------------------------------------- /docs/LICENSE.txt: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016-present Parker Moore and the minima contributors 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in 13 | all copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 21 | THE SOFTWARE. 22 | -------------------------------------------------------------------------------- /docs/_config.yml: -------------------------------------------------------------------------------- 1 | title: Canvas Layer for AWS Lambda 2 | description: > 3 | Cairo backed Mozilla Web Canvas API implementation layer for AWS Lambda 4 | theme: minima 5 | minima: 6 | social_links: 7 | - { platform: github, user_url: "https://github.com/charoitel/lambda-layer-canvas-nodejs" } 8 | - { platform: linkedin, user_url: "https://www.linkedin.com/in/charoitel" } 9 | -------------------------------------------------------------------------------- /docs/_includes/custom-head.html: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/disqus_comments.html: -------------------------------------------------------------------------------- 1 | {%- if page.comments != false and jekyll.environment == "production" -%} 2 | 3 |
4 | 19 | 20 | {%- endif -%} 21 | -------------------------------------------------------------------------------- /docs/_includes/footer.html: -------------------------------------------------------------------------------- 1 | 30 | -------------------------------------------------------------------------------- /docs/_includes/google-analytics.html: -------------------------------------------------------------------------------- 1 | 2 | 10 | -------------------------------------------------------------------------------- /docs/_includes/head.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | {%- seo -%} 6 | 7 | {%- feed_meta -%} 8 | {%- if jekyll.environment == 'production' and site.google_analytics -%} 9 | {%- include google-analytics.html -%} 10 | {%- endif -%} 11 | 12 | {%- include custom-head.html -%} 13 | 14 | 15 | -------------------------------------------------------------------------------- /docs/_includes/header.html: -------------------------------------------------------------------------------- 1 | 32 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/devto.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/dribbble.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/facebook.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/flickr.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/github.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/gitlab.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/google_scholar.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/instagram.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/keybase.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/linkedin.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/mastodon.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/microdotblog.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/pinterest.svg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/rss.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/stackoverflow.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/telegram.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/twitter.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-icons/youtube.svg: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /docs/_includes/social-item.html: -------------------------------------------------------------------------------- 1 |
  • {% assign entry = include.item %} 2 | 3 | 4 | 5 | 6 | 7 |
  • 8 | -------------------------------------------------------------------------------- /docs/_includes/social.html: -------------------------------------------------------------------------------- 1 | 6 | -------------------------------------------------------------------------------- /docs/_includes/svg_symbol.html: -------------------------------------------------------------------------------- 1 | 2 | {%- include social-icons/{{ include.key }}.svg -%} 3 | 4 | -------------------------------------------------------------------------------- /docs/_layouts/base.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | {%- include head.html -%} 5 | 6 | 7 | 8 | {%- include header.html -%} 9 | 10 |
    11 |
    12 | {{ content }} 13 |
    14 |
    15 | 16 | {%- include footer.html -%} 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /docs/_layouts/home.html: -------------------------------------------------------------------------------- 1 | --- 2 | layout: base 3 | --- 4 | 5 |
    6 | {%- if page.title -%} 7 |

    {{ page.title }}

    8 | {%- endif -%} 9 | 10 | {{ content }} 11 | 12 | 13 | {% if site.paginate %} 14 | {% assign posts = paginator.posts %} 15 | {% else %} 16 | {% assign posts = site.posts %} 17 | {% endif %} 18 | 19 | 20 | {%- if posts.size > 0 -%} 21 | {%- if page.list_title -%} 22 |

    {{ page.list_title }}

    23 | {%- endif -%} 24 |
      25 | {%- assign date_format = site.minima.date_format | default: "%b %-d, %Y" -%} 26 | {%- for post in posts -%} 27 |
    • 28 | 29 |

      30 | 31 | {{ post.title | escape }} 32 | 33 |

      34 | {%- if site.show_excerpts -%} 35 | {{ post.excerpt }} 36 | {%- endif -%} 37 |
    • 38 | {%- endfor -%} 39 |
    40 | 41 | {% if site.paginate %} 42 |
    43 | 56 |
    57 | {%- endif %} 58 | 59 | {%- endif -%} 60 | 61 |
    -------------------------------------------------------------------------------- /docs/_layouts/page.html: -------------------------------------------------------------------------------- 1 | --- 2 | layout: base 3 | --- 4 |
    5 | 6 |
    7 |

    {{ page.title | escape }}

    8 |
    9 | 10 |
    11 | {{ content }} 12 |
    13 | 14 |
    15 | -------------------------------------------------------------------------------- /docs/_layouts/post.html: -------------------------------------------------------------------------------- 1 | --- 2 | layout: base 3 | --- 4 |
    5 | 6 |
    7 |

    {{ page.title | escape }}

    8 | 27 |
    28 | 29 |
    30 | {{ content }} 31 |
    32 | 33 | {%- if site.disqus.shortname -%} 34 | {%- include disqus_comments.html -%} 35 | {%- endif -%} 36 | 37 | 38 |
    39 | -------------------------------------------------------------------------------- /docs/_sass/minima/_base.scss: -------------------------------------------------------------------------------- 1 | html { 2 | font-size: $base-font-size; 3 | } 4 | 5 | /** 6 | * Reset some basic elements 7 | */ 8 | body, h1, h2, h3, h4, h5, h6, 9 | p, blockquote, pre, hr, 10 | dl, dd, ol, ul, figure { 11 | margin: 0; 12 | padding: 0; 13 | 14 | } 15 | 16 | 17 | 18 | /** 19 | * Basic styling 20 | */ 21 | body { 22 | font: $base-font-weight #{$base-font-size}/#{$base-line-height} $base-font-family; 23 | color: $text-color; 24 | background-color: $background-color; 25 | -webkit-text-size-adjust: 100%; 26 | -webkit-font-feature-settings: "kern" 1; 27 | -moz-font-feature-settings: "kern" 1; 28 | -o-font-feature-settings: "kern" 1; 29 | font-feature-settings: "kern" 1; 30 | font-kerning: normal; 31 | display: flex; 32 | min-height: 100vh; 33 | flex-direction: column; 34 | overflow-wrap: break-word; 35 | } 36 | 37 | 38 | 39 | /** 40 | * Set `margin-bottom` to maintain vertical rhythm 41 | */ 42 | h1, h2, h3, h4, h5, h6, 43 | p, blockquote, pre, 44 | ul, ol, dl, figure, 45 | %vertical-rhythm { 46 | margin-bottom: $spacing-unit * .5; 47 | } 48 | 49 | hr { 50 | margin-top: $spacing-unit; 51 | margin-bottom: $spacing-unit; 52 | } 53 | 54 | /** 55 | * `main` element 56 | */ 57 | main { 58 | display: block; /* Default value of `display` of `main` element is 'inline' in IE 11. */ 59 | } 60 | 61 | 62 | 63 | /** 64 | * Images 65 | */ 66 | img { 67 | max-width: 100%; 68 | vertical-align: middle; 69 | } 70 | 71 | 72 | 73 | /** 74 | * Figures 75 | */ 76 | figure > img { 77 | display: block; 78 | } 79 | 80 | figcaption { 81 | font-size: $small-font-size; 82 | } 83 | 84 | 85 | 86 | /** 87 | * Lists 88 | */ 89 | ul, ol { 90 | margin-left: $spacing-unit; 91 | } 92 | 93 | li { 94 | > ul, 95 | > ol { 96 | margin-bottom: 0; 97 | } 98 | } 99 | 100 | 101 | 102 | /** 103 | * Headings 104 | */ 105 | h1, h2, h3, h4, h5, h6 { 106 | font-weight: $base-font-weight; 107 | } 108 | 109 | 110 | 111 | /** 112 | * Links 113 | */ 114 | a { 115 | color: $link-base-color; 116 | text-decoration: none; 117 | 118 | &:visited { 119 | color: $link-visited-color; 120 | } 121 | 122 | &:hover { 123 | color: $link-hover-color; 124 | text-decoration: underline; 125 | } 126 | 127 | .social-media-list &:hover { 128 | text-decoration: none; 129 | 130 | .username { 131 | text-decoration: underline; 132 | } 133 | } 134 | } 135 | 136 | 137 | /** 138 | * Blockquotes 139 | */ 140 | blockquote { 141 | color: $brand-color; 142 | border-left: 4px solid $border-color-01; 143 | padding-left: $spacing-unit * .5; 144 | @include relative-font-size(1.125); 145 | font-style: italic; 146 | 147 | > :last-child { 148 | margin-bottom: 0; 149 | } 150 | 151 | i, em { 152 | font-style: normal; 153 | } 154 | } 155 | 156 | 157 | 158 | /** 159 | * Code formatting 160 | */ 161 | pre, 162 | code { 163 | font-family: $code-font-family; 164 | font-size: 0.9375em; 165 | border: 1px solid $border-color-01; 166 | border-radius: 3px; 167 | background-color: $code-background-color; 168 | } 169 | 170 | code { 171 | padding: 1px 5px; 172 | } 173 | 174 | pre { 175 | padding: 8px 12px; 176 | overflow-x: auto; 177 | 178 | > code { 179 | border: 0; 180 | padding-right: 0; 181 | padding-left: 0; 182 | } 183 | } 184 | 185 | .highlight { 186 | border-radius: 3px; 187 | background: $code-background-color; 188 | @extend %vertical-rhythm; 189 | 190 | .highlighter-rouge & { 191 | background: $code-background-color; 192 | } 193 | } 194 | 195 | 196 | 197 | /** 198 | * Wrapper 199 | */ 200 | .wrapper { 201 | max-width: calc(#{$content-width} - (#{$spacing-unit})); 202 | margin-right: auto; 203 | margin-left: auto; 204 | padding-right: $spacing-unit * .5; 205 | padding-left: $spacing-unit * .5; 206 | @extend %clearfix; 207 | 208 | @media screen and (min-width: $on-large) { 209 | max-width: calc(#{$content-width} - (#{$spacing-unit} * 2)); 210 | padding-right: $spacing-unit; 211 | padding-left: $spacing-unit; 212 | } 213 | } 214 | 215 | 216 | 217 | /** 218 | * Clearfix 219 | */ 220 | %clearfix:after { 221 | content: ""; 222 | display: table; 223 | clear: both; 224 | } 225 | 226 | 227 | 228 | /** 229 | * Icons 230 | */ 231 | 232 | .orange { 233 | color: #f66a0a; 234 | } 235 | 236 | .grey { 237 | color: #828282; 238 | } 239 | 240 | .svg-icon { 241 | width: 1.25em; 242 | height: 1.25em; 243 | display: inline-block; 244 | fill: currentColor; 245 | vertical-align: text-bottom; 246 | } 247 | 248 | 249 | /** 250 | * Tables 251 | */ 252 | table { 253 | margin-bottom: $spacing-unit; 254 | width: 100%; 255 | text-align: $table-text-align; 256 | color: $table-text-color; 257 | border-collapse: collapse; 258 | border: 1px solid $table-border-color; 259 | tr { 260 | &:nth-child(even) { 261 | background-color: $table-zebra-color; 262 | } 263 | } 264 | th, td { 265 | padding: ($spacing-unit * 33.3333333333 * .01) ($spacing-unit * .5); 266 | } 267 | th { 268 | background-color: $table-header-bg-color; 269 | border: 1px solid $table-header-border; 270 | } 271 | td { 272 | border: 1px solid $table-border-color; 273 | } 274 | 275 | @include media-query($on-laptop) { 276 | display: block; 277 | overflow-x: auto; 278 | -webkit-overflow-scrolling: touch; 279 | -ms-overflow-style: -ms-autohiding-scrollbar; 280 | } 281 | } 282 | -------------------------------------------------------------------------------- /docs/_sass/minima/_layout.scss: -------------------------------------------------------------------------------- 1 | /** 2 | * Site header 3 | */ 4 | .site-header { 5 | border-top: 5px solid $border-color-03; 6 | border-bottom: 1px solid $border-color-01; 7 | min-height: $spacing-unit * 1.865; 8 | line-height: $base-line-height * $base-font-size * 2.25; 9 | 10 | // Positioning context for the mobile navigation icon 11 | position: relative; 12 | } 13 | 14 | .site-title { 15 | @include relative-font-size(1.625); 16 | font-weight: 300; 17 | letter-spacing: -1px; 18 | margin-bottom: 0; 19 | float: left; 20 | 21 | @include media-query($on-palm) { 22 | padding-right: 45px; 23 | } 24 | 25 | &, 26 | &:visited { 27 | color: $site-title-color; 28 | } 29 | } 30 | 31 | .site-nav { 32 | position: absolute; 33 | top: 9px; 34 | right: $spacing-unit * .5; 35 | background-color: $background-color; 36 | border: 1px solid $border-color-01; 37 | border-radius: 5px; 38 | text-align: right; 39 | 40 | .nav-trigger { 41 | display: none; 42 | } 43 | 44 | .menu-icon { 45 | float: right; 46 | width: 36px; 47 | height: 26px; 48 | line-height: 0; 49 | padding-top: 10px; 50 | text-align: center; 51 | 52 | > svg path { 53 | fill: $border-color-03; 54 | } 55 | } 56 | 57 | label[for="nav-trigger"] { 58 | display: block; 59 | float: right; 60 | width: 36px; 61 | height: 36px; 62 | z-index: 2; 63 | cursor: pointer; 64 | } 65 | 66 | input ~ .trigger { 67 | clear: both; 68 | display: none; 69 | } 70 | 71 | input:checked ~ .trigger { 72 | display: block; 73 | padding-bottom: 5px; 74 | } 75 | 76 | .page-link { 77 | color: $text-color; 78 | line-height: $base-line-height; 79 | display: block; 80 | padding: 5px 10px; 81 | 82 | // Gaps between nav items, but not on the last one 83 | &:not(:last-child) { 84 | margin-right: 0; 85 | } 86 | margin-left: 20px; 87 | } 88 | 89 | @media screen and (min-width: $on-medium) { 90 | position: static; 91 | float: right; 92 | border: none; 93 | background-color: inherit; 94 | 95 | label[for="nav-trigger"] { 96 | display: none; 97 | } 98 | 99 | .menu-icon { 100 | display: none; 101 | } 102 | 103 | input ~ .trigger { 104 | display: block; 105 | } 106 | 107 | .page-link { 108 | display: inline; 109 | padding: 0; 110 | 111 | &:not(:last-child) { 112 | margin-right: 20px; 113 | } 114 | margin-left: auto; 115 | } 116 | } 117 | } 118 | 119 | 120 | 121 | /** 122 | * Site footer 123 | */ 124 | .site-footer { 125 | border-top: 1px solid $border-color-01; 126 | padding: $spacing-unit 0; 127 | } 128 | 129 | .footer-heading { 130 | @include relative-font-size(1.125); 131 | margin-bottom: $spacing-unit * .5; 132 | } 133 | 134 | .feed-subscribe .svg-icon { 135 | padding: 5px 5px 2px 0 136 | } 137 | 138 | .contact-list, 139 | .social-media-list { 140 | list-style: none; 141 | margin-left: 0; 142 | } 143 | 144 | .footer-col-wrapper, 145 | .social-links { 146 | @include relative-font-size(0.9375); 147 | color: $brand-color; 148 | } 149 | 150 | .footer-col { 151 | margin-bottom: $spacing-unit * .5; 152 | } 153 | 154 | .footer-col-1, 155 | .footer-col-2 { 156 | width: calc(50% - (#{$spacing-unit} / 2)); 157 | } 158 | 159 | .footer-col-3 { 160 | width: calc(100% - (#{$spacing-unit} / 2)); 161 | } 162 | 163 | @media screen and (min-width: $on-large) { 164 | .footer-col-1 { 165 | width: calc(35% - (#{$spacing-unit} / 2)); 166 | } 167 | 168 | .footer-col-2 { 169 | width: calc(20% - (#{$spacing-unit} / 2)); 170 | } 171 | 172 | .footer-col-3 { 173 | width: calc(45% - (#{$spacing-unit} / 2)); 174 | } 175 | } 176 | 177 | @media screen and (min-width: $on-medium) { 178 | .footer-col-wrapper { 179 | display: flex 180 | } 181 | 182 | .footer-col { 183 | width: calc(100% - (#{$spacing-unit} / 2)); 184 | padding: 0 ($spacing-unit * .5); 185 | 186 | &:first-child { 187 | padding-right: $spacing-unit * .5; 188 | padding-left: 0; 189 | } 190 | 191 | &:last-child { 192 | padding-right: 0; 193 | padding-left: $spacing-unit * .5; 194 | } 195 | } 196 | } 197 | 198 | 199 | 200 | /** 201 | * Page content 202 | */ 203 | .page-content { 204 | padding: $spacing-unit 0; 205 | flex: 1 0 auto; 206 | } 207 | 208 | .page-heading { 209 | @include relative-font-size(2); 210 | } 211 | 212 | .post-list-heading { 213 | @include relative-font-size(1.75); 214 | } 215 | 216 | .post-list { 217 | margin-left: 0; 218 | list-style: none; 219 | 220 | > li { 221 | margin-bottom: $spacing-unit; 222 | } 223 | } 224 | 225 | .post-meta { 226 | font-size: $small-font-size; 227 | color: $brand-color; 228 | } 229 | 230 | .post-link { 231 | display: block; 232 | @include relative-font-size(1.5); 233 | } 234 | 235 | 236 | 237 | /** 238 | * Posts 239 | */ 240 | .post-header { 241 | margin-bottom: $spacing-unit; 242 | } 243 | 244 | .post-title, 245 | .post-content h1 { 246 | @include relative-font-size(2.625); 247 | letter-spacing: -1px; 248 | line-height: 1.15; 249 | 250 | @media screen and (min-width: $on-large) { 251 | @include relative-font-size(2.625); 252 | } 253 | } 254 | 255 | .post-content { 256 | margin-bottom: $spacing-unit; 257 | 258 | h1, h2, h3, h4, h5, h6 { margin-top: $spacing-unit } 259 | 260 | h2 { 261 | @include relative-font-size(1.75); 262 | 263 | @media screen and (min-width: $on-large) { 264 | @include relative-font-size(2); 265 | } 266 | } 267 | 268 | h3 { 269 | @include relative-font-size(1.375); 270 | 271 | @media screen and (min-width: $on-large) { 272 | @include relative-font-size(1.625); 273 | } 274 | } 275 | 276 | h4 { 277 | @include relative-font-size(1.25); 278 | } 279 | 280 | h5 { 281 | @include relative-font-size(1.125); 282 | } 283 | h6 { 284 | @include relative-font-size(1.0625); 285 | } 286 | } 287 | 288 | 289 | .social-media-list { 290 | display: table; 291 | margin: 0 auto; 292 | li { 293 | float: left; 294 | margin: 5px 10px 5px 0; 295 | &:last-of-type { margin-right: 0 } 296 | a { 297 | display: block; 298 | padding: 10px 12px; 299 | border: 1px solid $border-color-01; 300 | &:hover { border-color: $border-color-02 } 301 | } 302 | } 303 | } 304 | 305 | 306 | 307 | /** 308 | * Pagination navbar 309 | */ 310 | .pagination { 311 | margin-bottom: $spacing-unit; 312 | @extend .social-media-list; 313 | li { 314 | a, div { 315 | min-width: 41px; 316 | text-align: center; 317 | box-sizing: border-box; 318 | } 319 | div { 320 | display: block; 321 | padding: $spacing-unit * .25; 322 | border: 1px solid transparent; 323 | 324 | &.pager-edge { 325 | color: $border-color-01; 326 | border: 1px dashed; 327 | } 328 | } 329 | } 330 | } 331 | 332 | 333 | 334 | /** 335 | * Grid helpers 336 | */ 337 | @media screen and (min-width: $on-large) { 338 | .one-half { 339 | width: calc(50% - (#{$spacing-unit} / 2)); 340 | } 341 | } 342 | -------------------------------------------------------------------------------- /docs/_sass/minima/custom-styles.scss: -------------------------------------------------------------------------------- 1 | // Placeholder to allow defining custom styles that override everything else. 2 | // (Use `_sass/minima/custom-variables.scss` to override variable defaults) 3 | -------------------------------------------------------------------------------- /docs/_sass/minima/custom-variables.scss: -------------------------------------------------------------------------------- 1 | // Placeholder to allow overriding predefined variables smoothly. 2 | -------------------------------------------------------------------------------- /docs/_sass/minima/initialize.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | // Define defaults for each variable. 4 | 5 | $base-font-family: -apple-system, system-ui, BlinkMacSystemFont, "Segoe UI", "Segoe UI Emoji", "Segoe UI Symbol", "Apple Color Emoji", Roboto, Helvetica, Arial, sans-serif !default; 6 | $code-font-family: "Menlo", "Inconsolata", "Consolas", "Roboto Mono", "Ubuntu Mono", "Liberation Mono", "Courier New", monospace; 7 | $base-font-size: 16px !default; 8 | $base-font-weight: 400 !default; 9 | $small-font-size: $base-font-size * 0.875 !default; 10 | $base-line-height: 1.5 !default; 11 | 12 | $spacing-unit: 30px !default; 13 | 14 | $table-text-align: left !default; 15 | 16 | // Width of the content area 17 | $content-width: 800px !default; 18 | 19 | $on-palm: 600px !default; 20 | $on-laptop: 800px !default; 21 | 22 | $on-medium: $on-palm !default; 23 | $on-large: $on-laptop !default; 24 | 25 | // Use media queries like this: 26 | // @include media-query($on-palm) { 27 | // .wrapper { 28 | // padding-right: $spacing-unit / 2; 29 | // padding-left: $spacing-unit / 2; 30 | // } 31 | // } 32 | // Notice the following mixin uses max-width, in a deprecated, desktop-first 33 | // approach, whereas media queries used elsewhere now use min-width. 34 | @mixin media-query($device) { 35 | @media screen and (max-width: $device) { 36 | @content; 37 | } 38 | } 39 | 40 | @mixin relative-font-size($ratio) { 41 | font-size: #{$ratio}rem; 42 | } 43 | 44 | // Import pre-styling-overrides hook and style-partials. 45 | @import 46 | "minima/custom-variables", // Hook to override predefined variables. 47 | "minima/base", // Defines element resets. 48 | "minima/layout", // Defines structure and style based on CSS selectors. 49 | "minima/custom-styles" // Hook to override existing styles. 50 | ; 51 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/auto.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | // Default color scheme settings 4 | // These are overridden in classic.css and dark.scss 5 | 6 | $color-scheme-auto: true !default; 7 | $color-scheme-dark: false !default; 8 | 9 | 10 | // Light mode 11 | // ---------- 12 | 13 | $lm-brand-color: #828282 !default; 14 | $lm-brand-color-light: lighten($lm-brand-color, 40%) !default; 15 | $lm-brand-color-dark: darken($lm-brand-color, 25%) !default; 16 | 17 | $lm-site-title-color: $lm-brand-color-dark !default; 18 | 19 | $lm-text-color: #111111 !default; 20 | $lm-background-color: #fdfdfd !default; 21 | $lm-code-background-color: #eeeeff !default; 22 | 23 | $lm-link-base-color: #2a7ae2 !default; 24 | $lm-link-visited-color: darken($lm-link-base-color, 15%) !default; 25 | $lm-link-hover-color: $lm-text-color !default; 26 | 27 | $lm-border-color-01: $lm-brand-color-light !default; 28 | $lm-border-color-02: lighten($lm-brand-color, 35%) !default; 29 | $lm-border-color-03: $lm-brand-color-dark !default; 30 | 31 | $lm-table-text-color: lighten($lm-text-color, 18%) !default; 32 | $lm-table-zebra-color: lighten($lm-brand-color, 46%) !default; 33 | $lm-table-header-bg-color: lighten($lm-brand-color, 43%) !default; 34 | $lm-table-header-border: lighten($lm-brand-color, 37%) !default; 35 | $lm-table-border-color: $lm-border-color-01 !default; 36 | 37 | 38 | // Syntax highlighting styles should be adjusted appropriately for every "skin" 39 | // ---------------------------------------------------------------------------- 40 | 41 | @mixin lm-highlight { 42 | .highlight { 43 | .c { color: #998; font-style: italic } // Comment 44 | .err { color: #a61717; background-color: #e3d2d2 } // Error 45 | .k { font-weight: bold } // Keyword 46 | .o { font-weight: bold } // Operator 47 | .cm { color: #998; font-style: italic } // Comment.Multiline 48 | .cp { color: #999; font-weight: bold } // Comment.Preproc 49 | .c1 { color: #998; font-style: italic } // Comment.Single 50 | .cs { color: #999; font-weight: bold; font-style: italic } // Comment.Special 51 | .gd { color: #000; background-color: #fdd } // Generic.Deleted 52 | .gd .x { color: #000; background-color: #faa } // Generic.Deleted.Specific 53 | .ge { font-style: italic } // Generic.Emph 54 | .gr { color: #a00 } // Generic.Error 55 | .gh { color: #999 } // Generic.Heading 56 | .gi { color: #000; background-color: #dfd } // Generic.Inserted 57 | .gi .x { color: #000; background-color: #afa } // Generic.Inserted.Specific 58 | .go { color: #888 } // Generic.Output 59 | .gp { color: #555 } // Generic.Prompt 60 | .gs { font-weight: bold } // Generic.Strong 61 | .gu { color: #aaa } // Generic.Subheading 62 | .gt { color: #a00 } // Generic.Traceback 63 | .kc { font-weight: bold } // Keyword.Constant 64 | .kd { font-weight: bold } // Keyword.Declaration 65 | .kp { font-weight: bold } // Keyword.Pseudo 66 | .kr { font-weight: bold } // Keyword.Reserved 67 | .kt { color: #458; font-weight: bold } // Keyword.Type 68 | .m { color: #099 } // Literal.Number 69 | .s { color: #d14 } // Literal.String 70 | .na { color: #008080 } // Name.Attribute 71 | .nb { color: #0086B3 } // Name.Builtin 72 | .nc { color: #458; font-weight: bold } // Name.Class 73 | .no { color: #008080 } // Name.Constant 74 | .ni { color: #800080 } // Name.Entity 75 | .ne { color: #900; font-weight: bold } // Name.Exception 76 | .nf { color: #900; font-weight: bold } // Name.Function 77 | .nn { color: #555 } // Name.Namespace 78 | .nt { color: #000080 } // Name.Tag 79 | .nv { color: #008080 } // Name.Variable 80 | .ow { font-weight: bold } // Operator.Word 81 | .w { color: #bbb } // Text.Whitespace 82 | .mf { color: #099 } // Literal.Number.Float 83 | .mh { color: #099 } // Literal.Number.Hex 84 | .mi { color: #099 } // Literal.Number.Integer 85 | .mo { color: #099 } // Literal.Number.Oct 86 | .sb { color: #d14 } // Literal.String.Backtick 87 | .sc { color: #d14 } // Literal.String.Char 88 | .sd { color: #d14 } // Literal.String.Doc 89 | .s2 { color: #d14 } // Literal.String.Double 90 | .se { color: #d14 } // Literal.String.Escape 91 | .sh { color: #d14 } // Literal.String.Heredoc 92 | .si { color: #d14 } // Literal.String.Interpol 93 | .sx { color: #d14 } // Literal.String.Other 94 | .sr { color: #009926 } // Literal.String.Regex 95 | .s1 { color: #d14 } // Literal.String.Single 96 | .ss { color: #990073 } // Literal.String.Symbol 97 | .bp { color: #999 } // Name.Builtin.Pseudo 98 | .vc { color: #008080 } // Name.Variable.Class 99 | .vg { color: #008080 } // Name.Variable.Global 100 | .vi { color: #008080 } // Name.Variable.Instance 101 | .il { color: #099 } // Literal.Number.Integer.Long 102 | } 103 | } 104 | 105 | 106 | // Dark mode 107 | // --------- 108 | 109 | $dm-brand-color: #999999 !default; 110 | $dm-brand-color-light: lighten($dm-brand-color, 5%) !default; 111 | $dm-brand-color-dark: darken($dm-brand-color, 35%) !default; 112 | 113 | $dm-site-title-color: $dm-brand-color-light !default; 114 | 115 | $dm-text-color: #bbbbbb !default; 116 | $dm-background-color: #181818 !default; 117 | $dm-code-background-color: #212121 !default; 118 | 119 | $dm-link-base-color: #79b8ff !default; 120 | $dm-link-visited-color: $dm-link-base-color !default; 121 | $dm-link-hover-color: $dm-text-color !default; 122 | 123 | $dm-border-color-01: $dm-brand-color-dark !default; 124 | $dm-border-color-02: $dm-brand-color-light !default; 125 | $dm-border-color-03: $dm-brand-color !default; 126 | 127 | $dm-table-text-color: $dm-text-color !default; 128 | $dm-table-zebra-color: lighten($dm-background-color, 4%) !default; 129 | $dm-table-header-bg-color: lighten($dm-background-color, 10%) !default; 130 | $dm-table-header-border: lighten($dm-background-color, 21%) !default; 131 | $dm-table-border-color: $dm-border-color-01 !default; 132 | 133 | 134 | // Syntax highlighting styles should be adjusted appropriately for every "skin" 135 | // List of tokens: https://github.com/rouge-ruby/rouge/wiki/List-of-tokens 136 | // Some colors come from Material Theme Darker: 137 | // https://github.com/material-theme/vsc-material-theme/blob/master/scripts/generator/settings/specific/darker-hc.ts 138 | // https://github.com/material-theme/vsc-material-theme/blob/master/scripts/generator/color-set.ts 139 | // ---------------------------------------------------------------------------- 140 | 141 | @mixin dm-highlight { 142 | .highlight { 143 | .c { color: #545454; font-style: italic } // Comment 144 | .err { color: #f07178; background-color: #e3d2d2 } // Error 145 | .k { color: #89DDFF; font-weight: bold } // Keyword 146 | .o { font-weight: bold } // Operator 147 | .cm { color: #545454; font-style: italic } // Comment.Multiline 148 | .cp { color: #545454; font-weight: bold } // Comment.Preproc 149 | .c1 { color: #545454; font-style: italic } // Comment.Single 150 | .cs { color: #545454; font-weight: bold; font-style: italic } // Comment.Special 151 | .gd { color: #000; background-color: #fdd } // Generic.Deleted 152 | .gd .x { color: #000; background-color: #faa } // Generic.Deleted.Specific 153 | .ge { font-style: italic } // Generic.Emph 154 | .gr { color: #f07178 } // Generic.Error 155 | .gh { color: #999 } // Generic.Heading 156 | .gi { color: #000; background-color: #dfd } // Generic.Inserted 157 | .gi .x { color: #000; background-color: #afa } // Generic.Inserted.Specific 158 | .go { color: #888 } // Generic.Output 159 | .gp { color: #555 } // Generic.Prompt 160 | .gs { font-weight: bold } // Generic.Strong 161 | .gu { color: #aaa } // Generic.Subheading 162 | .gt { color: #f07178 } // Generic.Traceback 163 | .kc { font-weight: bold } // Keyword.Constant 164 | .kd { font-weight: bold } // Keyword.Declaration 165 | .kp { font-weight: bold } // Keyword.Pseudo 166 | .kr { font-weight: bold } // Keyword.Reserved 167 | .kt { color: #FFCB6B; font-weight: bold } // Keyword.Type 168 | .m { color: #F78C6C } // Literal.Number 169 | .s { color: #C3E88D } // Literal.String 170 | .na { color: #008080 } // Name.Attribute 171 | .nb { color: #EEFFFF } // Name.Builtin 172 | .nc { color: #FFCB6B; font-weight: bold } // Name.Class 173 | .no { color: #008080 } // Name.Constant 174 | .ni { color: #800080 } // Name.Entity 175 | .ne { color: #900; font-weight: bold } // Name.Exception 176 | .nf { color: #82AAFF; font-weight: bold } // Name.Function 177 | .nn { color: #555 } // Name.Namespace 178 | .nt { color: #FFCB6B } // Name.Tag 179 | .nv { color: #EEFFFF } // Name.Variable 180 | .ow { font-weight: bold } // Operator.Word 181 | .w { color: #EEFFFF } // Text.Whitespace 182 | .mf { color: #F78C6C } // Literal.Number.Float 183 | .mh { color: #F78C6C } // Literal.Number.Hex 184 | .mi { color: #F78C6C } // Literal.Number.Integer 185 | .mo { color: #F78C6C } // Literal.Number.Oct 186 | .sb { color: #C3E88D } // Literal.String.Backtick 187 | .sc { color: #C3E88D } // Literal.String.Char 188 | .sd { color: #C3E88D } // Literal.String.Doc 189 | .s2 { color: #C3E88D } // Literal.String.Double 190 | .se { color: #EEFFFF } // Literal.String.Escape 191 | .sh { color: #C3E88D } // Literal.String.Heredoc 192 | .si { color: #C3E88D } // Literal.String.Interpol 193 | .sx { color: #C3E88D } // Literal.String.Other 194 | .sr { color: #C3E88D } // Literal.String.Regex 195 | .s1 { color: #C3E88D } // Literal.String.Single 196 | .ss { color: #C3E88D } // Literal.String.Symbol 197 | .bp { color: #999 } // Name.Builtin.Pseudo 198 | .vc { color: #FFCB6B } // Name.Variable.Class 199 | .vg { color: #EEFFFF } // Name.Variable.Global 200 | .vi { color: #EEFFFF } // Name.Variable.Instance 201 | .il { color: #F78C6C } // Literal.Number.Integer.Long 202 | } 203 | } 204 | 205 | 206 | // Mode selection 207 | // -------------- 208 | 209 | 210 | // Classic skin (always light mode) 211 | // Assign outside of the if construct to establish global variable scope 212 | 213 | $brand-color: $lm-brand-color; 214 | $brand-color-light: $lm-brand-color-light; 215 | $brand-color-dark: $lm-brand-color-dark; 216 | 217 | $site-title-color: $lm-site-title-color; 218 | 219 | $text-color: $lm-text-color; 220 | $background-color: $lm-background-color; 221 | $code-background-color: $lm-code-background-color; 222 | 223 | $link-base-color: $lm-link-base-color; 224 | $link-visited-color: $lm-link-visited-color; 225 | $link-hover-color: $lm-link-hover-color; 226 | 227 | $border-color-01: $lm-border-color-01; 228 | $border-color-02: $lm-border-color-02; 229 | $border-color-03: $lm-border-color-03; 230 | 231 | $table-text-color: $lm-table-text-color; 232 | $table-zebra-color: $lm-table-zebra-color; 233 | $table-header-bg-color: $lm-table-header-bg-color; 234 | $table-header-border: $lm-table-header-border; 235 | $table-border-color: $lm-table-border-color; 236 | 237 | 238 | @if $color-scheme-auto { 239 | 240 | // Auto mode 241 | 242 | :root { 243 | --minima-brand-color: #{$lm-brand-color}; 244 | --minima-brand-color-light: #{$lm-brand-color-light}; 245 | --minima-brand-color-dark: #{$lm-brand-color-dark}; 246 | 247 | --minima-site-title-color: #{$lm-site-title-color}; 248 | 249 | --minima-text-color: #{$lm-text-color}; 250 | --minima-background-color: #{$lm-background-color}; 251 | --minima-code-background-color: #{$lm-code-background-color}; 252 | 253 | --minima-link-base-color: #{$lm-link-base-color}; 254 | --minima-link-visited-color: #{$lm-link-visited-color}; 255 | --minima-link-hover-color: #{$lm-link-hover-color}; 256 | 257 | --minima-border-color-01: #{$lm-border-color-01}; 258 | --minima-border-color-02: #{$lm-border-color-02}; 259 | --minima-border-color-03: #{$lm-border-color-03}; 260 | 261 | --minima-table-text-color: #{$lm-table-text-color}; 262 | --minima-table-zebra-color: #{$lm-table-zebra-color}; 263 | --minima-table-header-bg-color: #{$lm-table-header-bg-color}; 264 | --minima-table-header-border: #{$lm-table-header-border}; 265 | --minima-table-border-color: #{$lm-table-border-color}; 266 | } 267 | 268 | @include lm-highlight; 269 | 270 | @media (prefers-color-scheme: dark) { 271 | :root { 272 | --minima-brand-color: #{$dm-brand-color}; 273 | --minima-brand-color-light: #{$dm-brand-color-light}; 274 | --minima-brand-color-dark: #{$dm-brand-color-dark}; 275 | 276 | --minima-site-title-color: #{$dm-site-title-color}; 277 | 278 | --minima-text-color: #{$dm-text-color}; 279 | --minima-background-color: #{$dm-background-color}; 280 | --minima-code-background-color: #{$dm-code-background-color}; 281 | 282 | --minima-link-base-color: #{$dm-link-base-color}; 283 | --minima-link-visited-color: #{$dm-link-visited-color}; 284 | --minima-link-hover-color: #{$dm-link-hover-color}; 285 | 286 | --minima-border-color-01: #{$dm-border-color-01}; 287 | --minima-border-color-02: #{$dm-border-color-02}; 288 | --minima-border-color-03: #{$dm-border-color-03}; 289 | 290 | --minima-table-text-color: #{$dm-table-text-color}; 291 | --minima-table-zebra-color: #{$dm-table-zebra-color}; 292 | --minima-table-header-bg-color: #{$dm-table-header-bg-color}; 293 | --minima-table-header-border: #{$dm-table-header-border}; 294 | --minima-table-border-color: #{$dm-table-border-color}; 295 | } 296 | 297 | @include dm-highlight; 298 | } 299 | 300 | $brand-color: var(--minima-brand-color); 301 | $brand-color-light: var(--minima-brand-color-light); 302 | $brand-color-dark: var(--minima-brand-color-dark); 303 | 304 | $site-title-color: var(--minima-site-title-color); 305 | 306 | $text-color: var(--minima-text-color); 307 | $background-color: var(--minima-background-color); 308 | $code-background-color: var(--minima-code-background-color); 309 | 310 | $link-base-color: var(--minima-link-base-color); 311 | $link-visited-color: var(--minima-link-visited-color); 312 | $link-hover-color: var(--minima-link-hover-color); 313 | 314 | $border-color-01: var(--minima-border-color-01); 315 | $border-color-02: var(--minima-border-color-02); 316 | $border-color-03: var(--minima-border-color-03); 317 | 318 | $table-text-color: var(--minima-table-text-color); 319 | $table-zebra-color: var(--minima-table-zebra-color); 320 | $table-header-bg-color: var(--minima-table-header-bg-color); 321 | $table-header-border: var(--minima-table-header-border); 322 | $table-border-color: var(--minima-table-border-color); 323 | 324 | 325 | } @else if $color-scheme-dark { 326 | 327 | // Dark skin (always dark mode) 328 | 329 | $brand-color: $dm-brand-color; 330 | $brand-color-light: $dm-brand-color-light; 331 | $brand-color-dark: $dm-brand-color-dark; 332 | 333 | $site-title-color: $dm-site-title-color; 334 | 335 | $text-color: $dm-text-color; 336 | $background-color: $dm-background-color; 337 | $code-background-color: $dm-code-background-color; 338 | 339 | $link-base-color: $dm-link-base-color; 340 | $link-visited-color: $dm-link-visited-color; 341 | $link-hover-color: $dm-link-hover-color; 342 | 343 | $border-color-01: $dm-border-color-01; 344 | $border-color-02: $dm-border-color-02; 345 | $border-color-03: $dm-border-color-03; 346 | 347 | $table-text-color: $dm-table-text-color; 348 | $table-zebra-color: $dm-table-zebra-color; 349 | $table-header-bg-color: $dm-table-header-bg-color; 350 | $table-header-border: $dm-table-header-border; 351 | $table-border-color: $dm-table-border-color; 352 | 353 | @include dm-highlight; 354 | 355 | 356 | } @else { 357 | 358 | // Classic skin syntax highlighting 359 | @include lm-highlight; 360 | 361 | } 362 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/classic.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | $color-scheme-auto: false; 4 | $color-scheme-dark: false; 5 | @import "minima/skins/auto"; 6 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/dark.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | $color-scheme-auto: false; 4 | $color-scheme-dark: true; 5 | @import "minima/skins/auto"; 6 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/solarized-dark.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | $sol-is-auto: false; 4 | $sol-is-dark: true; 5 | @import "minima/skins/solarized"; 6 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/solarized-light.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | $sol-is-auto: false; 4 | @import "minima/skins/solarized"; 5 | -------------------------------------------------------------------------------- /docs/_sass/minima/skins/solarized.scss: -------------------------------------------------------------------------------- 1 | @charset "utf-8"; 2 | 3 | // Solarized skin 4 | // ============== 5 | // Created by Sander Voerman using the Solarized 6 | // color scheme by Ethan Schoonover . 7 | 8 | // This style sheet implements three options for the minima.skin setting: 9 | // "solarized-light" for light mode, "solarized-dark" for dark mode, and 10 | // "solarized" for light or dark mode depending on user preference. 11 | $sol-is-auto: true !default; 12 | $sol-is-dark: false !default; 13 | 14 | 15 | // Color scheme 16 | // ------------ 17 | // The inline comments show the canonical L*a*b values for each color. 18 | 19 | $sol-base03: #002b36; // 15 -12 -12 20 | $sol-base02: #073642; // 20 -12 -12 21 | $sol-base01: #586e75; // 45 -07 -07 22 | $sol-base00: #657b83; // 50 -07 -07 23 | $sol-base0: #839496; // 60 -06 -03 24 | $sol-base1: #93a1a1; // 65 -05 -02 25 | $sol-base2: #eee8d5; // 92 -00 10 26 | $sol-base3: #fdf6e3; // 97 00 10 27 | $sol-yellow: #b58900; // 60 10 65 28 | $sol-orange: #cb4b16; // 50 50 55 29 | $sol-red: #dc322f; // 50 65 45 30 | $sol-magenta: #d33682; // 50 65 -05 31 | $sol-violet: #6c71c4; // 50 15 -45 32 | $sol-blue: #268bd2; // 55 -10 -45 33 | $sol-cyan: #2aa198; // 60 -35 -05 34 | $sol-green: #859900; // 60 -20 65 35 | 36 | 37 | // Mixed colors 38 | // ------------ 39 | // While not part of the original Solarized base tones, these derived tones 40 | // are meant to replicate the visual style of the classic skin. They should 41 | // not be used in cases where sufficiently contrasting colors are needed. 42 | 43 | $sol-light-mix1: mix($sol-base1, $sol-base3); 44 | $sol-light-mix2: mix($sol-blue, $sol-base00); 45 | $sol-light-mix3: mix($sol-base2, $sol-base3); 46 | $sol-dark-mix1: mix($sol-base01, $sol-base03); 47 | $sol-dark-mix2: mix($sol-blue, $sol-base0); 48 | $sol-dark-mix3: mix($sol-base02, $sol-base03); 49 | 50 | 51 | // Mode selection 52 | // -------------- 53 | 54 | $sol-mono3: $sol-base3; 55 | $sol-mono2: $sol-base2; 56 | $sol-mono1: $sol-base1; 57 | $sol-mono00: $sol-base00; 58 | $sol-mono01: $sol-base01; 59 | $sol-mix1: $sol-light-mix1; 60 | $sol-mix2: $sol-light-mix2; 61 | $sol-mix3: $sol-light-mix3; 62 | 63 | @if $sol-is-dark { 64 | $sol-mono3: $sol-base03; 65 | $sol-mono2: $sol-base02; 66 | $sol-mono1: $sol-base01; 67 | $sol-mono00: $sol-base0; 68 | $sol-mono01: $sol-base1; 69 | $sol-mix1: $sol-dark-mix1; 70 | $sol-mix2: $sol-dark-mix2; 71 | $sol-mix3: $sol-dark-mix3; 72 | } 73 | 74 | @if $sol-is-auto { 75 | :root { 76 | --solarized-mono3: #{$sol-base3}; 77 | --solarized-mono2: #{$sol-base2}; 78 | --solarized-mono1: #{$sol-base1}; 79 | --solarized-mono00: #{$sol-base00}; 80 | --solarized-mono01: #{$sol-base01}; 81 | --solarized-mix1: #{$sol-light-mix1}; 82 | --solarized-mix2: #{$sol-light-mix2}; 83 | --solarized-mix3: #{$sol-light-mix3}; 84 | } 85 | 86 | @media (prefers-color-scheme: dark) { 87 | :root { 88 | --solarized-mono3: #{$sol-base03}; 89 | --solarized-mono2: #{$sol-base02}; 90 | --solarized-mono1: #{$sol-base01}; 91 | --solarized-mono00: #{$sol-base0}; 92 | --solarized-mono01: #{$sol-base1}; 93 | --solarized-mix1: #{$sol-dark-mix1}; 94 | --solarized-mix2: #{$sol-dark-mix2}; 95 | --solarized-mix3: #{$sol-dark-mix3}; 96 | } 97 | } 98 | 99 | $sol-mono3: var(--solarized-mono3); 100 | $sol-mono2: var(--solarized-mono2); 101 | $sol-mono1: var(--solarized-mono1); 102 | $sol-mono00: var(--solarized-mono00); 103 | $sol-mono01: var(--solarized-mono01); 104 | $sol-mix1: var(--solarized-mix1); 105 | $sol-mix2: var(--solarized-mix2); 106 | $sol-mix3: var(--solarized-mix3); 107 | } 108 | 109 | 110 | // Minima color variables 111 | // ---------------------- 112 | 113 | $brand-color: $sol-mono1 !default; 114 | $brand-color-light: $sol-mix1 !default; 115 | $brand-color-dark: $sol-mono00 !default; 116 | 117 | $site-title-color: $sol-mono00 !default; 118 | 119 | $text-color: $sol-mono01 !default; 120 | $background-color: $sol-mono3 !default; 121 | $code-background-color: $sol-mono2 !default; 122 | 123 | $link-base-color: $sol-blue !default; 124 | $link-visited-color: $sol-mix2 !default; 125 | $link-hover-color: $sol-mono00 !default; 126 | 127 | $border-color-01: $brand-color-light !default; 128 | $border-color-02: $sol-mono1 !default; 129 | $border-color-03: $sol-mono00 !default; 130 | 131 | $table-text-color: $sol-mono00 !default; 132 | $table-zebra-color: $sol-mix3 !default; 133 | $table-header-bg-color: $sol-mono2 !default; 134 | $table-header-border: $sol-mono1 !default; 135 | $table-border-color: $sol-mono1 !default; 136 | 137 | 138 | // Syntax highlighting styles 139 | // -------------------------- 140 | 141 | .highlight { 142 | .c { color: $sol-mono1; font-style: italic } // Comment 143 | .err { color: $sol-red } // Error 144 | .k { color: $sol-mono01; font-weight: bold } // Keyword 145 | .o { color: $sol-mono01; font-weight: bold } // Operator 146 | .cm { color: $sol-mono1; font-style: italic } // Comment.Multiline 147 | .cp { color: $sol-mono1; font-weight: bold } // Comment.Preproc 148 | .c1 { color: $sol-mono1; font-style: italic } // Comment.Single 149 | .cs { color: $sol-mono1; font-weight: bold; font-style: italic } // Comment.Special 150 | .gd { color: $sol-red } // Generic.Deleted 151 | .gd .x { color: $sol-red } // Generic.Deleted.Specific 152 | .ge { color: $sol-mono00; font-style: italic } // Generic.Emph 153 | .gr { color: $sol-red } // Generic.Error 154 | .gh { color: $sol-mono1 } // Generic.Heading 155 | .gi { color: $sol-green } // Generic.Inserted 156 | .gi .x { color: $sol-green } // Generic.Inserted.Specific 157 | .go { color: $sol-mono00 } // Generic.Output 158 | .gp { color: $sol-mono00 } // Generic.Prompt 159 | .gs { color: $sol-mono01; font-weight: bold } // Generic.Strong 160 | .gu { color: $sol-mono1 } // Generic.Subheading 161 | .gt { color: $sol-red } // Generic.Traceback 162 | .kc { color: $sol-mono01; font-weight: bold } // Keyword.Constant 163 | .kd { color: $sol-mono01; font-weight: bold } // Keyword.Declaration 164 | .kp { color: $sol-mono01; font-weight: bold } // Keyword.Pseudo 165 | .kr { color: $sol-mono01; font-weight: bold } // Keyword.Reserved 166 | .kt { color: $sol-violet; font-weight: bold } // Keyword.Type 167 | .m { color: $sol-cyan } // Literal.Number 168 | .s { color: $sol-magenta } // Literal.String 169 | .na { color: $sol-cyan } // Name.Attribute 170 | .nb { color: $sol-blue } // Name.Builtin 171 | .nc { color: $sol-violet; font-weight: bold } // Name.Class 172 | .no { color: $sol-cyan } // Name.Constant 173 | .ni { color: $sol-violet } // Name.Entity 174 | .ne { color: $sol-violet; font-weight: bold } // Name.Exception 175 | .nf { color: $sol-blue; font-weight: bold } // Name.Function 176 | .nn { color: $sol-mono00 } // Name.Namespace 177 | .nt { color: $sol-blue } // Name.Tag 178 | .nv { color: $sol-cyan } // Name.Variable 179 | .ow { color: $sol-mono01; font-weight: bold } // Operator.Word 180 | .w { color: $sol-mono1 } // Text.Whitespace 181 | .mf { color: $sol-cyan } // Literal.Number.Float 182 | .mh { color: $sol-cyan } // Literal.Number.Hex 183 | .mi { color: $sol-cyan } // Literal.Number.Integer 184 | .mo { color: $sol-cyan } // Literal.Number.Oct 185 | .sb { color: $sol-magenta } // Literal.String.Backtick 186 | .sc { color: $sol-magenta } // Literal.String.Char 187 | .sd { color: $sol-magenta } // Literal.String.Doc 188 | .s2 { color: $sol-magenta } // Literal.String.Double 189 | .se { color: $sol-magenta } // Literal.String.Escape 190 | .sh { color: $sol-magenta } // Literal.String.Heredoc 191 | .si { color: $sol-magenta } // Literal.String.Interpol 192 | .sx { color: $sol-magenta } // Literal.String.Other 193 | .sr { color: $sol-green } // Literal.String.Regex 194 | .s1 { color: $sol-magenta } // Literal.String.Single 195 | .ss { color: $sol-magenta } // Literal.String.Symbol 196 | .bp { color: $sol-mono1 } // Name.Builtin.Pseudo 197 | .vc { color: $sol-cyan } // Name.Variable.Class 198 | .vg { color: $sol-cyan } // Name.Variable.Global 199 | .vi { color: $sol-cyan } // Name.Variable.Instance 200 | .il { color: $sol-cyan } // Literal.Number.Integer.Long 201 | } 202 | -------------------------------------------------------------------------------- /docs/assets/DejaVuSansCircle.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/docs/assets/DejaVuSansCircle.png -------------------------------------------------------------------------------- /docs/assets/css/style.scss: -------------------------------------------------------------------------------- 1 | --- 2 | # Only the main Sass file needs front matter (the dashes are enough) 3 | --- 4 | 5 | @import 6 | "minima/skins/{{ site.minima.skin | default: 'classic' }}", 7 | "minima/initialize"; 8 | -------------------------------------------------------------------------------- /docs/assets/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/docs/assets/favicon.ico -------------------------------------------------------------------------------- /docs/index.md: -------------------------------------------------------------------------------- 1 | --- 2 | layout: home 3 | --- 4 | 5 | ![GitHub](https://img.shields.io/github/license/charoitel/lambda-layer-canvas-nodejs)  ![Watch on GitHub](https://img.shields.io/github/watchers/charoitel/lambda-node-canvas.svg?style=social)  ![Fork on GitHub](https://img.shields.io/github/forks/charoitel/lambda-node-canvas.svg?style=social)  ![Star on GitHub](https://img.shields.io/github/stars/charoitel/lambda-node-canvas.svg?style=social) 6 | 7 | Canvas Layer for AWS Lambda is published and available on [AWS Serverless Application Repository](https://serverlessrepo.aws.amazon.com/applications/arn:aws:serverlessrepo:us-east-1:990551184979:applications~lambda-layer-canvas-nodejs), and GitHub at [charoitel/lambda-layer-canvas-nodejs](https://github.com/charoitel/lambda-layer-canvas-nodejs). The layer aims to provide a Cairo backed Mozilla Web Canvas API implementation layer for AWS Lambda, powered by [node-canvas](https://github.com/Automattic/node-canvas). 8 | 9 | ## About node-canvas 10 | 11 | [node-canvas](https://github.com/Automattic/node-canvas) is a Cairo backed Canvas implementation for Node.js. It implements the [Mozilla Web Canvas API](https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API) as closely as possible. For the latest API compliance, you may check [Compatibility Status](https://github.com/Automattic/node-canvas/wiki/Compatibility-Status). 12 | 13 | ## How this layer is built? 14 | 15 | The layer is built from source of node-canvas npm package on [amazonlinux](https://hub.docker.com/_/amazonlinux) dev container instance, with following native dependencies installed. You may check the build layer script, ``` build-layer.sh ```, which is available in repository, for details. 16 | 17 | ```bash 18 | gcc-c++ cairo-devel pango-devel libjpeg-turbo-devel giflib-devel librsvg2-devel pango-devel bzip2-devel jq python3 19 | ``` 20 | 21 | Since AWS Lambda is a secure and isolated runtime and execution environment, the layer aims to target AWS Lambda compatible and native build. As there are canvas libraries and frameworks relying on node-canvas running on Node.js runtime, this layer may also try to include and support those libraries and frameworks. Currently, following libraries and frameworks are included when building and packaging the layer: 22 | 23 | - [Chart.js](#chartjs-support) 24 | - [Fabric.js](#fabricjs-support) 25 | - [Konva](#konva-support) 26 | 27 | ### Chart.js support 28 | 29 | [Chart.js](https://github.com/chartjs/chart.js) provides a set of frequently used chart types, plugins, and customization options. In addition to a reasonable set of built-in chart types, there are also community-maintained chart types. 30 | 31 | ### Fabric.js support 32 | 33 | [Fabric.js](https://github.com/fabricjs/fabric.js) provides a missing and interactive object model for canvas, as well as an SVG parser, layer of interactivity, and a whole suite of other indispensable tools. 34 | 35 | ### Konva support 36 | 37 | [Konva](https://github.com/konvajs/konva) enables high performance animations, transitions, node nesting, layering, filtering, caching, event handling for desktop and mobile applications, and much more. 38 | -------------------------------------------------------------------------------- /docs/minima.gemspec: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Gem::Specification.new do |spec| 4 | spec.name = "minima" 5 | spec.version = "3.0.0.dev" 6 | spec.authors = ["Joel Glovier"] 7 | spec.email = ["jglovier@github.com"] 8 | 9 | spec.summary = "A beautiful, minimal theme for Jekyll." 10 | spec.homepage = "https://github.com/jekyll/minima" 11 | spec.license = "MIT" 12 | 13 | spec.metadata["plugin_type"] = "theme" 14 | 15 | spec.files = `git ls-files -z`.split("\x0").select do |f| 16 | f.match(%r!^(assets|_(includes|layouts|sass)/|(LICENSE|README)((\.(txt|md|markdown)|$)))!i) 17 | end 18 | 19 | spec.add_runtime_dependency "jekyll", ">= 3.5", "< 5.0" 20 | spec.add_runtime_dependency "jekyll-feed", "~> 0.9" 21 | spec.add_runtime_dependency "jekyll-seo-tag", "~> 2.1" 22 | 23 | spec.add_development_dependency "bundler" 24 | end 25 | -------------------------------------------------------------------------------- /docs/script/bootstrap: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | set -e 4 | 5 | gem install bundler 6 | bundle install 7 | -------------------------------------------------------------------------------- /docs/script/build: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | set -e 4 | 5 | echo "Building the example site..." 6 | bundle exec jekyll build 7 | -------------------------------------------------------------------------------- /docs/script/cibuild: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | set -e 4 | 5 | script/build 6 | 7 | if test -e "./_site/index.html";then 8 | echo "It builds!" 9 | rm -Rf _site 10 | else 11 | echo "Huh. That's odd. The example site doesn't seem to build." 12 | exit 1 13 | fi 14 | 15 | gem build minima.gemspec 16 | -------------------------------------------------------------------------------- /docs/script/server: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | 3 | bundle exec jekyll serve 4 | -------------------------------------------------------------------------------- /lib/libX11.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libX11.so.6 -------------------------------------------------------------------------------- /lib/libXau.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libXau.so.6 -------------------------------------------------------------------------------- /lib/libXext.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libXext.so.6 -------------------------------------------------------------------------------- /lib/libXrender.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libXrender.so.1 -------------------------------------------------------------------------------- /lib/libblkid.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libblkid.so.1 -------------------------------------------------------------------------------- /lib/libbrotlicommon.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libbrotlicommon.so.1 -------------------------------------------------------------------------------- /lib/libbrotlidec.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libbrotlidec.so.1 -------------------------------------------------------------------------------- /lib/libbz2.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libbz2.so.1 -------------------------------------------------------------------------------- /lib/libc.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libc.so.6 -------------------------------------------------------------------------------- /lib/libcairo-gobject.so.2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libcairo-gobject.so.2 -------------------------------------------------------------------------------- /lib/libcairo.so.2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libcairo.so.2 -------------------------------------------------------------------------------- /lib/libdatrie.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libdatrie.so.1 -------------------------------------------------------------------------------- /lib/libffi.so.8: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libffi.so.8 -------------------------------------------------------------------------------- /lib/libfontconfig.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libfontconfig.so.1 -------------------------------------------------------------------------------- /lib/libfreetype.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libfreetype.so.6 -------------------------------------------------------------------------------- /lib/libfribidi.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libfribidi.so.0 -------------------------------------------------------------------------------- /lib/libgcc_s.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgcc_s.so.1 -------------------------------------------------------------------------------- /lib/libgdk_pixbuf-2.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgdk_pixbuf-2.0.so.0 -------------------------------------------------------------------------------- /lib/libgif.so.7: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgif.so.7 -------------------------------------------------------------------------------- /lib/libgio-2.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgio-2.0.so.0 -------------------------------------------------------------------------------- /lib/libglib-2.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libglib-2.0.so.0 -------------------------------------------------------------------------------- /lib/libgmodule-2.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgmodule-2.0.so.0 -------------------------------------------------------------------------------- /lib/libgobject-2.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgobject-2.0.so.0 -------------------------------------------------------------------------------- /lib/libgraphite2.so.3: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libgraphite2.so.3 -------------------------------------------------------------------------------- /lib/libharfbuzz.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libharfbuzz.so.0 -------------------------------------------------------------------------------- /lib/libjpeg.so.62: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libjpeg.so.62 -------------------------------------------------------------------------------- /lib/liblzma.so.5: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/liblzma.so.5 -------------------------------------------------------------------------------- /lib/libm.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libm.so.6 -------------------------------------------------------------------------------- /lib/libmount.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libmount.so.1 -------------------------------------------------------------------------------- /lib/libpango-1.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpango-1.0.so.0 -------------------------------------------------------------------------------- /lib/libpangocairo-1.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpangocairo-1.0.so.0 -------------------------------------------------------------------------------- /lib/libpangoft2-1.0.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpangoft2-1.0.so.0 -------------------------------------------------------------------------------- /lib/libpcre2-8.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpcre2-8.so.0 -------------------------------------------------------------------------------- /lib/libpixman-1.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpixman-1.so.0 -------------------------------------------------------------------------------- /lib/libpng16.so.16: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libpng16.so.16 -------------------------------------------------------------------------------- /lib/librsvg-2.so.2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/librsvg-2.so.2 -------------------------------------------------------------------------------- /lib/libselinux.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libselinux.so.1 -------------------------------------------------------------------------------- /lib/libstdc++.so.6: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libstdc++.so.6 -------------------------------------------------------------------------------- /lib/libthai.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libthai.so.0 -------------------------------------------------------------------------------- /lib/libxcb-render.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libxcb-render.so.0 -------------------------------------------------------------------------------- /lib/libxcb-shm.so.0: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libxcb-shm.so.0 -------------------------------------------------------------------------------- /lib/libxcb.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libxcb.so.1 -------------------------------------------------------------------------------- /lib/libxml2.so.2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libxml2.so.2 -------------------------------------------------------------------------------- /lib/libz.so.1: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/charoitel/lambda-layer-canvas-nodejs/3a2157addadf0dbd46cf83ca9a36f4aa96c51ac1/lib/libz.so.1 -------------------------------------------------------------------------------- /nodejs/test/canvas-font.js: -------------------------------------------------------------------------------- 1 | let { registerFont, createCanvas } = require('canvas'); 2 | var assert = require('assert'); 3 | 4 | describe('Font', function() { 5 | describe('#registerFont', function() { 6 | it('should register defined Font by name', function() { 7 | registerFont('/usr/share/fonts/dejavu-sans-fonts/DejaVuSans.ttf', { family: 'DejaVu Sans' }); 8 | }); 9 | }), 10 | describe('#fillText', function() { 11 | it('should write defined Text in canvas', function() { 12 | registerFont('/usr/share/fonts/dejavu-sans-fonts/DejaVuSans-Bold.ttf', { family: 'DejaVu Sans Bold' }); 13 | let canvas = createCanvas(300, 300); 14 | let ctx1 = canvas.getContext('2d'); 15 | ctx1.font = '30px'; 16 | ctx1.fillText('Default', 5, 25); 17 | let ctx2 = canvas.getContext('2d'); 18 | ctx2.font = '30px "DejaVu Sans"'; 19 | ctx2.fillText('DejaVu Sans', 5, 65); 20 | let ctx3 = canvas.getContext('2d'); 21 | ctx3.font = '30px "DejaVu Sans Bold"'; 22 | ctx3.fillText('DejaVu Sans Bold', 5, 95); 23 | // console.log(canvas.toDataURL()); 24 | assert.strictEqual(canvas.toDataURL(), ''); 25 | }) 26 | }); 27 | }); 28 | -------------------------------------------------------------------------------- /nodejs/test/canvas-svg.js: -------------------------------------------------------------------------------- 1 | let { registerFont, createCanvas } = require('canvas'); 2 | var assert = require('assert'); 3 | 4 | describe('SVG', function () { 5 | describe('#fillTextInSVG', function () { 6 | it('should write defined Text in canvas', function () { 7 | registerFont('/usr/share/fonts/dejavu-sans-fonts/DejaVuSans-Bold.ttf', { family: 'DejaVu Sans Bold' }); 8 | let canvas = createCanvas(300, 300, 'svg'); 9 | let ctx1 = canvas.getContext('2d'); 10 | ctx1.font = '30px'; 11 | ctx1.fillText('Default', 5, 25); 12 | let ctx2 = canvas.getContext('2d'); 13 | ctx2.font = '30px "DejaVu Sans"'; 14 | ctx2.fillText('DejaVu Sans', 5, 65); 15 | let ctx3 = canvas.getContext('2d'); 16 | ctx3.font = '30px "DejaVu Sans Bold"'; 17 | ctx3.fillText('DejaVu Sans Bold', 5, 95); 18 | // console.log(canvas.toDataURL()); 19 | assert.strictEqual(canvas.toDataURL(), ""); 20 | }) 21 | }); 22 | }); 23 | -------------------------------------------------------------------------------- /nodejs/test/chart-line.js: -------------------------------------------------------------------------------- 1 | let { registerFont, createCanvas } = require('canvas'); 2 | let { CategoryScale, Chart, LinearScale, LineController, LineElement, PointElement } = require('chart.js'); 3 | var assert = require('assert'); 4 | 5 | describe('Chart', function () { 6 | describe('#drawLineGraph', function () { 7 | it('should draw a line graph', function () { 8 | Chart.register([ 9 | CategoryScale, 10 | LineController, 11 | LineElement, 12 | LinearScale, 13 | PointElement 14 | ]); 15 | 16 | const canvas = createCanvas(400, 300); 17 | const chart = new Chart( 18 | canvas, // TypeScript needs "as any" here 19 | { 20 | type: 'line', 21 | data: { 22 | labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], 23 | datasets: [{ 24 | label: '# of Votes', 25 | data: [12, 19, 3, 5, 2, 3], 26 | borderColor: 'red' 27 | }] 28 | } 29 | } 30 | ); 31 | const dataURL = canvas.toDataURL(); 32 | // console.log(dataURL); 33 | chart.destroy(); 34 | }); 35 | }) 36 | }); -------------------------------------------------------------------------------- /nodejs/test/fabric-rectangle.js: -------------------------------------------------------------------------------- 1 | let fabric = require('fabric/node'); 2 | var assert = require('assert'); 3 | 4 | describe('Fabric', function() { 5 | describe('#drawRectangle', function() { 6 | it('should draw a red rectangle', function() { 7 | const canvas = new fabric.Canvas(null, { width: 100, height: 100 }); 8 | const rect = new fabric.Rect({ width: 20, height: 50, fill: '#ff0000' }); 9 | const text = new fabric.Text('fabric.js', { fill: 'blue', fontSize: 24 }); 10 | canvas.add(rect, text); 11 | canvas.renderAll(); 12 | // console.log(canvas.toDataURL()); 13 | assert.strictEqual(canvas.toDataURL(), ''); 14 | }); 15 | }) 16 | }); -------------------------------------------------------------------------------- /nodejs/test/konva-circle.js: -------------------------------------------------------------------------------- 1 | let Konva = require('konva'); 2 | var assert = require('assert'); 3 | const semver = require('semver'); 4 | 5 | describe('Konva', function () { 6 | describe('#drawCircle', function () { 7 | it('should draw a red circle', function () { 8 | const stage = new Konva.Stage({ 9 | width: 500, 10 | height: 500 11 | }); 12 | // then create layer 13 | var layer = new Konva.Layer(); 14 | 15 | // create our shape 16 | var circle = new Konva.Circle({ 17 | x: stage.width() / 2, 18 | y: stage.height() / 2, 19 | radius: 70, 20 | fill: 'red', 21 | stroke: 'black', 22 | strokeWidth: 4 23 | }); 24 | 25 | // add the shape to the layer 26 | layer.add(circle); 27 | 28 | // add the layer to the stage 29 | stage.add(layer); 30 | 31 | // draw the image 32 | layer.draw(); 33 | 34 | if (semver.lt(process.version, '20.0.0')) { 35 | assert.strictEqual(stage.toDataURL({pixelRatio: 1}), ''); 36 | } else { 37 | assert.strictEqual(stage.toDataURL({pixelRatio: 1}), ''); 38 | } 39 | }); 40 | }) 41 | }); -------------------------------------------------------------------------------- /package-sam.yml: -------------------------------------------------------------------------------- 1 | AWSTemplateFormatVersion : '2010-09-09' 2 | Transform: AWS::Serverless-2016-10-31 3 | Resources: 4 | NodeCanvasLayer: 5 | Type: AWS::Serverless::LayerVersion 6 | Properties: 7 | LayerName: canvas-nodejs 8 | Description: canvas@ 9 | ContentUri: s3:///canvas-nodejs_v.zip 10 | CompatibleRuntimes: 11 | - nodejs 12 | RetentionPolicy: Retain 13 | LicenseInfo: MIT 14 | 15 | Outputs: 16 | LayerVersion: 17 | Description: Layer ARN Reference 18 | Value: !Ref NodeCanvasLayer 19 | --------------------------------------------------------------------------------