@quasar/icon-factory
This node module outputs a set of SQUARE favicons, webicons, pwa-icons and electron-icons as well as iOS, Windows Store and MacOS icons from an original 1240x1240 square icon that retains transparency and also minifies the assets. It will also create splash screens for Cordova and even a minified svg.
It works cross-platform to generate those pesky .icns
and .ico
files used by Electron apps and in the case of the latter preferred by some browsers and webscrapers (favicon.ico) - even though modern development guidelines for Apple and Windows recommend using .png
.
It has two primary interfaces (with Quasar CTX and as a standalone CLI) and although it is built for the Quasar Framework, it should work anywhere you can run node. You can even import it and use it in your own pipelines if that's your thing. It is designed to be a very useful tool that you will be glad to have lying around.
You MUST use PNG. If you use a source image that is smaller than 1240x1240 some icons will be naively upscaled. If you do not use a square original, it will be cropped square from the center using the smaller dimension as width and height - but will never upscale - which can potentially result in non-square results. You have been warned.
A final note: You should always pad your icon design with about 1% of empty space. This is because you will lose aliasing resolution when downscaling, which means at smaller sizes your round icon (if it doesn't have padding) will seem to have flattened top, bottom, left and right sides.
Installation and Usage
Requires
- node / yarn
- Linux, MacOS or Windows
- A square image in png format that is at least 1240px x 1240px (much bigger will merely slow down the conversions)
- @quasar/cli version 1.0.0-beta.7 or higher (if building a new project) or @quasar/app v1.0.0-beta.25 or later in order to add this module as an app-extension.
Note for early adopters
Things have changed along the way to the RC, and if you have a version of the Icon Factory that is less than v1.0.0-beta.30 (you can find out by running $ quasar info
), please follow these instructions:
- delete the
.icon-factory/
and all its components - if its still there, delete the
quasar.icon-factory.json
file - run:
$ quasar ext remove @quasar/icon-factory
- move your icon source-file to
app-icon.png
in the root of your app - if you want a custom splashscreen, put that file at
app-splashscreen.png
- run:
$ quasar ext add @quasar/icon-factory
Install as an App Extension (Quasar v1.0+)
$ quasar ext add @quasar/icon-factory
If you are on Windows and seeing an error the likes of pngquant failed to build, make sure that libpng-dev is installed
, please do the following:
$ npm install --global --production windows-build-tools
The most important part (and indeed the only reason to use this extension) is pointing it at your shiny icon (and splashscreen if you are building for Cordova.) So the first thing you are reminded of is that you need to do that!
--------------------------- ATTENTION! -----------------------------
You must replace app-icon.png in the root folder of your project.
If you plan on building for Cordova, you must also replace the
app-splashscreen.png image in the same place. File details:
-> app-icon.png 1240x1240 (with transparency)
-> app-splashscreen.png 2436x2436 (transparency optional)
--------------------------------------------------------------------
Note: Please use a valid png of 1240x1240 pixels. If you choose an image that is not square or has smaller dimensions, the icon-factory will do its best, but the results will not be optimal. Transparency is recommended. PNG is required.
Then choose a minification strategy:
? Minify strategy to be used for development:
pngquant => quality: lossy | time: 1x
pngcrush => quality: lossless+ | time: 10x
optipng => quality: lossless+ | time: 4x
zopfli => quality: lossless++ | time: 80x
Note: we recommend using pngquant for dev because it is the fastest minification. Other targets will take more time, but that is highly dependent on both the mode and the underlying hardware.
You will be asked the same question for production. Our recommendation is to choose optipng
. It has the best time / quality trade-off for a lossless minification - but zopfli
WILL shave off a bit more bytes.
You will also be asked for a background color. This is used in the few cases that a background is required, as with Cordova splashscreens and Cordova iOS icons.
Triggering
The first time you start Quasar, icon-factory will create the images needed for the specific app artifacts. They will not automatically be added to git, so you will need to manage that yourself.
$ quasar dev --mode electron
You will also be asked which method of splashscreen generation you prefer, ranging from the mere placement of your logo upon the background color you specified, overlaying your icon on top of a splashscreen image, or just using the splashscreen image. If you aren't happy with the results, don't forget you can change it in quasar.extensions.json
.
The final option during the install phase is to "always rebuild", which is useful for fine-tuning e.g. background colors, but if you don't remove this flag in quasar.extensions.json (or set it to false), the icon-factory will always run and slow down your dev buildtime.
Intermediary Folder
The icon-factory makes an intermediary folder in the node_modules/@quasar/icon-factory/tmp
folder to host the images when you switch between dev and build. If you haven't changed the source icon, these will merely be copied to the right destination folders.
Changing the Source Image
If you don't change the source image for the icon or the splashscreen, you will see a default iamge that reminds you to do this. Any time you change the source image, its hash won't match what is recorded in the settings - so it will trigger a rebuild.
Changing the Settings
All relevant settings are stored in quasar.extensions.json
, and if you change them, you will notice that the extension is not rerun. To force the rerun, remove the hash of the respective target.
Uninstalling
Run:
$ quasar ext remove @quasar/icon-factory
This will remove the extension, its dependencies - but not any of the icons it created.
Special notes about Cordova (iOS and Android only)
If you choose to build icons for Cordova, on iOS they WILL have a colored background (because transparency is not allowed), and this is why you are asked for an RGB value during the install phase. (Android allows transparency, btw.) You can change this in the quasar.extensions.json, but be sure to use a valid hex code like: #c0ff33
.
There are three methods to create your splashscreens:
- Only use app-splashscreen.png (default)
- Generate with background color and icon
- Overlay app-icon.png centered on top of app-splashscreen.png
This colored background color will be used for the splashscreen (if you choose the background-color + icon or if your splashscreen is transparent. If you don't provide one (or your color code is invalid), black will be used. If you haven't already installed the cordova-plugin-splashscreen, the process will attempt to install the plugin first and then continue to build the icons before proceeding to the actual Cordova dev or build pipeline.
Splashscreens are obviously a little different depending on whether you are targeting iOS or Android. Please read this document to find out more:
How it Works
There are 5 things that Icon Factory does with your original file. It will create a set of webicons for your project, it will minify those icons, it can make special icns (Mac) / ico (Windows) files for apps that need them and it will create SVG assets. Finally it will sort these icons into folders.
Here is the description of these general functions that are used internally to compose icon sets and if you just want to use them, feel free:
- build - This function creates all assets requested from the sizes object or presets object.
-
minify - The default usage minifies all assets in the target folder in-place with
pngquant
. Compared to usingpngcrush --brute
, it is a relatively fast process. If you are not in a hurry and want the best results, consider using zopfli for your production assets. - icns - This will create the special MacOS (icns) and Windows icon (ico) files.
- favicon - This will create the classical and never going out of style favicon.ico
- svg - With this command you can trace the outlines of your PNG to create an SVG and style it with some options.
There are five composed "recipes" that will create icons for you and place them in the appropriate folders according to your needs and as assembled by our research:
- cordova (all icons, splashscreens, iOS & Android)
- electron (all platforms)
- pwa, spa, ssr (incl. chrome special icon name, favicon)
- kitchensink (all icons, all platforms)
Head's Up!
You may notice that very small icons (like 16x16 and 32x32) look a little strange. Achieving good results by simply downscaling to a very small size depends a great deal on your original, and it is highly recommended that you at least look at all of the icons before you publish your project to a store. While integration testing can make sure that you have an asset, judging the ability of your small icon to express the same content as a large one is highly subjective and something better left to humans. Not even hamming distance will get this right!
But why did you make this?
There are literally dozens of other projects out there that more or less do the same thing, why did you bother to make a new one? The answer is quite simple: Because none of them fulfill all the needs of the quasar project, and although lots of really smart people built them, we want to be able to maintain this library and grow it as quasar grows. Especially since many such image-library projects quickly grow outdated and maintainers drift away into a sea of micro-modules, we felt that it is a good property to keep "in-house".
Furthermore, we wanted to pay special attention to your security and it was obvious that other libraries aren't doing enough to protect developers.
Minify
There are four different minification algorithms available (because it is possible that one or the other just won't build on your system or worse becomes compromised via some kind of exploit). Using totally non-scientific tests, these are the results of this command that ran one time on a 3,2 GHz Quad-Core Intel Xeon with 16 GB of ram:
$ time node ./bin/cli.js -p=spa -s=test/example-1240x1240.png -t=test/output -m="$minify" && du -sh test/output/spa
It takes a source image, scales it down according to the settings and then minifies it according to the library listed and the default settings tuned to give the best results.
Library | Time | Quality | Size |
---|---|---|---|
no minify | 00.7s | original | 664K |
pngquant-good | 01.4s | lossy | 144K |
optipng | 13.9s | lossless | 404K |
pngcrush | 28.1s | lossless | 404K |
zopfli.more | 81.3s | lossless | 336K |
This is why it is recommended to use pngquant during development (to have a proxy image), but to use optipng (or zopfli) when building for production.
.icns & .ico
These are notoriously difficult to acquire and make. For icns you usually need a mac and ico is really just a sequence of images with a special header - but there are very few tools that will let you do both cross-platform - and this is one case where we needed several tools.
This module uses the amazing png2icons
module, which actually does all of the decoding and encoding on a byte-level with javascript. This of course takes a bit of time, but it also works cross-platform, so please think about going over there and giving those devs a kitchensink
and the files are huge. By feeding it from the sharp
buffer it has been sped up a little bit (and the final icns file is actually about 20% smaller!)
To make the favicon.ico, it uses to-ico
and concats a 16x16 and a 32x32 png.
SVG
The safari-pinned-tab.svg
mask is created by adding contrast (via threshold) to the original and then applying even more threshold to the SVG tracing.
If you are indeed of a discerning nature (and have used gradients in your icon design), there is another option available to you:svg-duochrome. It too will be created in the spa folder within the spa task, but it will be up to you to rename it to safari-pinned-tab.svg
. Be forewarned, that gradients used in a duochrome svg have a very particular character - and you will have a very large file.
To make these SVG's as small as possible, they are compressed with SVGO. There are no scripts or remote resources included in these SVG assets.
CLI Usage
The Icon Factory can be used as a command line tool for batch invocation, and you can simply add it by installing it "globally" with your node package manager:
$ yarn global add @quasar/quasar-app-extension-icon-factory
- or -
$ npm install --global @quasar/quasar-app-extension-icon-factory
To find out about the settings, just use
$ iconfactory --help
Quasar Icon Factory: v.1.0.0
License: MIT
Icon Factory is a node utility to create a batch of icons for your app.
Designed to work seamlessly with the Quasar Framework, but probably
useful for other build pipelines.
Flags:
-p, --preset Choose a preset output or make your own
[minify|splash|svg|favicon]
[spa|pwa|cordova|electron|kitchensink|custom]
-s, --source Your source image as a large square png
-t, --target The destination directory for the files created
-o, --options path to file that overrides defaults (if custom)
-m, --minify Minify strategy to use.
[pngcrush|optipng|pngquant|zopfli]
-d, --mode Minify mode if minify preset [folder|singlefile]
-v, --version display version information
-h, --help display this information
Usage:
$ iconfactory -p=kitchensink -s=icon-1280x1280.png -t=./outputFolder -m=pngquant
$ iconfactory -p=minify -s=icon-1240x1240.png -t=./output -m=pngquant -d=singlefile
Consuming as a library
You can use this module as a library to provide production and minification assets if you so desire. It will only work in the server context. Import and use at your discretion, but please note: this is not the primary purpose of this library and only security / core functionality issues will be addressed. Please do not file feature requests for this unless you provide a PR.
Performance
Solving compression problems takes time, and the more complex the approach the more linear time is needed. Some compression algorithms are fast, some are great. None are both. This package tries to write from the buffer only when a file is created and (mostly) avoids creating intermediary files.
Testing
git clone
, yarn install
, yarn test
Security
This library only permits the usage of .png
files as the source, and there is a magic-number
check to ensure that the file being processed is indeed a valid and proper image/png
. Furthermore, there are neither imagemagick nor graphicsmagick dependencies and this project should build and run on all platforms supported by @quasar/cli
.
If you discover a security issue, please contact us via email: security@quasar-framework.org
Contributing
You are welcome to join this project. Please file issues and make PRs! Let us know how it goes and join us at our discord server to talk shop.
Resources for more information about App Icons
- Favicon Cheat Sheet
- PWA Icons
- PWA for Chrome
- PWA on iOS
- Lighthouse Web App Audit
- MacOS App Icon
- iOS App Icons
- iOS Ad Hoc Mode
- Cordova Icons
- Electron Icons
- Windows Icon Assets
- Windows APPX Icons
- Windows MRT Cordova Image Spec
Other nice resources
- favicon.ico vs favicon.png
- Material Icons
- Favicon Checker
- Wikipedia PNG Optimizing
- PNG Minification Comparison
- .ico file-header Information
Thanks to
- image-trace-loader
- the image-min team
- @TobyMosque, @maxMatteo, @Robin, @Rob, @trendchaser4u, @bloo_df
Licenses
- Code: MIT
- © 2018: Present Daniel Thompson-Yvetot & Razvan Stoenescu
- Original Image for iconfactory: Public Domain
- Modifications: Daniel Thompson-Yvetot. CC-BY