Skip to content

Commit d6e6611

Browse files
committed
Initial Release
0 parents  commit d6e6611

15 files changed

+733
-0
lines changed

.eslintrc.json

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
{
2+
"extends": ["eslint:recommended"],
3+
"env": {
4+
"node": true,
5+
"browser": true,
6+
"es6": true
7+
},
8+
"globals": {
9+
"globalThis": "writable"
10+
},
11+
"parserOptions": {
12+
"ecmaVersion": 2023,
13+
"sourceType": "module"
14+
},
15+
"rules": {
16+
"no-unused-vars": "error",
17+
"arrow-spacing": ["warn", { "before": true, "after": true }],
18+
"brace-style": ["error", "1tbs", { "allowSingleLine": true }],
19+
"comma-dangle": ["error", "always-multiline"],
20+
"comma-spacing": "error",
21+
"comma-style": "error",
22+
"curly": ["error", "multi-line", "consistent"],
23+
"dot-location": ["error", "property"],
24+
"handle-callback-err": "off",
25+
"indent": ["error", "tab"],
26+
"keyword-spacing": "error",
27+
"max-nested-callbacks": ["error", { "max": 5 }],
28+
"max-statements-per-line": ["error", { "max": 3 }],
29+
"no-console": "off",
30+
"no-empty-function": "error",
31+
"no-floating-decimal": "error",
32+
"no-inline-comments": "error",
33+
"no-lonely-if": "error",
34+
"no-multi-spaces": "error",
35+
"no-multiple-empty-lines": [
36+
"error",
37+
{ "max": 2, "maxEOF": 1, "maxBOF": 0 }
38+
],
39+
"no-shadow": ["error", { "allow": ["err", "resolve", "reject"] }],
40+
"no-trailing-spaces": ["error"],
41+
"no-var": "error",
42+
"object-curly-spacing": ["error", "always"],
43+
"prefer-const": "error",
44+
"quotes": ["error", "single"],
45+
"semi": ["error", "always"],
46+
"no-control-regex": "off",
47+
"space-before-blocks": "error",
48+
"space-before-function-paren": [
49+
"error",
50+
{
51+
"anonymous": "never",
52+
"named": "never",
53+
"asyncArrow": "always"
54+
}
55+
],
56+
"space-in-parens": "error",
57+
"space-infix-ops": "error",
58+
"space-unary-ops": "error",
59+
"spaced-comment": "error",
60+
"yoda": "error"
61+
}
62+
}

.gitignore

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# Local history
2+
.history/
3+
4+
# Dependency directories
5+
node_modules/

.npmignore

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# Local history
2+
.history/
3+
4+
# Github assets
5+
assets/
6+
7+
# ESlint config
8+
.eslintrc.json

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2018 Viktor Shmigol
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.

README.md

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
[![version](https://img.shields.io/npm/v/electron-taskbar-badge?color=blueviolet&style=for-the-badge "Version")](https://github.com/KK-Designs/electron-taskbar-badge/releases/tag/v1.4.1)
2+
3+
[![weekly_downloads](https://img.shields.io/npm/dw/electron-taskbar-badge?color=blue&style=for-the-badge "Weekly Downloads")](https://www.npmjs.com/package/electron-taskbar-badge#:~:text=Weekly%20Downloads)
4+
5+
![downloads](https://badgen.net/npm/dt/electron-taskbar-badge "Downloads")
6+
7+
[![issues](https://img.shields.io/github/issues/KK-Designs/KK-Designs/electron-taskbar-badge?style=for-the-badge "Issues")](https://github.com/KK-Designs/electron-taskbar-badge/issues)
8+
9+
[![license](https://img.shields.io/github/license/KK-Designs/electron-taskbar-badge?color=important&style=for-the-badge "License")](https://github.com/KK-Designs/electron-taskbar-badge/blob/master/LICENSE)
10+
11+
[![electron-taskbar-badge](https://nodei.co/npm/electron-taskbar-badge.png "electron-taskbar-badge on NPM")](https://www.npmjs.com/package/electron-taskbar-badge)
12+
---
13+
14+
# Electron Taskbar Badge
15+
An easy way for electron apps to add app badges to the taskbar to indicate notifications and other countable things, with maximum compatibility and customizability.
16+
17+
---
18+
19+
# Changelog (`v1.0.0`)
20+
21+
• This release is the first release of Electron Taskbar Badge
22+
23+
---
24+
25+
# Installation
26+
27+
```sh-session
28+
npm i electron-taskbar-badge
29+
```
30+
31+
---
32+
33+
# Basic Usage
34+
35+
> **This library is ONLY compatible with node version 14 and above**
36+
37+
First you must import the library using the following code:
38+
```javascript
39+
const Badge = require('electron-taskbar-badge');
40+
// or `import * as Badge from 'electron-taskbar-badge';` for ESM users
41+
```
42+
\
43+
For basic usage, all you have to do is call the function with the options:
44+
```javascript
45+
const Badge = require('electron-taskbar-badge');
46+
// or `import * as Badge from 'electron-taskbar-badge';` for ESM users
47+
48+
// NOTE: Although the font size 62px seems large, it is not. It is relative the the radius. Lowering both of these values can decrease quality significantly. Increasing them can reduce performance. Leave the font size as is for basic usage
49+
const badgeOptions = {
50+
fontColor: '#FFFFFF', // The font color
51+
font: '62px Microsoft Yahei', // The font and its size. You shouldn't have to change this at all
52+
color: '#FF0000', // The background color
53+
radius: 48, // The radius for the badge circle. You shouldn't have to change this at all
54+
updateBadgeEvent: 'notificationCount', // The IPC event name to listen on
55+
badgeDescription: 'Unread Notifications', // The badge description
56+
invokeType: 'handle', // The IPC event type
57+
max: 9, // The maximum integer allowed for the badge. Anything above this will have "+" added to the end of it.
58+
fit: false, // Useful for multi-digit numbers. For single digit keep this set to false
59+
additionalFunc: (count) => {
60+
// An additional function to run whenever the IPC event fires. It has a count parameter which is the number that the badge was set to.
61+
console.log(`Received ${count} new notifications!`);
62+
},
63+
};
64+
65+
// "win" would be your Electron BrowserWindow object
66+
new Badge(win, badgeOptions);
67+
```
68+
**Thats it! Now you have it running!**
69+
70+
## More examples
71+
### Native look
72+
If you want your badge to look native to the operating system, which means that it follows the default OS's font and color, you can use the `useSystemAccentTheme` option. Here's an example:
73+
74+
```javascript
75+
// DO NOT change the font or the radius
76+
// fontColor and color will be overridden. The background color would be the system accent color, and the font color would be automatically chosen between black or white, whichever looks best.
77+
const badgeOptions = {
78+
fontColor: '#000000',
79+
font: '62px Microsoft Yahei',
80+
color: '#000000',
81+
radius: 48,
82+
updateBadgeEvent: 'notificationCount',
83+
badgeDescription: 'Unread Notifications',
84+
invokeType: 'handle',
85+
max: 9,
86+
fit: false,
87+
useSystemAccentTheme: true,
88+
additionalFunc: (count) => {
89+
console.log(`Received ${count} new notifications!`);
90+
},
91+
};
92+
93+
new Badge(win, badgeOptions);
94+
```
95+
96+
**Examples**
97+
98+
![Taskbar Badge Purple](assets/taskbarBadgePurple.gif?raw=true) \
99+
![Taskbar Badge Blue](assets/taskbarBadgeBlue.gif?raw=true) \
100+
![Taskbar Badge Green](assets/taskbarBadgeGreen.gif?raw=true)
101+
102+
### Auto font color
103+
If you want your badge's font color to be automatically chosen, simply set `fontColor` to `auto`. This will chose the font color between black or white, whichever looks best. Here's an example:
104+
105+
```javascript
106+
const badgeOptions = {
107+
fontColor: 'auto',
108+
font: '62px Microsoft Yahei',
109+
color: '#00FF00',
110+
radius: 48,
111+
updateBadgeEvent: 'notificationCount',
112+
badgeDescription: 'Unread Notifications',
113+
invokeType: 'handle',
114+
max: 9,
115+
fit: false,
116+
additionalFunc: (count) => {
117+
console.log(`Received ${count} new notifications!`);
118+
},
119+
};
120+
121+
new Badge(win, badgeOptions);
122+
```
123+
124+
# Options
125+
### Options info for `Badge(options)`
126+
127+
| Parameters | Type | Description | Default |
128+
|---------------|---------|----------------------------------------|---------|
129+
| `fontColor` | string (required) | The font color. Pretty self explanatory. | auto |
130+
| `font` | string | The font for the badge icon. The format is [size]px [Font family name] **ALWAYS SET THE FONT SIZE TO 62px FOR BEST QUALITY** | 62px Microsoft Yahei |
131+
| `color` | string (required) | The background color for the badge icon | `null` |
132+
| `radius` | number | The radius for the badge icon **ALWAYS SET TO 48 FOR BEST QUALITY** | 48 |
133+
| `updateBadgeEvent` | string (required) | The IPC event name to listen on | `null` |
134+
| `badgeDescription` | string | A description that will be provided to Accessibility screen readers | `this.updateBadgeEvent` |
135+
| `invokeType` | string | The IPC event type. Can be `send` or `handle`. | send |
136+
| `max` | number | The maximum integer allowed for the badge. Anything above this will have "+" added to the end of it. | 99 |
137+
| `fit` | boolean | The maximum integer allowed for the badge. Anything above this will have "+" added to the end of it. | `false` |
138+
| `useSystemAccentTheme` | boolean | Whether to use the system accent color for the background color. fontColor and color will be overridden. It would be automatically chosen between black or white, whichever looks best. | `false` |
139+
| `additionalFunc` | function(count) | An additional function to run whenever the IPC event fires. It has a count parameter which is the number that the badge was set to. | `null` |
140+
141+
#
142+
[![](assets/backToTop.png?raw=true "Back to top")](#readme)

assets/backToTop.png

1.2 KB
Loading

assets/taskbarBadgeBlue.gif

164 KB
Loading

assets/taskbarBadgeGreen.gif

156 KB
Loading

assets/taskbarBadgePurple.gif

201 KB
Loading

lib/cjs/badge_generator.js

+124
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,124 @@
1+
'use strict';
2+
module.exports = class BadgeGenerator {
3+
constructor(win, opts = {}, systemAccentTheme) {
4+
const defaultStyle = {
5+
decimals: 0,
6+
systemAccentTheme,
7+
};
8+
this.win = win;
9+
this.style = Object.assign(defaultStyle, opts);
10+
}
11+
12+
generate(number) {
13+
const opts = JSON.stringify(this.style);
14+
return this.win.webContents.executeJavaScript(`window.drawBadge = function ${this.drawBadge}; window.drawBadge(${number}, ${opts});`);
15+
}
16+
17+
drawBadge(number, style) {
18+
if (typeof style?.color !== 'string') {
19+
throw new TypeError(`Invalid color specified\nExpected: string\nGot: ${typeof style?.color}`);
20+
}
21+
if (!/^#[0-9A-F]{6}$/i.test(style?.fontColor)) {
22+
style.fontColor = 'auto';
23+
}
24+
style.font = style?.font ?? '62px Microsoft Yahei';
25+
style.fontColor = style?.fontColor ?? 'auto';
26+
style.radius = style?.radius ?? 48;
27+
style.fit = style?.fit ?? false;
28+
style.useSystemAccentTheme = style?.useSystemAccentTheme ?? false;
29+
style.decimals = style?.decimals === undefined || isNaN(style.decimals) ? 0 : style.decimals;
30+
style.max = style?.max ?? 99;
31+
32+
const radius = style.radius;
33+
const img = document.createElement('canvas');
34+
img.width = Math.ceil(radius * 2);
35+
img.height = Math.ceil(radius * 2);
36+
img.ctx = img.getContext('2d');
37+
img.radius = radius;
38+
img.number = number;
39+
40+
const accentColor = style.systemAccentTheme;
41+
const accentColorLight = lightenColor(`#${accentColor.slice(0, -2)}`, 35);
42+
43+
if (style?.useSystemAccentTheme) {
44+
style.color = accentColorLight;
45+
style.fontColor = getTextColor(accentColorLight);
46+
}
47+
48+
if (style.fontColor === 'auto') {
49+
style.fontColor = getTextColor(accentColorLight);
50+
}
51+
52+
img.displayStyle = style;
53+
54+
img.draw = function() {
55+
let fontScale, fontWidth, fontSize, integer;
56+
this.width = Math.ceil(this.radius * 2);
57+
this.height = Math.ceil(this.radius * 2);
58+
this.ctx.clearRect(0, 0, this.width, this.height);
59+
this.ctx.fillStyle = this.displayStyle.color;
60+
this.ctx.beginPath();
61+
this.ctx.arc(radius, radius, radius, 0, Math.PI * 2);
62+
this.ctx.fill();
63+
this.ctx.font = this.displayStyle.font;
64+
this.ctx.textAlign = 'center';
65+
this.ctx.textBaseline = 'middle';
66+
this.ctx.fillStyle = this.displayStyle.fontColor;
67+
integer = this.number.toFixed(this.displayStyle.decimals);
68+
69+
fontSize = Number(/[0-9.]+/.exec(this.ctx.font)[0]);
70+
71+
if (integer > style.max) {
72+
if (this.displayStyle.fit) {
73+
fontSize = fontSize * 1.175;
74+
integer = `${style.max}+`;
75+
} else {
76+
this.ctx.font = this.ctx.font.replace(/\d+/, Number(this.ctx.font.match(/\d+/)[0]) / 1.175);
77+
integer = `${style.max}+`;
78+
}
79+
}
80+
81+
if (!this.displayStyle.fit || isNaN(fontSize)) {
82+
this.ctx.fillText(integer, radius, radius);
83+
} else {
84+
fontWidth = this.ctx.measureText(integer).width;
85+
fontScale = (Math.cos(Math.atan(fontSize / fontWidth)) * this.radius * 2 / fontWidth);
86+
this.ctx.setTransform(fontScale, 0, 0, fontScale, this.radius, this.radius);
87+
this.ctx.fillText(integer, 0, 0);
88+
this.ctx.setTransform(1, 0, 0, 1, 0, 0);
89+
}
90+
91+
return this;
92+
};
93+
94+
function getTextColor(bgColor) {
95+
// Convert hex color to RGB values
96+
const red = parseInt(bgColor.substr(1, 2), 16);
97+
const green = parseInt(bgColor.substr(3, 2), 16);
98+
const blue = parseInt(bgColor.substr(5, 2), 16);
99+
100+
// Calculate perceived brightness using the luminosity algorithm
101+
const brightness = (0.2126 * red) + (0.7152 * green) + (0.0722 * blue);
102+
103+
// Return white or black based on the perceived brightness
104+
return (brightness > 128) ? '#000000' : '#FFFFFF';
105+
}
106+
107+
function lightenColor(hex, percent) {
108+
let red = parseInt(hex.substring(1, 3), 16);
109+
let green = parseInt(hex.substring(3, 5), 16);
110+
let blue = parseInt(hex.substring(5, 7), 16);
111+
112+
const amt = Math.round(2.55 * percent);
113+
red = Math.min(255, Math.round(red + amt));
114+
green = Math.min(255, Math.round(green + amt));
115+
blue = Math.min(255, Math.round(blue + amt));
116+
117+
const newHex = ((red << 16) | (green << 8) | blue).toString(16);
118+
return '#' + ('000000' + newHex).slice(-6);
119+
}
120+
121+
img.draw();
122+
return img.toDataURL();
123+
}
124+
};

0 commit comments

Comments
 (0)