アイコンバッジ
Electronでアプリを作っていると、新着のお知らせなどがあった場合にアプリアイコンにバッジを付けたくなります。
Electronにはそれを実現するAPIが用意されているので、基本的には簡単なのですがWindowsでは非常に面倒だったのでまとめておきます。
ちなみにバッジには任意の数値を出すことを想定しています。
Linuxはデスクトップ環境を用意したことがないので対象外です。
Mac
とても簡単です。
mainプロセス内で app.setBadgeCount(X)
を呼ぶだけです。
const electron = require('electron')
const app = electron.app
const BrowserWindow = electron.BrowserWindow
const path = require('path')
const url = require('url')
let mainWindow
function createWindow () {
mainWindow = new BrowserWindow({width: 800, height: 600})
mainWindow.loadURL(url.format({
pathname: path.join(__dirname, 'index.html'),
protocol: 'file:',
slashes: true
}))
// set badge count
if (process.platform === 'darwin') {
app.setBadgeCount(10)
}
}
ただ、このメソッドの引数には数値しか受け取れません。
たとえば、99件までは数値を表示し99件以降は99+として表示したい場合にはこのメソッドは使うことは出来ません。
その為に任意の文字列を設定出来るメソッドが用意されています。
appのなかのdockプロパティの setBadge(string)
がそのメソッドです。
app.setBadgeではないので気をつけてください。
function createWindow () {
mainWindow = new BrowserWindow({width: 800, height: 600})
mainWindow.loadURL(url.format({
pathname: path.join(__dirname, 'index.html'),
protocol: 'file:',
slashes: true
}))
// set badge count
if (process.platform === 'darwin') {
app.dock.setBadge("99+");
}
}
表示は下記のようになります。
app.setBadgeCount
と app.dock.setBadge
、使い分けが面倒ではあるので、基本的に app.dock.setbadge
で良い気もします。
数値を扱いたいときは app.dock.setBadge(""+99)
というように。
Windows
ここからが鬼門です。
Windowsでは上記メソッドは使えません。
タスクバーに数値や文字を表示させる、というよりもタスクバーの上に画像を表示させる、といったことをします。
そのためのメソッドがsetOverlayIcon(overlay, description)
です。
引数は2つあり、一つ目に表示させる(オーバーレイ)画像、2つめは適当な説明です。(descriptionは画面読み上げ用のアクセシビリティがどうのっぽいです)
上記メソッドは、Rendererプロセスのみで使用できません(このあたりでまたコードが煩雑になって辛いです)。
現在表示しているBrowserWindowを取得し、画像パスを第一引数に指定することで表示することが出来ます。
画像は下記サイトより生成したファイルを利用しました。
http://www.flaticon.com/free-icon/exclamation-mark-sign_12136
const electron = require('electron');
const app = electron.app;
const remote = electron.remote;
window.onload = function(){
if (process.platform == 'win32') {
var mainWindow = remote.getCurrentWindow()
mainWindow.setOverlayIcon(__dirname+'/icon.png', "icon");
}
}
Windows上ではこのように表示することが出来ます。
エクスクラメーション部分が透過なのでちょっと不格好ですが、表示することが出来ました。
ちなみに非表示にするにはnullを引数にすると非表示になります
mainWindow.setOverlayIcon(null, "empty");
任意の数値を表示したい
ここで問題なのが、事前に画像を用意する必要があるということです。
たとえば先程のMac版でやったような、任意の数値を表示するといったことをしたい場合、1〜99まで画像を用意してさらに99+といった画像を用意する必要があります。
99個くらいなら..とは思いますが、メールクライアントなら999件くらいは表示して欲しくなりますし、現実的ではなくなります。アプリケーションにバンドルする画像が増えるとアプリケーションサイズが増えてしまいますし、管理も大変です。
そこで、必要な画像は必要になったときに生成することにします。
canvasのAPIを用い、都度画像を生成しましょう。
まずはコードがこんな感じに。
var nativeImage = electron.nativeImage;
//1
var canvas = document.createElement("canvas");
//2
canvas.height = 140;
canvas.width = 140;
var ctx = canvas.getContext("2d");
ctx.fillStyle = "red";
ctx.beginPath();
ctx.ellipse(70, 70, 70, 70, 0, 0, 2 * Math.PI);
ctx.fill();
ctx.textAlign = "center";
ctx.fillStyle = "white";
//3
var text = "1";
if (text.length > 2) {
ctx.font = "75px sans-serif";
ctx.fillText("" + text, 70, 98);
} else if (text.length > 1) {
ctx.font = "100px sans-serif";
ctx.fillText("" + text, 70, 105);
} else {
ctx.font = "125px sans-serif";
ctx.fillText("" + text, 70, 112);
}
//4
var badgeDataURL = canvas.toDataURL();
//5
var mainWindow = remote.getCurrentWindow()
mainWindow.setOverlayIcon(badgeDataURL, "icon");
- canvas要素を生成し
- 赤い丸を書きます。
- 桁数によってフォントサイズを調整しつつ文字を入れます。sans-selifじゃなくてもいいです。
- キャンバスの中身をtoDataURLメソッドでURL形式に出力
- setOverlayIconで表示
が、実行してみると下記のようなエラーになってしまいます。
C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron…:217 Uncaught Error: Could not call remote function 'setOverlayIcon'. Check that the function signature is correct. Underlying error: Error processing argument at index 0, conversion failure from …ZBeGjKmGGu7lzA36rRGE1JdeEXoWDGlablemh+iI2s6vr8B3KGUA4FAg25AAAAAElFTkSuQmCC
Error: Could not call remote function 'setOverlayIcon'. Check that the function signature is correct. Underlying error: Error processing argument at index 0, conversion failure from …ZBeGjKmGGu7lzA36rRGE1JdeEXoWDGlablemh+iI2s6vr8B3KGUA4FAg25AAAAAElFTkSuQmCC
at callFunction (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\rpc-server.js:235:11)
at EventEmitter.<anonymous> (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\rpc-server.js:342:5)
at emitMany (events.js:127:13)
at EventEmitter.emit (events.js:201:7)
at WebContents.<anonymous> (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\api\web-contents.js:231:13)
at emitTwo (events.js:106:13)
at WebContents.emit (events.js:191:7)
NativeImage形式である必要があるので、dataURLでは駄目なようです。
なので、下記の様にコードを変えます。
NativeImageのクラスメソッドで createFromDataURL
といったものがあるので利用してNativeImageを出力します。
//4
var badgeDataURL = canvas.toDataURL();
var img = nativeImage.createFromDataURL(badgeDataURL);
//5
var mainWindow = remote.getCurrentWindow()
mainWindow.setOverlayIcon(img, "icon")
ですが、再度実行しても下記のようなエラーを出力します。
C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron…:217 Uncaught Error: Could not call remote function 'setOverlayIcon'. Check that the function signature is correct. Underlying error: Error processing argument at index 0, conversion failure from #<Object>
Error: Could not call remote function 'setOverlayIcon'. Check that the function signature is correct. Underlying error: Error processing argument at index 0, conversion failure from #<Object>
at callFunction (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\rpc-server.js:235:11)
at EventEmitter.<anonymous> (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\rpc-server.js:342:5)
at emitMany (events.js:127:13)
at EventEmitter.emit (events.js:201:7)
at WebContents.<anonymous> (C:\Users\Administrator\Documents\sample\node_modules\electron\dist\resources\electron.asar\browser\api\web-contents.js:231:13)
at emitTwo (events.js:106:13)
at WebContents.emit (events.js:191:7)
下記リンクでも同様のエラーが報告されていますが、特に進展はありません。
Electronのバグなのか、そもそもserializeしてない画像は表示できないのが仕様なのかはちょっとわかりません。
https://discuss.atom.io/t/nativeimage-not-working-in-setoverlayicon-in-electron/37648
ローカルに保存されている画像ファイルが表示出来ることは前述でわかっているので、生成したファイルを保存してそれを参照すれば良さそうです。
canvasをbufferに書き出すには electron-canvas-to-buffer
モジュールを利用します。
https://www.npmjs.com/package/electron-canvas-to-buffer
canvasBufferにcanvasを食わせて画像形式を指定します。
生成したbufferをfs.writeFileSyncによってファイルに書き出します。
Syncにしてるのには特に理由がないのでasyncにしてcallbackでsetOverlayIconしても良いと思います。
const fs = require('fs');
const canvasBuffer = require('electron-canvas-to-buffer')
var buffer = canvasBuffer(canvas, 'image/png')
fs.writeFileSync(__dirname+'/test.png', buffer)
上記コードを実行すると
フォルダ直下にファイルが出力されます。
あとはこの画像をsetOverlayIconに指定すれば任意の値を表示することが可能になりました。
注意点
上記コードでは __dirname+'/test.png'
というように画像ファイルを出力し参照しています。
ですが、electron-packagerなどでパッケージングしてしまうと、__dirnameも配下などはasarとして固められてしまうのでファイルが書き出せません(このあたりは詳しくないので間違っているかもしれません)
npm start
などで実行してるときは気づきませんが、パッケージングしてから気づきました。
これを避けるために、適当にアクセス出来るパスにファイルを保存しましょう。
Rendererプロセス内では、remote.app.getPath('temp')
から、tempフォルダを得られるのでここに保存することにしましょう。
tempはちょっと...という方は下記リンクから要件にあうフォルダを選んでください。r/wの権限には気をつけてください。
http://qiita.com/progre/items/2718f4ad20eecf27d599
得たパスを、保存フォルダとして使ってください。
var buffer = canvasBuffer(canvas, 'image/png')
var iconSavePath = remote.app.getPath('temp')+ '/gen_icon.png'
fs.writeFileSync(iconSavePath, buffer);
var mainWindow = BrowserWindow.getCurrentWindow();
mainWindow.setOverlayIcon(iconSavePath ,"count");
ファイル生成処理は多少はパフォーマンスに難ありだとは思うので、表示したい件数に変更があったときのみ処理が実行するなどの工夫はしたほうがいいでしょう。
実行テスト環境
macOS Sierra 10.12.2
Windows Server2016(AWS ec2 instance)
Node.js 6.5.0
Chromium 53.0.2785.143
Electron 1.4.14
サンプルプロジェクト
今回使用したコードは下記プロジェクトにまとめてあります。
electron-quick-startをベースにしています。
https://github.com/dmnlkGarbage/electron-dock-icon-sample
参考リンク
https://github.com/electron/electron/tree/master/docs
http://mylifeforthecode.com/setting-a-custom-taskbar-icon-for-an-electron-window/
https://gist.github.com/johnthedebs/c22a04fcd08e598e69b8
https://discuss.atom.io/t/nativeimage-not-working-in-setoverlayicon-in-electron/37648
http://qiita.com/progre/items/2718f4ad20eecf27d599