💨呼吸時間の記録
Apple Watchには「呼吸」アプリがある。深呼吸によるリラックスを支援をするアプリである。(参考: 呼吸 App を使う)
呼吸した時間をPixelaに記録し始めた。
🎯一覧性と継続のモチベーション
呼吸した時間はヘルスケアアプリに記録される。 しかし、継続日数が読み取りづらく、毎日続けるモチベーションになりづらい。
✨Pixela
Pixelaは任意のデータをGitHubのコントリビューショングラフ(通称: 草)にするサービスである。
⏫ショートカットを用いた呼吸時間の記録
ショートカット(旧: WorkFlow)で呼吸時間をPixelaに記録する。
ショートカットを不特定多数に公開する方法はないので、画像を貼っておく。
おもに次の処理を行なっている。
- ヘルスケアから呼吸時間を取得する
- 合計時間を計算する
- Pixelaに記録する
- 1〜3を過去n日分に繰替えす
最初は作成したのち更新する、という流れにしたが、更新のみでできるように pixe.la の仕様が変更された。(最高!!)
ありがとうございます!Pixelの、ですよね!
— a-know (@a_know) November 23, 2018
PUT の挙動を「なければ作成」にしようと思っております...!🙏
対応してみました!😀https://t.co/5Yb7c1hEds
— a-know (@a_know) November 24, 2018
💓所感
iPhoneアプリを書かずにiPhoneの情報を扱えるショートカットは便利。ただ版管理機能や共有機能が弱いのは厳しい。
Pixelaは手軽でいい。見慣れた見た目になる、簡単なAPIで使えるので使い出がありそう。ただ、データを確認するときに、いちいちcurlを叩くのは大変だったのでPawを使った。
🔐二要素認証
Twitterなどの二要素認証では、二要素認証用のアプリで生成した認証コードを使う。
認証コードをどのように生成しているのか、Twitterとは無関係のアプリで生成した認証コードが利用できるのか、クラウドで同期できるのかが疑問だったので実装した。
🔍調査
なにから調べていいか分からなかったので、普段使っているAuthyのサイトを見ていたら、二要素認証の種類について説明しているページがあった。
これによると「TOTP(Time-based One-Time Password)」という方式らしい。
あの6桁の数字はワンタイムパスワードなのかと気づいた。
📕 TOTPアルゴリズム
TOTPというキーワードをもとに検索すると、RFCまで辿りつく。
TOTPは以下の式で計算される。
時刻T(Unix秒)におけるは以下の式であたえられる。 はワンタイムパスワードが有効な秒数であり、通常は30である。
Swiftで書くとこのようになる。
public final class TOTPGenerator { private let hotp: HOTPGenerator public init(secret : [UInt8]) { self.hotp = HOTPGenerator(secret: secret) } public func generate(at date : Date, format: OTPFormat) -> String? { let count = UInt64(date.timeIntervalSince1970) / 30 return hotp.generate(at: count, format: format) } }
📒HOTPアルゴリズム
TOTPはHOTP(HMAC-based One-time Password)のパラメータを時刻にしたものなので、メインのアルゴリズムはHOTPで定義される。
Truncateはハッシュ値を指定の桁までに切りつめる操作だが、数式で書くとつらいのでSwiftで書くとこのようになる。 ほぼこのままの式がRFCにのっている。
func truncate(hash : [UInt8]) -> Int { let offset = Int(hash[19] & 0x0f) return Int(hash[offset] & 0x7f) << 24 | Int(hash[offset + 1]) << 16 | Int(hash[offset + 2]) << 8 | Int(hash[offset + 3]) }
これを使いうとHOTPは以下のコードで生成できる。
import class CryptoSwift.HMAC class HOTPGenerator { private let hmac: HMAC init(secret : [UInt8]) { self.hmac = HMAC(key: secret, variant: .sha1) } func generate(at count : UInt64, format: OTPFormat) -> String? { guard let hash = try? hmac.authenticate(Array(uint64: count)) else { return nil } // 先頭6文字のみを使う return String(String(truncate(code)).suffix(6)) } }
🤝秘密鍵の交換
TOTP/HOTPでは秘密鍵を共有する必要がある。
秘密鍵はBase32でエンコードされてやりとりされる。 二要素認証の設定画面に表示される「7nx ofic ...」がそれにあたる。QRコードにも同様の内容が含まれている。
🛠サンプルアプリ
アルゴリズムだけ調べても正しいか不安なので、Swiftでコマンドラインツールを作った。
Mastodonの二要素認証に表示されるSecret keyを入力して、二要素認証に使えることを確認した。
Swift Package Managerを使ったが、プレインテキストでプロジェクトの構成を管理できてよかった。
xcodeprojファイル管理するの大変だからフォルダ構成とかから生成できないかなーと思ってたけど、swift package generate-xcodeprojでできたわ
— mzp (@mzp) 2018年11月3日
👀参考にしたサイト
- Swift Package Manager V4 + Utilityで作る!コマンドラインツール – Eureka Engineering – Medium
- Swift Package Manager (SwiftPM) で作るコマンドラインツール - Qiita
- ワンタイムパスワードジェネレータを作った - ぶていのログでぶログ
💕感想
普段、使ってるものの仕組みをちゃんと調べるのはたのしい。
Authyとかの二段階認証、どういう仕組みになってんの? なんで独立したアプリがTwitterのコードを生成できんの? というのが不思議だったので、RFCとか読みながら作った。 pic.twitter.com/eDGWQd3bWP
— mzp (@mzp) 2018年11月3日
要は事前に共有した秘密鍵と時刻から、認証用のコードを生成してるんだ
— mzp (@mzp) 2018年11月3日
🖥43インチモニタ
43インチの4Kモニタを購入した。 体験がいい。
🛒購入したもの
- I-O DATA 4K モニター ディスプレイ 43型 EX-LD4K431DB
- エルゴトロン LX デスクマウント モニターアーム
- あとでHP(ヒューレット・パッカード) シングルモニターアーム BT861AA](https://www.amazon.co.jp/dp/B006PYJD44/)のほうが安いと教えてもらった。
- サンコー VESA変換プレート
- QTop Mini DisplayPort to HDMI変換ケーブル
- ベッセル(VESSEL) ドライバーセット
🍖焼き肉寿司
@izmさんに誘われて焼き肉寿司を食べにいった。
うまいうまいと食べてたら、テンションがあがって4Kモニタを注文してしまった。 もともと欲しいと思ってたので、「ボク使っていますが最高ですよ」と言われて背中を押されてしまった。
私は焼肉寿司食べてる間にテンションがあがって40インチモニタを購入した気がします
— mzp (@mzp) October 15, 2018
@izmさんはiPhoneXSを買っていた。
焼肉寿司食べてる間にテンションがあがってiPhone XSを購入した気がする
— izm (@izm) October 15, 2018
📦設置
数日後に届いた。 「二人以上で作業しろ」と書いてあった。そんなこと言われても困る。
詰んだ pic.twitter.com/1eZ9m9i0lX
— mzp (@mzp) October 17, 2018
そこまで重くなかったので、見なかったことにして一人で取り出した。
机に置けないかなと思って挑戦したが、まったく置ける気配がなかった。さらにドライバーがなかったので台の取り付けもできなかった。
とりあえず壁に立て掛けて、MacBookProをつないだ。Netflixで言の葉の庭を再生したら最高になった。
デスクにのらなかったので、とりあえず床に置いた pic.twitter.com/KJW0qF6Os0
— mzp (@mzp) October 17, 2018
🔧モニタアーム
最高だけど床に置いたままにはできないのでモニタアームを注文した。ドライバーのセットも買った。
次の商品を購入しました:エルゴトロン 『エルゴトロン LX デスクマウント モニターアーム 45-241-026』 via @amazonJP https://t.co/5cgGFwOqFk
— mzp (@mzp) October 17, 2018
到着したあと説明書を見ながらアダプタを取り付けたり、モニタアームをデスクに装着したりした。2回くらい取り付け方向をミスったがなんとかなった。
怖かったので、ベッドの位置をずらして、万が一落下しても頭にあたらないようにした。
🔌HDMI変換アダプタ
使ってたHDMI変換アダプタが4K解像度に対応しておらずフルHD解像度しかでなかった。
フルHDはつらいので変換アダプタを注文した。コンセントまで電源ケーブルが届かなかったのでOAタップも買った。あわせ買い対象商品のため、金額調整のために洗濯用洗剤も買った。
これで4K解像度で出力できるようになったので、最高になった。
ばしばしウインドウ開いても狭くならないので最高。若干もてあましている。
🗑今後の課題
モニタを買った結果、古いモニタの廃棄方法を調べることになってる。
— mzp (@mzp) October 20, 2018
PCリサイクルマーク...?
— mzp (@mzp) October 20, 2018
💡スマート家電
引越ししたので家電をいくつか進捗した。 せっかくなので、リモートで制御できるものを買った。
🤖ルンバ
床がちらかる前にルンバを買った。iPhoneアプリで制御できるなかでは一番安いモデルにした。
【Amazon.co.jp限定】アイロボット ルンバ691 wifi対応 複数床面対応 自動充電 ロボット掃除機 R691060
- 出版社/メーカー: iRobot (アイロボット)
- メディア: ホーム&キッチン
- この商品を含むブログを見る
動かしたらいきなりケーブルを巻き込んで停止した。ケーブルボックスを導入したら、なんとかなった。
ルンバちゃんがベースステーションのコンセントに突進していって自殺みたいになってる
— mzp (@mzp) August 15, 2018
@blackenedgold OAタップ & ケーブルボックスありがとうございます! これで、ルンバちゃんがケーブルを食べることがなくなります!! pic.twitter.com/i7mXrTPuhI
— mzp (@mzp) August 17, 2018
ルンバがケーブルを巻き込むのでケーブルボックスを追加購入した。なんかペット飼ってる気分になってきた
— mzp (@mzp) September 3, 2018
いまも週一くらいで停止している。かわいい。
💨エアコン
エアコンは部屋についてたので、NatureRemo miniを購入した。
スマートリモコン Nature Remo mini【Amazon Echo/Google Home対応】
- 出版社/メーカー: Nature, Inc.
- メディア: エレクトロニクス
- この商品を含むブログを見る
あまり調べずに買ったが、温度調整のUIが格好よくていい。温度計がついてるのもおもしろい。
💡照明
照明も部屋に備え付けだったが、調べたらリモコン操作に対応していた。
リモコンの信号をNatureRemo miniに覚えさせたいけどどうしたらいいのかなーと調べたが、どうやら実物が必須らしかった。Amazonで購入して初期設定したあとはしまってある。
スマートリモコンに覚えされるために照明の純正リモコン買ったけど、これは正しい道を歩んでるんだろうか
— mzp (@mzp) August 27, 2018
スマートリモコンがパターンを記憶したので、エアコンと照明のリモコンをしまった。なんか自炊したあとの物理本の処理に困る話と似てる
— mzp (@mzp) August 28, 2018
⚡️HomeBridge
それぞれ別のアプリから制御する必要があって不便だった。 iPhoneのHomeアプリから一括制御するためにnfarina/homebridgeを導入した。
部屋はひとつしかないがベッドルームということにしている。
HomeBridgeはRaspberryPiZeroWで動かしている。Raspberry Pi Zero W ケースキット - SWITCH-SCIENCEを買ったが、SDカードがついてないことを見落してて追加購入した。
家電をhomekitから扱えるようにするぞ!!ということで買ったRaspberryPi ZeroWが届いたが、micro SDカードがなかったので、おしまいです
— mzp (@mzp) September 3, 2018
セットアップ以下のサイトを参考にした。
- USB OTGを使ったRapsberry Pi Zero WH のセットアップ - Qiita
- Raspberry Pi 3 にhomebridgeを導入してsiriから家電操作できるようにした際の覚書 - Qiita
モジュールは以下のものをいれた。
💕感想
一番の利点はいつも手元にあるiOSデバイスから制御できるようになった点である。リモコンはどこに置いたか忘れがちだが、iPhoneかiPadかmacOSのどれかが手元にあることは多い。
Siri経由で操作できるけどそこまで便利ではない。
スマフォで家電制御するやつ、音声で制御できるとかよりも、どのiPhoneからでも電気を消せるのが便利な気がする
— mzp (@mzp) September 9, 2018
あとは以下のものをスマート家電化したいができてない。
- スマートロック。Qrioがよさそうに見えるが。
- コーヒーメーカー。自動応答するコーヒーメーカーは夢がある。RFC 2324 Hyper Text Coffee Pot Control Protocolをしゃべってほしい。
- 洗濯機・炊飯器。残り時間をアプリから確認したい。
☀️ダイナミックデスクトップ壁紙
Mojaveのダイナミックデスクトップでは時刻によって壁紙が変化する。 これを用いてミクさんが部屋の中を歩きまわる壁紙を作成した。
(livetune feat. 初音ミク「Redial」Music Video - YouTubeより)
🖥ダイナミックデスクトップ
macOS Mojaveの紹介ページに記載されているとおり、Mojaveでは時間の経過に応じて壁紙が変化する。
WWDCのキーノートでも、生活にあわせて変化する壁紙のデモがあった。
🛠仕組み
ダイナミックデスクトップ用の壁紙は他の壁紙と同様 /Library/Desktop Pictures
に配置されている。 異様にファイルサイズがでかい。
プレビューで開くと16枚の画像が確認できる。HEIFでは一つのファイルに複数の画像をまとめられるので、その機能を利用している。
それに加えてメタデータに時刻と画像の対応、より正確には太陽の位置と画像の対応を格納している。 ここに関しては、以下の記事が詳しい。
- macOS Mojave dynamic wallpaper – ITNEXT
- macOS Mojave dynamic wallpapers (II) – ITNEXT
- macOS Mojave wallpaper (III) – ITNEXT
💕 作る
WallpaperEngineを使った以下の壁紙の再現を目指す。動きまわってるのがかわいい。
ついったでオススメが流れてきて衝動買いしたWallpaper Engineいいぞ!
— にゃいぼん aka へらぶな (@nyaibon) 2017年1月4日
壁紙が動くだけで謎の未来感に感動している!steamで買えるぞ! pic.twitter.com/hHRSTRAuFg
livetune feat. 初音ミク「Redial」Music Video - YouTubeをダウンロードしてフレームごとに分割する。
youtube-dl 'https://www.youtube.com/watch?v=243vPl8HdVk' ffmpeg -i 'livetune feat. 初音ミク「Redial」Music Video-243vPl8HdVk.mkv' -f image2 %d.png
ここからいい感じの16枚を選ぶ。
複数の画像を束ねた上でメタデータを書き込むのは、mczachurski/wallpapper: Console application for creating dynamic wallpapers for macOS Mojaveで行なう。
brew tap mczachurski/wallpapper brew install wallpapper
macOS Mojave dynamic wallpaper – ITNEXTを参考に定義ファイルを雑に作る。
info.json
[ { "altitude": -0.3427528387535028, "azimuth": 270.9334057827345, "fileName": "1.png", "isPrimary": true, "isForLight": true, "isForDark": false }, { "altitude": -10.239758644725045, "azimuth": 81.77588714480999, "fileName": "2.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -4.247734408075456, "azimuth": 86.33545030477751, "fileName": "3.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 1.3890866331008431, "azimuth": 90.81267037496195, "fileName": "4.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 7.167168970526129, "azimuth": 95.30740958876589, "fileName": "5.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 13.08619419164163, "azimuth": 99.92062963268938, "fileName": "6.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 40.41563946490428, "azimuth": 129.18652208191958, "fileName": "7.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 53.43347266172774, "azimuth": 182.2330942549791, "fileName": "8.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 38.793128200638634, "azimuth": 233.5515919580959, "fileName": "9.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 11.089423171265878, "azimuth": 261.87159046576664, "fileName": "10.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": 5.1845753236736245, "azimuth": 266.4432737071051, "fileName": "11.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -6.248309374122789, "azimuth": 275.44204536695247, "fileName": "12.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -12.20770735214888, "azimuth": 280.07031589401174, "fileName": "13.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -39.48933951993012, "azimuth": 309.41857318745144, "fileName": "14.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -52.75318137879935, "azimuth": 2.1750965538675473, "fileName": "15.png", "isPrimary": false, "isForLight": false, "isForDark": false }, { "altitude": -38.04743388682423, "azimuth": 53.50908581251309, "fileName": "16.png", "isPrimary": false, "isForLight": false, "isForDark": true } ]
このファイルを用いてダイナミックデスクトップ用の壁紙を生成する。
wallpapper -o ~/Pictures/redial.heic -i info.json *.png
システム環境設定から設定する。なぜかサムネ画像がおかしい。
時刻を変更して動作を確認する。
✨感想
動画を壁紙にする場合と比較して変化がおだやかなので、そこまで気がちらない。よい。
こちらを覗き込む画像は夜に表示されるようにしている。 なので、出社前は空の部屋で、家かえってきてPCをつけるとミクさんと目があうので、体験としてよい。
もうちょっと応用したいが、変化してたのしい壁紙のアイデアが足りない。WallpaperEngineのときもそういう話をしていた。
「wallpaper engineめっちゃいいじゃん」「作るか」みたいな話をしたけど、結局、壁紙が動画になってるのが最高なんじゃなくて、ミクさんがかわいいだけだわ、という結論になった。
— mzp (@mzp) January 7, 2017
♨️銭湯
東京に引っ越したら銭湯によく行くようになった。東京にはいっぱい銭湯があってよい。
💼持ち物
タオルやシャンプーなどは置いてないことが多いので、自分のものを持っていく。 ダイソーで買ったビニールバッグにコンビニで買ったシャンプーセットを入れている。
水を持ち込んでる人もたまに見かける。 炭酸水を持ち込んで、飲み水と頭を洗うのに使ってて、なるほど感があった。
👍よい
- 大きい風呂に入れる。
- 10分くらい歩いたとこにあるので便利。ご飯を炊いてる間に風呂、とかをやっている。お風呂の四階に住む。 - 物件ファンほど近くはない。
- 銭湯によっては温泉にはいれるとこもあってお得。
- あまり混んでないので、雑に行ける。
🤔微妙
- 銭湯の料金 > 水道代なので、やりすぎると破産するかもしれない。
- 街中を歩いてた直後、店にはいって数十秒で全裸になるという行為に違和感を覚えてしまう。慣れない。
- 結構話しかけられる。街中で話しかけられることはほぼないのに、全裸だと話しかけられるのおもしろい。
AquaSKK 4.6.0/4.6.1: Mojave対応
Mojave対応を行なったAquaSKKを4.6.0としてリリースした。2018-9-1: HighSierraで一部機能(#84)が動かない不具合が見付かったので、修正して4.6.1としてリリースした。
https://github.com/codefirst/aquaskk/releases/tag/4.6.1
🌓ダークモード
Mojaveではダークモードが導入された。
(WWDC2018 Keynoteより引用)
AquaSKKもダークモードに対応させ、候補ウインドウがダークモードで表示されるようにした。 さらに、アクセントカラーも扱うようにした。
🎨 アイコン色の調整
ダークモードではASCIIモードのモードアイコンの視認性がかなり低くなる。
Mojaveより前から存在していた問題(Issue #75)だが、ダークモードが強化されたMojaveではより顕著になる。
色合いを調整して、視認性を向上させた。
👋32bitサポートの廃止/libc++への切り替え
Xcode10から32ビットアプリケーションのサポートが廃止された。Mojave以降は64ビットアプリケーションのみのサポートとなるので、これを気に32ビットサポートは廃止した。
(WWDC2017 Keynoteより引用)
またlibstdc++のサポートが廃止されたので、libc++に切り替えた。これは目に見える影響はないと思う。
🙊NDAへの配慮
本記事はNDAに配慮し、Xcode 10やmacOS Mojaveのスクショショットは利用していない。Mojaveが正式リリースしてから書けば楽だったが、キリがついたのでリリースした。若干、無理がある。
ダークモード対応中の様子は楽しいので、それはどこかでロンダリングした上で出したい。