tag:blogger.com,1999:blog-420675798056405352024-03-06T01:31:44.400+09:00A-Liaison BLOGakisuteが主に技術的なネタを書き溜めるブログです。akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comBlogger331125tag:blogger.com,1999:blog-42067579805640535.post-86737307552058687532021-12-01T00:00:00.463+09:002021-12-01T00:37:45.282+09:00セキュリティを一切考慮しないMMORPGを開発するとどうなるか<p style="text-align: left;">どうもご無沙汰しております。本Blogが私の年1回の生存報告、兼、<a href="https://adventar.org/calendars/6234" target="_blank">アドベントカレンダー用</a>と相成って久しいですが、今年も一発恒例行事として筆を取らせていただきたいと思います。</p><p style="text-align: left;">今年、私が話題に取り上げますのは、とあるゲームです。<a href="https://ja.wikipedia.org/wiki/%E3%82%A2%E3%83%9E%E3%82%BE%E3%83%B3%E3%82%B2%E3%83%BC%E3%83%A0%E3%82%B9%E3%82%BF%E3%82%B8%E3%82%AA" target="_blank">Amazon Game Studios</a>という会社が開発・リリースしました、<a href="https://store.steampowered.com/app/1063730/New_World/" target="_blank">New World</a>というMMORPGについてご紹介させていただきたいのです。ゲームの話題には一切興味がない読者諸君も、どうか少し我慢して、私に騙されたと思って最後まで話を聞いていただけませんでしょうか。そもそも、<b>あの</b>Amazonが開発したMMORPGというのですから、どれほどゲームに興味がなくても、技術に興味のある方でしたら、少しは興味深く感じられるのではないでしょうか?</p><p style="text-align: left;">けして後悔はさせませんよ。<b>悪い方向にね。</b></p><p style="text-align: left;"><br /></p><p style="text-align: left;">さて、ゲームに何ら興味知識のない方にもわかるように少し解説を入れさせていただきますと、<a href="https://ja.wikipedia.org/wiki/MMORPG" target="_blank">MMORPG</a>というのは「数千人単位の人間がネットワーク上に構築された一つの世界を共有するRPG」です。要するに<a href="https://ja.wikipedia.org/wiki/%E3%82%BD%E3%83%BC%E3%83%89%E3%82%A2%E3%83%BC%E3%83%88%E3%83%BB%E3%82%AA%E3%83%B3%E3%83%A9%E3%82%A4%E3%83%B3" target="_blank">SAO</a>です・・・いや、この説明だと最近の若いのには通じないですかね・・・まぁそういう感じのアレで、要するに滅茶苦茶多数の人間がサーバーに同時接続するゲームってことです。そして説明するまでもございませんが、Amazonという会社は世界で最も巨大なクラウドサービスの一つを展開している会社であります。我々技術畑の人間がシンプルに連想するのは、これほど相性の良い組み合わせはないに違いない、そうですね?</p><p style="text-align: left;">実際、発売前から私もそう思っていましたし、発売された後も、彼らはAmazonの名に相応しい仕事を見せつけてくれました。発売前から話題の合った本作はあれよあれよという間に<b>最大同時接続90万人</b>を記録するモンスターゲームに成り上がったのです。いいですか、90万人<b>同時接続</b>ですよ、90万人。90万DAUとか、90万PV/dayとか、90万Session/dayだとか、そういう雑魚みたいな指標ではありません、<b>同時接続</b>で90万です。貴方は90万の同時接続とリアルタイムレスポンスを保証できるシステムを構築できますか?言うまでも有りませんが私ならその場で全く不可能だと匙を投げますね。</p><p>さすがは開発元がAmazonの名を冠するゲーム会社だけあって、サーバーは相当な負荷に耐えきったと言えます。最大同時接続90万人をリリース直後から支えられるMMORPGというのはなかなか世に存在するものではありません。私も思わず「やっぱりAmazonはすごい」と驚嘆したものです。</p><p><br /></p><p>ですが、我々は後から思い知ることになります。何故、彼らのサーバーが、同時接続90万という途方も無いアクセスに耐えることが出来たのか、その本当の理由を。</p><p><i><b><span style="color: red;">・・・サーバーが何もしていないからです。</span></b></i></p><p><br /></p><h1 style="text-align: left;">dupe, exploit, cheatの軌跡 </h1><p>先に説明したとおり、MMORPGというのはネットワーク上に構築された一つの世界を共有するゲームです。つまり、プレイヤーは全員常時サーバーにリアルタイムで接続された状態になっていますし、必然、全てのプレイヤーのデータはサーバ上で管理されています。</p><p>普通はね。</p><p>だって、各自のローカルサイドでプレイヤーのデータが管理されていたら、皆が好き勝手にローカルサイドでセーブデータなりメモリなりを書き換えたら、簡単にお金やアイテムが無限に増殖されてしまうでしょう?それでは皆のゲームが成り立ちませんから困りますね?そういう悪いことをゲームで行う・・・<i>チート</i>をする人間は、ここ数年右肩上がりに増えておりますから、ゲームを作る側も畢竟、高レベルのチート対策をゲームに組み入れるのが当然の事項となっているわけです。</p><p>そう、だから、普通は、どんなMMORPGを作るときもね、チート対策をするんですよ。</p><p>普通はね。</p><p><br /></p><p>さて、あれはゲームがリリースされて30日もしたころでしょうか。</p><p>コミュニティの中で「アイテムが増殖できる」だとか「Goldが無限に増やせる」だとか、そういう噂が流れ始めました。</p><p>最初はただの噂だと思われていたのですが、<a href="https://forums.newworld.com/t/treasury-coins-and-mule-accounts/494402" target="_blank">1000万Goldを超える所持金や</a>、ゲーム内で最も貴重なVoidmetalと呼ばれる金属を数千個単位で所持するスクリーンショットが暴露され始めると、「どうやら本当にアイテム増殖バグがあるらしいぞ」ということがわかってきました。私は当初、高度なハッカーの類がパケットを改ざんして攻撃しているのだと推測していたのですが、一週間もしたころでしょうか、突然公式Forumにアイテム増殖の具体的な方法が書き込まれ、一同が唖然となります。</p><p>その方法とは・・・</p><p><br /></p><blockquote style="border: none; margin: 0px 0px 0px 40px; padding: 0px;"><p style="text-align: left;">1. 個人間トレードを開始する。</p></blockquote><blockquote style="border: none; margin: 0px 0px 0px 40px; padding: 0px; text-align: left;"><p>2. 渡す側は、増殖したいアイテムを相手に渡すようにして、Confirmボタンを押す。</p><p>3. 受け取る側は、増殖したいアイテムがトレードに乗ったのを確認したら、意図的に通信を遮断する。</p><p>4. 受け取る側は、Confirmボタンを押す。</p><p>5. しばらく待ってから、渡す側がCancelボタンを押して、取引を中断する。</p><p>6. 中断されたのを確認してから、受け取る側は、通信を復旧する。</p><p>7. 双方がアイテムを持った状態になる。</p></blockquote><p> </p><p><b>そう、トレード中に、通信を止めるだけ。</b></p><p>たったのこれだけでアイテムが増殖できてしまうのです。</p><p><br /></p><p>まさか21世紀も1/5以上が終わった現代において「サーバーを一切介さないでトレードが成立するMMORPG」をあのAmazonの名を冠するゲーム会社がリリースする日が来ようなどとは一体全体誰が予想し得たでしょうか。</p><p><br /></p><p>結果、サーバー全土に渡る経済封鎖が行われ、個人間トレード、交易所、ギルドバンクの利用などあらゆる富の移動が不可能にされたのですが、現実世界だけではなくゲーム内でも経済活動の制限が行われる日が来るとは実に皮肉なものです。</p><p>しかも、問題はこれだけに留まりませんでした。<b>ゲームをWindowedモードで起動し、戦闘中にゲーム画面のWindowを激しく動かしまくって意図的にクライアントサイドでの処理を止めると、なんと完全無敵になる</b>というバグが発見されたのです。全く理解できません。</p><p><br /></p><p>まさか21世紀も1/5以上が終わった現代において「クライアントサイドの処理を止めると一切ダメージを受けなくなるMMORPG」をあのAmazonの名を冠するゲーム会社がリリースする日が来ようなどとは一体全体誰が予想し得たでしょうか。</p><p><br /></p><p>あまりの事態に、とうとう開発者たちも重い腰を上げて火消し文章を書きます。</p><p><a href="https://forums.newworld.com/t/dev-blog-update-on-current-issues/504297/1" target="_blank">https://forums.newworld.com/t/dev-blog-update-on-current-issues/504297/1</a></p><p>それによると、本作は<i>「To be very clear, New World is not client authoritative — from a simulation standpoint, New World is entirely server based」</i>などと主張されていましたが、誰もこのような世迷い言を信じる者はいませんでした。クライアントサイドでの入力によってサーバーサイドの挙動が不正に変化してしまうのであれば、それは入力に対するValidationが何ら行われていないということであり、Server-Based Authoritativeなアプリケーションとは言えません。この声明発表は単に開発者たちが自らの無能を晒しただけに終わりました。</p><p><br /></p><p>数日後。</p><p>アップデートによりアイテム増殖も無敵化も修正され、経済封鎖も解除され、事態は沈静化するかのように思われました・・・</p><p>が、今度は家具を無限に増殖させるバグが発見されました。</p><p>その方法とは・・・</p><p><br /></p><blockquote style="border: none; margin: 0px 0px 0px 40px; padding: 0px;"><p>1. 家のデコレーションを開始する。</p></blockquote><blockquote style="border: none; margin: 0px 0px 0px 40px; padding: 0px;"><p>2. 増殖したいアイテムを家に配置する。</p><p>3. 増殖したいアイテムが家に配置されたら、意図的に通信を遮断する。</p><p>4. しばらく待ってから、Cancelボタンを押して、家具配置をキャンセルする。</p><p>5. 中断されたのを確認してから、通信を復旧する。</p><p>6. アイテムが家に配置された状態のまま、手元にも残る。</p></blockquote><p><br /></p><p><b>お気づきかと思いますが、前回と全く同じ手順です、彼らは何も学んでいません。</b></p><p>あの日本を代表するみずほ銀行ですらこんなデタラメな修正はしないのでは無いでしょうか。こちらの問題も2回目の経済封鎖の後、現在では修正されているのですが、2回も同じ手口で成功したのですから、今でも似たような手口によるアイテム増殖が可能なのではないかという噂がコミュニティ内で絶えません。もはや完全な無法地帯です。</p><p><br /></p><p><a href="https://www.reddit.com/r/newworldgame/comments/qhxi9s/chat_in_my_server_wts_large_sausage/" target="_blank">他にも、チャット欄にHTMLタグが使えてしまうバグだとか、</a></p><p>チャット欄のHTMLタグを利用してリンクをmouseoverした瞬間に攻撃するコードを仕込むバグだとか(onmouseoverが動作してしまうらしい)</p><p>全く、話題に事欠かないゲームです。</p><p><br /></p><p>さて、ここまでお読みいただいたエンジニア諸兄の中にはお気づきの方もいらっしゃることかと思いますが、このNew Worldというゲーム、根幹がそもそもMMORPGとして動作するように設計されていない可能性が極めて高いです。実際、企画の初期段階では、MMOではなく、MOのPvPゲーム(ゲームに詳しい方なら<a href="https://store.steampowered.com/agecheck/app/252490/" target="_blank">Rust</a>といえばご存知でしょうか)として作られていたという噂があります。そのため中央のサーバーが全ての処理を解する設計になっていないのでしょう。現在のNew Worldは単に我々が動作させているクライアントと全く同じものをHeadlessモードでサーバー側で動作させ、その処理を真とするような実装になっているだけなのではないか、と私は勝手に推測していますが、あくまで推測の域を出ません。</p><p>ここでの学びは、根幹がそもそも間違っているシステムは、後から何をやっても救えないということです。</p><p>くれぐれも皆様もご用心を。</p><p><br /></p><h1 style="text-align: left;">bot、その進化の軌跡</h1><p>もう一つ、本作New Worldの世界で技術的に興味深い話題があります。それは<b>Bot</b>です。MMORPG界隈においてBotという単語を使った場合、それは「プログラムで自動的にキャラクターを操作し、何らかの行動を行わせ続ける」行為を指し、全世界に存在するほぼすべてのMMORPGのエンドユーザーライセンス契約条項において禁止されています。禁止されている理由は多々ありますが、基本的にMMORPGというのは単一の行為をひたすら尋常ではない回数と時間に渡って繰り返しまくることで資源やアイテムやお金や経験値といったリソースを収集して強くなるのが目的ですから、当然<b>自動化と相性がいい・・・</b>失礼、自動化すると真面目に<strike>刺し身たんぽぽをポチポチ</strike>やってる人間が不利益を被るわけです。そういったわけで禁止されているんですね。</p><p>まぁ、禁止されてるってことは、当然やる人間が居るわけです。本作New Worldにおいても全く例外ではないのですが、ただでさえバグだらけでマトモな入力検証すら行っていないゲームであることがすでに露呈しているわけですから、今や世界中のBot職人の格好の的となり、過去のゲームで見られた単純なBotとは比較にならないほど洗練されたBotの数々が世界の至るところで観察されるようになったのです。</p><p>そのいくつかをご紹介しましょう。</p><p><br /></p><p>バージョン1.0: Fishing Bot</p><iframe allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" allowfullscreen="" frameborder="0" height="315" src="https://www.youtube.com/embed/0BE_MiZ0R68" title="YouTube video player" width="560"></iframe><br /><p>最初期からいるBotです。多分ゲーム開始2週間後にはもう大量発生していたと思います。名前の通り、自動的に釣りを行うBotとなっています。釣りは一定の位置から動く必要もなく、ただひたすらマウスをタイミングよくクリックするだけで無限に実行できますから、大変自動化しやすいんですね。おまけに本作の釣りは<b>針を垂らした瞬間に針にかかった魚の種類がサーバからクライアントに送信される</b>というキ○○○・・・失礼、少々脳細胞が足りていない実装になっておりまして、そのためパケットをキャプチャすることで目的の魚が針にかかるまで即座にリセットを繰り返すという大変効率の良い機能まで兼ね備えております。</p><p>ちなみにこのBot、2021/12/01現在でも対策されておらず、世界中どこでも見ることができます。</p><p><br /></p><p>バージョン1.1: Mining/Logging Bot</p><iframe allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" allowfullscreen="" frameborder="0" height="315" src="https://www.youtube.com/embed/rAKtNbSXIt8" title="YouTube video player" width="560"></iframe><iframe allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" allowfullscreen="" frameborder="0" height="315" src="https://www.youtube.com/embed/R0GisA2DZ_s" title="YouTube video player" width="560"></iframe><p>次に現れたのがこちら。鉱石を自動で掘ったり、リソースを自動で採集したりしてくれるBotです。本作の鉱石やリソースは毎回全く同じ座標に再生するため、同じ座標まで歩いていってボタンを押すだけで自動化できちゃうんですね。とはいえ、先に他の人が採掘していたり、リソースが存在しなかった場合にどうするか、とか、移動が邪魔された場合に対応する必要があるとかで、単純な釣りBotよりは遥かに実装難易度が高いようで、私がプレイしてたサーバーではなかなか見かけませんでした。</p><p><br /></p><p>バージョン2.0: Questing Bot</p><p><a href="https://www.reddit.com/r/newworldgame/comments/qhck8d/complex_scripted_questing_bots_already_available/" target="_blank">https://www.reddit.com/r/newworldgame/comments/qhck8d/complex_scripted_questing_bots_already_available/</a></p><p>YouTubeに動画が見つからなかったのでRedditから引用してきたのですが、必見です。とうとう彼らBotは釣りや採集といった単純動作だけではなく、戦闘とクエストを自動でこなせるように進化してしまったのです。何十体ものBotが蟻の行列のように規則正しく動いて規則正しくクエスト目標の敵をしばき倒す姿は戦慄さえ覚えますが、AIだとかRPAだとかの波はこんなゲーム内にも訪れてきているわけですね。</p><p><br /></p><p>バージョン3.0: PvP Bot</p><p>2週間ほど前ぐらいに発見された最新のBotです。これは噂レベルでしか聞いたことがないので真偽不明ですが、とうとうこの最新のBotはPvP (Player vs Player:対人戦) クエストを自動でこなせるようになったとのこと。といってもBotが我々人間相手に斬りかかってくるわけではなく、対人戦モードを有効にした状態で敵の基地に忍び込んでアイテムを手に入れて帰ってくる、みたいなお使いクエストでしかありません。ただのゾンビですから発見さえすれば簡単に始末できるのですが、問題はこれらが自動化されていて24時間365日常に動き続けているという点にあります。我々人間には休息が必要ですが、彼らは一瞬一秒たりとも休むこと無く、何度殺されても諦めること無く、ひたすら目標めがけて突進してくるのです。これにはどれほど優れた人間も太刀打ちすることが出来ません。これはゲームの中のお遊びですから笑い話ですが、現実世界の戦争がこのように無人の自動兵器がひたすら突撃を繰り返してくる世界になってしまったらと思うと、薄ら寒い思いがしますね。</p><p><br /></p><p>ちなみにですが、今しかたgithubをちょっと調べてみたところ、golangで書かれたNew World向けのBotのソースコードが転がっていたりしましたので、ちょっと探せば皆様も簡単にBotを運用できるかと思いますが、<b><span style="color: red;">決して真似しないでくださいね。</span></b></p><p><br /></p><h1 style="text-align: left;">まとめ</h1><p>なかなか得られない機会かもしれませんが、もし読者の皆様におかれまして、MMORPGを開発するという案件に参画するご縁がお在りになりましたら、その際は是非本記事の内容をご参考にしていただければ幸いです。</p><p><br /></p><p><a href="https://adventar.org/calendars/6234" target="_blank">#pyspa Advent Calendar 2021 - 1日目</a></p><p><br /></p>akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-42491568806814341642020-12-01T00:00:00.492+09:002020-12-01T00:00:00.194+09:00マネージメントに必要なことは全てゲームから学んだ<p>この投稿は毎年恒例、<a href="https://adventar.org/calendars/5310" target="_blank">pyspa Advent Calendar 2020</a>の1日目の投稿になります。</p><p>どうもご無沙汰しております、akisuteです。すっかり年に1回アドベントカレンダーのときにだけ顔を見せる人になっておりますが、おかげさまで無事平穏に過ごしております。</p><p><br /></p><p>さて突然ですが私はプログラマーを引退しました。</p><p><br /></p><p>なぜなら今年で36歳だからです。<a href="https://tomosta.jp/2019/05/21/420/" target="_blank">プログラマーは35歳になったら定年</a>ですね。</p><p>実際のところ、このぐらいの年になると、よほど何らかの意志が働かない限り、技術に対する情熱みたいなものが失われてくると思います。もちろん本当に技術とプログラミングが好きな人は間違いなく35歳なんかで情熱を失ったりはしないと断言しますが、残念ながら私はそうではなく、もはやiPhoneには大した興味が湧いておりませんし、最近はJavaだのGoだのTypescriptだのVue.jsだのといったものを必要に応じて細々と書いている程度に過ぎません。</p><p>さてプログラマーを引退してどうしたかといいますと、マネージャーになりました。といっても、正確にはマネージャーなんてエラソーな立場ではありません。スクラムチームのリーダーといった程度の役割です。昔っぽく言えば、係長でもないので、班長ぐらいでしょうか。上長の人からお前にならできると思うからぜひやってくれとお願いされて引き受けることにしました。ついでにお給料もちょっと増やしてもらったので満足です。このままモチベの上がらない技術者を続けるよりは未来もあるでしょう。</p><p>とはいえ、問題点もあります。なにせ、私は基本的に人間が大嫌いです。可能であれば常に一人でいるほうが嬉しいと言うのに、何が楽しくて他人のマネージメントなんてしなければならないのでしょうか。実は私の上長には人を見る目が無いんじゃないでしょうか。さらに、そもそも私は生まれてこの方マネージメントに関する勉強も教育も一切受けたことがございません。ズブのド素人にプログラマができないのと同じことで、全くの未経験者がマネージャなんてできるわけがないでしょう。</p><p>まぁ、それでもいいや、別に会社の業績がどうなろうが知ったことではないし、お給料くれるというならやってやろうじゃないかと思って一念発起、できるかどうかわからないまま快く引き受けてしまったのですが、いざやってみたら案外マネージメント業務ができちゃったので、今日は今年一年のマネージメント体験で得られたことをつらつら書いてみようかと思っております。</p><p><br /></p><h2 style="text-align: left;">マネージメント業務を行う人間に必要な適正</h2><p>古今東西、何の仕事でもそうなんですが、仕事には適正というやつがあります。</p><p>まず能力面での適正というやつは以下の通りだと思っています。</p><p></p><ul style="text-align: left;"><li>論理的思考が人並み程度に可能</li><li>人前でしゃべることができる</li><li>恥ずかしがったり躊躇したりしない、慌てず騒がず堂々としている</li></ul><p></p><p>たったのこれだけです。</p><p>次に考え方、思考面での適正というやつは以下の通りだと思っています。</p><p></p><ul style="text-align: left;"><li>目的を達成するのが楽しいと思える</li><li>エンジニアの苦痛を理解できる、プログラマやエンジニアが何をされたら嬉しくて何をされたらクソムカつくかを心の底から共感できる</li><li><a href="https://imalabya.com/thoughts/2018/02/08/be-leader-not-boss/" target="_blank">ボスとリーダーの違い</a>を理解しており、ボスではなくリーダーになろうという意思がある</li><li>一方で自らがチームで最も価値のない人間であるということを自覚することができる・・・マネージャーは別に何も生産しません。生産価値ゼロです。単に生産価値を生み出している他のチームメンバーの雑用奴隷をしているだけです。自分のほうが上だとか勘違いするのは思い上がりも甚だしいです。したがって成功は全てメンバーのおかげであり、失敗は自分が無能な判断をしたためか、サボったから、と考えるのが当然だと言えます。</li></ul><p></p><p>この程度の要素があれば基本的に誰でもマネージメント業務は可能だと思います。別に人間が好きとか嫌いとかは必要な要素ではありません。もちろん、人によって方法論が異なるとは思いますが、少なくとも私にとっては必要ありませんでした。</p><p><br /></p><h2 style="text-align: left;">マネージメントはゲームである</h2><p>はい、ゲーム脳の私の考えることですが、マネージメントはゲームです。ゲーム機やコンピュータやゲーム盤を使うのか、現実世界で行うのかの違い程度しかありません。そもそもゲームというのは基本的にマネージメントそのものです。<a href="https://ja.wikipedia.org/wiki/%E3%82%B7%E3%83%A0%E3%82%B7%E3%83%86%E3%82%A3%E3%82%B7%E3%83%AA%E3%83%BC%E3%82%BA" target="_blank">シムシティ</a>のようなゲームだってマネージメントですし、<a href="https://ja.wikipedia.org/wiki/%E3%82%A8%E3%82%A4%E3%82%B8_%E3%82%AA%E3%83%96_%E3%82%A8%E3%83%B3%E3%83%91%E3%82%A4%E3%82%A2" target="_blank">エイジオブエンパイア</a>や<a href="https://ja.wikipedia.org/wiki/%E3%82%B9%E3%82%BF%E3%83%BC%E3%82%AF%E3%83%A9%E3%83%95%E3%83%88" target="_blank">スタークラフト</a>といったゲームもマネージメントですし、アクション要素以外のほぼすべての要素はマネージメントであると言えます。それどころか最近はチームで対戦するバトルロワイヤル系のゲームや対戦ゲームが増えてきました。これらのゲームで味方同士が協力して相手に打ち勝つというのはチームメイトのマネージメントそのものであると言えます。すなわち、ゲームをしている人なら、誰でも毎日マネージメントを実践していると言えるのではないでしょうか。</p><p><br /></p><h3 style="text-align: left;">最低限の定石を覚える</h3><p>古今東西いかなるゲームにも定石という概念は存在します。古くは将棋囲碁チェスにも存在しますし、リアルタイムストラテジーゲームでしたら初期オーダーと言う概念が存在しますし(HCCC→農民掴んで豚に送りつつ家1件建設・・・)最近流行りのバトルロワイヤルでも初手どこに降下するのが良いとか、どの武器が強いとか、何を確保すれば有利だとか、何のゲームにも必ず定石と呼ばれている概念が存在するわけです。</p><p>上達したいなら何のゲームでも必ず定石を学ばなければなりません。定石に当てはまらない動きができるのは定石を知っているからです。したがって、マネージメントもゲームですから、定石を学ぶ必要があります。</p><p>マネージメントにおける定石とは何かという話ですが、ありがたいことに世の中にはマネージメント論みたいな話があって、特にソフトウェアエンジニアリングのような知的労働をするチームをマネージメントするみたいな方法論だけでも沢山存在します。私の場合はスクラムという方法を学びました。まぁ、会社がカネ出してくれたので、専門の講座を受けに行っただけなのですが、おかげさまで最低限の定石は得られた気がしています。それにスクラムの元となっている経験主義という考え方は、答えのない決断を迫られることが多いマネージャーにとって必要不可欠な考え方ではないかと思いますし。</p><p><br /></p><h3 style="text-align: left;">マネージメントという仕事はどういうゲームか考える</h3><p>他の現実世界の事象と全て同じなのですが、ゲームと違い現実世界には明示的なクリア条件・達成条件・ゴールといったものがありません。またゲームと違いパラメータが明示的ではありません。全てのゲームは(アクションゲームですら)基本的に数字やフラグと行ったパラメータを適切に操作してゴールを達成するものですから、それらパラメータとゴールを適切に自分で設定してやる必要があります。</p><p><br /></p><h3 style="text-align: left;">マネージメントのゴールを理解する</h3><p>これは基本的に組織のゴールを達成することとほぼ等価になると思います。それとは別に、マネージメントに登場するパラメータの値をうまく調整してチームが破綻しないようにするというのも大事なゴールになります。すなわちパラメータを維持調整促進しつつ組織のゴールを達成するのを狙うことになるでしょう。</p><p>厄介なのは組織のゴールというやつが現実世界ではしばしば達成不可能な世迷い言だったり、達成するためにはチームメンバーやパラメータを犠牲にしなければならないようなものだったりする点です。そのため、現実世界においては俗に「調整」と呼ばれる、組織のゴールを達成するフリを上には見せつつ、自らの中では差し支えないゴールに置き換えておき、上に対してはフィルターを通して進捗を返すみたいな作法が必要になってきたりしますが、お察しの通り、そのような作法が必要な地点で大体何かが間違っています。人間のやることですから何かが間違うのは仕方のないことなのですが、せめて自らのチームの中ではそのような間違いと調整が起きないように最大限尽力するべきでしょう。<br /></p><p><br /></p><h3 style="text-align: left;">マネージメントするパラメータを理解する</h3><div>というわけで次はパラメータを定義します。私が定義しているチームマネージメントのパラメータは以下のとおりです。</div><div><ul style="text-align: left;"><li>士気</li><li>リスペクト</li><li>発言力</li></ul></div><p style="text-align: left;">これだけです。順番に見ていきます。</p><p style="text-align: left;"><br /></p><h3 style="text-align: left;">士気</h3><p style="text-align: left;">極めて重要な値で、端的にいえばやる気です。他のゲームに例えるならライフポイントやヒットポイントとバフ・デバフ係数の両方を兼ねている最重要値です。この値を減らしたらゲームオーバーだと思ってください。あらゆる手段を用いて高める必要があります。<a href="https://ja.wikipedia.org/wiki/Hearts_of_Iron" target="_blank">Hearts of Iron</a>という有名な第二次世界大戦シミュレーターゲームがあるのですが、あのゲームで戦闘が敗北するのは部隊の兵士が死ぬからではなく、士気がゼロになるからです。人はそうそうかんたんには死にませんが、やる気が無くなると死んだも同然の状態になります。そしてやる気はいともたやすく無くなります。</p><p style="text-align: left;">士気を上げる方法、下がってしまう要因などはいくつもあると思いますが、あまりにもたくさんありすぎるのと、余りにも重要な概念ゆえ、後ほど場を設けて述べさせていただきます。</p><p style="text-align: left;"><br /></p><h3 style="text-align: left;">リスペクト</h3><p style="text-align: left;"><a href="https://ja.wikipedia.org/wiki/%E3%82%B0%E3%83%A9%E3%83%B3%E3%83%89%E3%83%BB%E3%82%BB%E3%83%95%E3%83%88%E3%83%BB%E3%82%AA%E3%83%BC%E3%83%88" target="_blank">GTA</a>とか<a href="https://ja.wikipedia.org/wiki/%E3%83%AC%E3%83%83%E3%83%89%E3%83%BB%E3%83%87%E3%83%83%E3%83%89%E3%83%BB%E3%83%AA%E3%83%87%E3%83%B3%E3%83%97%E3%82%B7%E3%83%A7%E3%83%B3" target="_blank">Red Dead Redemption</a>とか<a href="https://ja.wikipedia.org/wiki/Fallout_%E3%82%B7%E3%83%AA%E3%83%BC%E3%82%BA" target="_blank">Fallout</a>とか、そういう洋物のゲームによく出てくる概念ですが、日本語化が難しいのでそのままリスペクトとします。これは何かというと「アイツすげえな」「アイツは便利なやつだ、俺にとって使える」「アイツは有能だ」などといった他人からの正の評価を表す概念です。</p><p style="text-align: left;">あなたも自分の身の回りにリスペクトがある人が一人ぐらいはいるのではないでしょうか。そして想像してほしいのですが、そういうリスペクトがある人物からお願いされたり依頼されたことと、道端の他人にお願いされたり依頼されたこと、どちらのほうを聞いてあげたくなる、頼まれたくなるでしょうか。考えるまでもありませんね。</p><p style="text-align: left;">そう。マネージメントを行う上で極めて大事な要素です。コンピューターゲームの世界ではユニットを選択して右クリックしたら文句も言わずにその通りに動いてくれるものなのですが、現実世界ではそのような魔法は発生しません。あなたがお願いする内容が、あなた自身のリスペクトと比較して十分であれば、高い確率で(絶対ではなく確率で!)聞いてもらえるものなのだ、というふうに理解してください。したがってより無茶苦茶な内容を聞いてもらいたいのであれば、あなた自身にとんでもない量のリスペクトが必要になります。もし、リスペクトがない人物の依頼や命令が通っているのであれば、それは全く自然な状態ではなく、「上司の命令だから」「金をもらえるから」とかそういう別の動機によって発生している状態になります。これは極めてチームの運営上危険な状態であり、どんどんと歪みを生み出し最後には爆発してチーム自体が消し飛ぶことになります。</p><p style="text-align: left;">健全なチームは、上司の命令だからでもなければ、金をもらえるからでもなく、依頼を受けた側が自主的に自分もそうしたいと思っているから依頼を受けるか、または依頼主に十分なリスペクトがあるから成り立っているのです。</p><p style="text-align: left;">そんな大事なリスペクトですが、これは他人からの相対的な評価ですので、評価する側が何を大事にして何を大事にしないかによっても高め方が変わってきます。例えば評価する側の人が技術を最も大事にするタイプの人間であれば、その人からのリスペクトを高める方法は自分自身の技術力を高める以外にありません。逆に評価する人が人間性を求めている人なのであれば、あなた自身に人間的な魅力、ユーモアがあるとか、明るい性格をしているとか、そういう方法が必要になります。</p><p style="text-align: left;">ですがもっと簡単にリスペクトを高める方法があります。ゲームと同じです。他人のお願いを聞いてあげるだけです。どんなに簡単な内容でも構いませんから、対価を求めず相手のお願いをすべて聞いてあげてください。失敗しないようにね。すると「アイツは自分のお願いしたことを色々きちんとこなしてくれたいいやつだな」と思ってもらえるものです。私はもはや技術力がないアラフォーですから、チームのために様々な雑用をこなしてエンジニアの苦痛を取り除くことでリスペクトを稼ぐしかありません。</p><div style="text-align: left;"><br /></div><h3 style="text-align: left;">発言力</h3><div>遥か大昔に初めて<a href="https://ja.wikipedia.org/wiki/%E9%AB%98%E6%A9%9F%E5%8B%95%E5%B9%BB%E6%83%B3%E3%82%AC%E3%83%B3%E3%83%91%E3%83%AC%E3%83%BC%E3%83%89%E3%83%BB%E3%83%9E%E3%83%BC%E3%83%81" target="_blank">ガンパレード・マーチ</a>というゲームをプレイしたときに度肝を抜かれたのがこの発言力というパラメータでした。要するにお金のようなもので、対人関係で他人にお願いをしたり、会議で発言する際に必要となり、勲章をもらったりすると増えます。当時はガンパレの独特で斬新すぎる仕組みを上手く理解できなかったり、最強の武器(NEP)を手に入れるための方法がカネで買うでも敵からのドロップでもなく発言力で取り寄せるという内容に正直何だこれはと思ったものなのですが、今なら言えます、このガンパレード・マーチというゲームほど対人関係や発言力の概念を上手く取り入れているゲームは今に至るまで他に存在しません。</div><p>発言力はリスペクトと直結しています。リスペクトの値が高ければ高いほど発言力は大きく生産されるようになり、自分の意見を通しやすくなりますが、発言力を浪費した結果成果が上がらなかったりすると、リスペクトが減少し、さらに発言が難しくなるという負の循環に陥ってしまいます。</p><p>ちょっとわかりにくいかも知れないので具体例を挙げますが、あなたの身の回りにいらっしゃるクソ無能認定している人間を一人思い浮かべて、そいつがエラそうに自分の参加する会議で何かクソ無能っぽい頓珍漢な世迷い言を発言しているさまを思い浮かべてみてください。どうでしょうか。口を閉ざして視界から失せろと思うに違いありません。そうして自分の士気が下がります。これが、リスペクトのない人間が、発言力を無駄遣いしてさらにリスペクトを落とし、他人の士気を落としてチーム全体にダメージを与える状態です。こういう状態を避ける必要があるわけです。簡単ですね。</p><p>発言力は基本的に何でも発言・・・すなわち自らの意思を表明すれば減り、発言しなければ自らのリスペクトに相当する値まで自動的に回復するようなものなのですが、面白いのは発言によってリスペクトを得ているタイプの人間が余りにも発言しなさすぎると価値がない人間だと思われてリスペクトが減少し結果として発言力も低下するとか、逆に十分なリスペクトがある人間が発言すると発言することによってさらにリスペクトが高まる結果話せば話すほど発言力が増えていくとか、ポジティブフィードバックを受けやすいパラメータに思えます。</p><p>増やす方法はリスペクトを増やすことですが、減ってきたときには不用意な発言を厳に慎み、ここぞという場面でのみ最も適切な意見を述べ、メンバーの信頼を再び勝ち取る必要があるでしょう。</p><p>エンジニアでよくあるのが、どれほどリスペクトのある技術力のある人でも、発言すれば発言するほど発言力は消費されてしまうということです。つまり、どれほどのド正論だったとしても、ああでもない、こうでもない、それは間違っているからこうしろ、と言い続けると発言力が低下してしまい、結果として本当に大事な場面で反対に合ってしまうことが往々にして発生します。残念ながら人類というのは完全に論理的な生き物ではなく、どれほどの正論でも、どれほど理論的に正しいことでも、自分がムカつくこと=すなわち相手ばかりが正論を発言して結果を得続けるという状況には容易に賛成してくれません。時には如何に相手が間違っていてもあえて無視し、自らの発言力を維持しより大事に備える必要がある、というケースも存在すると思っています。</p><p><br /></p><h2 style="text-align: left;">チームメンバーの自主性の話</h2><p>よくチームメンバーの自主性を重んじるとかなんとかそういう話がマネージメント論に登場してくると思います。ですが私に言わせれば「重んじる」なんてのはチャンチャラおかしい話で、まるでマネージャーの言うことに全てメンバーが同調して従い、たまに自主性のあることを言うからそれを聞いてあげるんだ、みたいなノリに聞こえます。</p><p>大間違いで、実際にはチームメンバーには自主性しかありません。すなわち、基本的には、自分のやりたいことを、やりたい通りにしかしません。マネージャーの話なんて聞いてくれるわけがない、というのが当然かつ自然な状態です。これは自虐でも何でも無く事実です。嘘だと思うなら今すぐ<a href="https://ja.wikipedia.org/wiki/League_of_Legends" target="_blank">League of Legend</a>や<a href="https://ja.wikipedia.org/wiki/VALORANT" target="_blank">Valorant</a>といったチーム型対戦ゲームをインストールして、野良のランクマッチに入って、周りのみんなにああしろこうしろと命令してみては如何でしょうか、誰も話を聞いてくれませんので。</p><p>現実の仕事の場でこのような間違ったことがまかり通るのは、マネージャーという肩書があるから、上司という肩書があるから、または彼らが金をもらって仕事をしているからであって、それらの要素がなければ命令という概念は一切成り立ちません。</p><p>したがってまず私がチームで仕事をする際に考えるのは、大変逆説的なのですが、全て自力でなんとかするということです。なぜならチームメンバーは自分のやりたいことをやりたいようにしかしないのであって、私が何をしたいか、チームが何をしなければならないか、案件のゴールが何でそのためには何を達成しなければならないかなんて知ったこっちゃないというのが全く自然な状態であるためです。</p><p>しかしながら誰が考えても分かる通り、これでは仕事になりません。私一人でできることは何もありません。技術力も若いのに劣りますし、作業量も作業速度も劣ります。したがって目的を達成するには、勝利するためには、メンバーに協力してもらわなければならないのです。</p><p>そこで、先程書いたリスペクトの話が大事になってきます。まずはメンバーにあなた自身が彼らにとって有益で使える存在であるということを認識してもらわなければなりません。彼ら自身とて目的を持ってチームに存在しています、それはチームに与えられたゴールを達成したい、と少しぐらいは思っているからそのチームに居るわけです。ゲームなら勝ちたいであるし、仕事ならプロジェクトを成功させたいでも、コード書きたいでもなんでもいいです、とにかくそういうことです。その彼ら自身の目的をマネージャーたる者が手助けすることができる、という認識を持ってもらうところからが大事になります。</p><p><br /></p><h3 style="text-align: left;">メンバーに自主性の範囲で何ができるのかを把握する</h3><p>再びLoLの例になって恐縮ですが、例えばチームメンバーといっても彼ら自身の能力は一人一人異なってきます。例えばボトムレーンでラストヒットを取る程度なら何一つ指図しなくても自力でできるが、ジャングラーがGankに来ているのを自力で把握できないとか、敵のローテーションタイミングを把握できてないとか、正面のエズ君のスキルショットを回避できないとか。そのようなプレイヤーを完全に放任してボトムレーンを任せたらどうなるでしょうか?言うまでもありませんがズタボロに負けますし、結果対面のエズ君がクソ育ってクソ無双して我々はボッコボコにされて敗北して目的を達成できず最悪な気分になること間違いなしでしょう。</p><p>このような与えられた仕事を達成するのに必要なスキルを自主性を持って発揮できないようなケースの場合は、自主性を持って達成できる範囲のみ、すなわちラストヒットを取らせるところまでは完全に放任し、Gank警告は全てこちらが指示、逃げ方も全て指示、スキルショットは回避してくれないから代わりに全部自らが受けて盾になる(その結果自分が最悪死ぬかも知れないがそれは知らん)、というようなケアが必要になってきます。</p><p>逆に運良く自らのチームメンバーの中にブロンズ5のフリをしたプラチナダイヤプレイヤーが混じっており、彼らは自らの判断で一人で敵ジャングルを荒らし回ったり、Gankに行ったり、ドラゴンを支配したり、視界を取ったりといった極めて高度な判断と実行をこなすことができる場合もあります。またそこまで有能でなかったとしても、集団戦でUltを3人にぶち当てることに関しては右に出る者がない、みたいな一芸に秀でた人もいるかもしれません。</p><p>そのような幸運なケースの場合は彼らの自主性を重んじ、基本的に彼らの方針をリスペクトして受け入れ従ったほうが良いです。たとえそれが自らの経験と食い違ったり、明らかに間違っていそうだったとしても、まず彼らに自主性を発揮してもらうべきです。その代わり、彼らのために視界を取ってやるとか、ヒールを使ってやるとか、そういうサポートに徹してあげ、そして万一彼らが失敗しても非難するようなことをしてはなりません。そうすれば彼らからしてみれば、「あいつは俺のやりたいことをやりたいように任せてくれるし、面倒事は全部引き受けてくれるし、一緒にチームを組んでいてやりやすいな」と思ってもらえるわけです。こういう状態になれば自らがミスした場合に「全て任せてもらったのに失敗してしまったから申し訳ないな」「たしかにさっきは俺のミスだった」「次はアイツの話を聞いてやるか」という具合に心を開いてもらいやすい。</p><p>もちろん中には全て他人のせいにするクズもいますが、基本的にそのような人間と一緒に何かゴールを達成するのは不可能ですので、自己反省ができない人間に自主性を与える必要はないし、もっと言うなら同じチームに置いておく必要もないでしょう。コマとして切り捨てるのが良いと思います。先述のスタンスを採用していれば自然とそのようなクズが炙り出されますから、その点でも一石二鳥と言えます。ですから大事なことは、自分が見てアイツは出来る!と思ったチームメイトに関して言えば、全て任せて彼らの言うことを受け入れ、自分は支援してあげる、というスタンスをとることです。まず自分が相手をリスペクトするってことですね。そうすれば彼らから自分へのリスペクトも得られてお互いが得をすることになります。</p><p><br /></p><h2 style="text-align: left;">チームの士気の話</h2><p>先程チームの士気を高い状態で維持するのが極めて大事だと書きましたが、では具体的にどうすればよいのだろうか、という話をちょっと残しておきます。</p><p>私はチーム運営にはノリすなわち本能的な要素と、理性的で計画的な要素の両方が絶対に必要だと思っています。なぜならチームメンバーは人間であり、人間である以上本能的な好みを見逃すことはできないからです。しかしながらチームメンバーは目的を達成するために集まっているものですから、目的を達成するためには理論と計画が絶対に不可欠です。ノリだけでは何も達成できない大学のサークルになってしまいますし、理論だけでは士気の維持ができなくなるでしょう。</p><p>良い例えとして<a href="https://en.wikipedia.org/wiki/RimWorld" target="_blank">Rimworld</a>の例を挙げたいと思います。Rimworldの世界において最も儲かる商売は何でしょうか、といいますと、臓器売買です。なにせ何もしないでも海賊やならず者、隣の蛮族どもが自分たちの街に武器持って大挙して押し寄せてくるわけですから、それを撃退し、息のあるものを牢屋に閉じ込めて、一人ずつ眼球・両手両足・胃腸・肝臓・腎臓・肺を取り除いてから心臓まで取って殺し、生皮を剥いで衣類にし、肉は家畜の餌に混ぜるなりすりつぶしてペーストにしてしまえばいいのです。材料代はタダ、墓場を掘る手間すら省けますし、臓器は非常にいい値段で売れます。人体解剖は医者のスキル上げにも役立ちます。これが理論的にはどう考えても最適解でしょう。ですがこのようなことをすると住人たちに凄まじい勢いで「捕虜を殺した」「捕虜の臓器を売り物にした」「人間を解体した」「人皮の服を着た」「人肉を食べた」といった感情デバフがついてしまい、あっというまに住人たちは精神崩壊を起こして街はめちゃくちゃになってしまいます。すなわち、理論的に最適だったとしても、人間の本能が耐えられないということを表していると思います。ちなみにRimworldにおいてはサイコパスという特性を持った人間がおり、サイコパスの人間であれば捕虜をどれだけ殺そうがバラバラにしようが臓器を取り除こうが一切の感情デバフを受けることがありません。すなわち現実世界のチームビルディングにおいてもサイコパスだけでチームを構成すればここで私が申しているような本能と理論のバランスを取るなんてことは不要でただひたすら理論最適解を追い求めることができるかと思いますが、現実味がないためここでは却下させていただきます。</p><p>さてここまで書いて何が申し上げたいかということなのですが、すなわちチームを運営するマネージャーたる者は、本能的な好みと理性的な計画の両方をチームメンバーに提供できなければならないということです。そうしなければチームの士気を上手く維持することはできないでしょう。理性的な計画の方に関してはまぁ私なんかより技術力があったりマネジメント論やチームビルディングをきちんと学んでいる皆様方のほうが得意でしょうから、それは置いておいて、ここではいくつか本能的な好みの話をしたいと思います。</p><p><br /></p><h3 style="text-align: left;">別にノリを高めるために飲み会をする必要はない</h3><p>まず最初に申し上げたいことです酒が嫌いな人も居ます。時間を拘束されるのが嫌な人も居ます。飯を食うことに執着がない人も居ます。完全に昭和の考えです。忘れましょう。そもそも私自身が体育会系のノリは嫌いですし、エンジニアでそういうノリが好きな人は少ないと私は思っていますから。もちろん、そういうのが好きな人だけが集まっているチームなら、思う存分やればよろしい。</p><p><br /></p><h3 style="text-align: left;">チームメンバーに安心感を提供してあげる</h3><p>私が推奨するのはこういう方法です。「安心感」というのは人間の本能の極めて根深いところに存在するものです。したがってメンバーは安心を求めます。こいつに任せておけば安心、こいつが間に立ってくれれば面倒な仕様は俺のところに降りてこないから安心、こいつがレビューしてくれればトラブルが起きない、こいつが起てるアイディアはだいたいうまくいくから楽ちん、そういったものです。</p><p><br /></p><h3 style="text-align: left;">安心感と自主性の二律背反</h3><p>人間は「考えること」「決定すること」に苦痛を覚えるようにできています。すなわち思考を放棄し、誰かに最適解を決めてもらえることができれば、これ以上無いほどの安心感と幸福に包まれるようになっているのです。</p><p>何を馬鹿げたことをとんでもない!と怒られるのは当然ですが、ですが考えてみてください、これまでにただの一度も今日の晩御飯を何にするか悩んだことはないでしょうか?転職先をどこにするか悩んだことは?iPhone 12を買うかiPhone 12 Proにするか悩んだことは?必ずあるはずです。基本的に、決定というのは、苦痛なのです。したがってこれを取り除いてあげるというのは極めてメンバーの士気を高めます。</p><p>しかしまるでそう言うと洗脳みたいに思われるかも知れません。実際そのとおりで、そもそも我々人間は考えるからこそ価値を生み出せるのであり、考えるからこそ成長するのです。全て決定して決めてしまうなどというのはありえないことですし、そもそも相手がそれを受け入れてくれるとも考えられません。</p><p>すなわち、メンバーに進んで考えてもらうべきものは、設計やアーキテクチャなどといった成果・成長に結びつく挑戦的にで面白い分野であり、逆に考えさせないほうがよいものは、「自転車置き場の屋根の色」的などうでもいい些細な問題、ないし正解のない決定などがあるでしょう。明日のミーティングの開催時間なんて時間帯が抑えられれば15時開始でも16時開始でも問題ありませんからマネージャーがさっさと決めて取ってしまえばいいのです。どっちがいい?なんて聞くだけ時間と思考の無駄です。それとかTrelloボードのタグの色なんてどっちが赤でどっちが黄色だろうが知ったこっちゃないでしょう。そういうのもちゃっちゃとあなたのセンスで決めてしまってください。苦手ならそういうのが好きな人に頼んでもいいです。</p><p>同様に、メンバーに対してお願いをするときは相手に考える余地や推測する余地を与えないぐらい些細に至るまで完璧に支持することを目指してください。間違っても主語や目的語を省略して話さないように。メンバーがそこを考えなければならないのは苦痛でしかありませんし、その結果マネージャー側の想定する結果と異なる解釈をしては悲劇です。メンバーの苦痛を取り除くための努力を惜しんではなりません。</p><p><br /></p><h3 style="text-align: left;">アットホーム感とか人間味を出す</h3><p>Trelloボードのタグの色の話が出たのでついでに書いてしまいますが、例えばTrelloボードの背景写真に何を選ぶかであなた自身のセンスや個性、人間味といった本能に訴えかけることができます。毎週毎週きれいな自然の写真を探してきてもいいですし、今週末行った観光地の写真でもいいですし、そういう一見どうでもいいけど人間味を感じる要素を大事にするというのが一つ有効なのではないかと思っています。私どものチームの例をあげますと、毎週のスプリントに名前をつけてみたりとか、スプリントレトロスペクティブの際にスプリント名に即したトリビアを用意するとか、そういうことをやっています。例えば11月16日~11月20日のスプリント名は「ウィリアム・テル」でした。11月18日が息子の頭のりんごを射抜いた日だから、とかそういう理由なのですが、そこから転じてその週のトリビアは<a href="https://ja.wikipedia.org/wiki/%E3%82%A6%E3%82%A3%E3%83%AA%E3%82%A2%E3%83%A0%E3%83%BB%E3%83%86%E3%83%AB%E5%BA%8F%E6%9B%B2" target="_blank">運動会のときに流れる曲はウィリアム・テル序曲と言う</a>、とか。私は知りませんでした。ちょっと面白いですよね。こういう人間味のある活動というのが大事なんじゃないのかなと。人間嫌いな私が言うのもどうかと思いますが。</p><p><br /></p><h2 style="text-align: left;">会議での細かいテクニックあれこれの話</h2><p style="text-align: left;">ここでは私が普段から気をつけている細かいテクの話をつらつら書いておこうかと思います。全て可能な限り明日からすぐに使えるテクとしたつもりでございます。</p><div><br /></div><h3 style="text-align: left;">会議のゴールは何か</h3><p style="text-align: left;">マネージメントというゲームのルールとほぼ一致しますが、会議の場合は以下の優先順位のとおりに実現させます。</p><div><ol style="text-align: left;"><li>会議自体のゴールを達成する。すなわち、何かを決定するだとか、結論を出すだとか、具体的なアクションを生み出す。失敗するとチームの士気が下がり、場合によってはあなたへのリスペクトが減ります。</li><li>士気を最大に稼ぐ。士気の低下は即ゲームオーバーです。絶対に避けてください。</li><li>リスペクトを可能な限り稼ぐ。最悪減らないようにする。</li><li>発言力の消費を最小にする、できればプラスにする。まぁ、発言してゴールを導かなければならないので、多かれ少なかれ発言力は使います。プラスになったら御の字ですが、他のところで貯めましょう。</li></ol></div><p style="text-align: left;"><br /></p><h3 style="text-align: left;">激流に身を任せ同化する</h3><p style="text-align: left;">要するに「空気を読め」「流れを読め」というやつです。ほら来ましたよ、全エンジニアがクソ喰らえと思っている単語。クウキヲヨメ。私だって大嫌いですよ、そりゃ。しかし、事実です。空気を読むのは日本人だけだとか思っているあなた、それは嘘です。韓国人も中国人もベトナム人もアメリカ人もスペイン人もドイツ人もブラジル人も全世界共通で彼らは空気を読みます。ただし程度が違う、性質が違うだけです。残念ながら人と人が話を持つ上で空気を読む、流れを読むというのは極めて大事な要素と言わざるを得ません。<br />とはいえ実は簡単です。単にこう覚えておきましょう、「自分より発言したがる人間がいる場合は聞き役・守勢に周り、補助を行い、一人が独壇場を作らないように調整を行う」「自分より発言したがる人間が居ない場合は自分が喋り役・攻勢に周り、議論の流れを作り、たたき台を作って周りから発言を引き出す」、これだけです。柔道とかと同じですね。相手が前に出たら下がり、後ろに下がったら前に出る。そうすればお互いがぶつかることがなく、無駄に衝突することもなければ、お互いから何の発言も出なくなって手詰まりになることもありません。基本的にはこのように現在自分が攻勢に出て引っ張るべきか、守勢に入って気を伺うべきか、だけを判断すれば良いと思います。お互いの意見が意味もなく衝突すると敵対心が生まれる危険性があり、何も意見が出ないとチーム全体の士気が劇的に下がります。どちらも避けなければなりません。</p><p><br /></p><h3 style="text-align: left;">流れを止めない</h3><p>まずは自らが喋り役・攻勢に回る際の細かいテクからいくつかご紹介します。</p><p>会議において最も避けたいのは沈黙です。全員が考え込んでいる、というのは正しい状態にも思えますが、大抵の場合は考え込んでいるフリをしているだけで全員が混乱しているだけか、発言したいのだがためらっている状態かのどちらかです。この状態は時間の無駄であり、士気の低下を招きます。自らが喋り役に回っている際にはこの自体を絶対に避ける必要があり、逆に他人が喋り役に回っている際に沈黙が訪れたらこのときこそが発言のチャンスです。</p><p>私個人的には、5秒間の沈黙が発生したら必ず何か発言します。そのときによく使うテクニックが以下のものになります。</p><p><br /></p><h3 style="text-align: left;">自ら殴られにいく、自らがたたき台となる</h3><p>一番有用で且つ私がよく使うのがこれです。どんなデタラメなアイディアでも構わないから思いつきを出します。大事なのは、周りの誰かに「そりゃダメだろ」と突っ込んでもらうことです。最悪、デタラメなアイディアに決まってしまうよりも、なにも決まらないよりは圧倒的にマシです。その場合、デタラメなアイディアがうまくいくこともありますし、それが失敗したなら失敗した原因を掘り下げることで、次回は成功に近づくことができます。</p><p>場合によっては「わざと」メチャクチャなアイディアを出すこともあります。これはツッコミどころ満載のアイディアを思いつきで出したかのように振る舞いつつ、他のメンバーが釣られて「いやそれは間違ってる」「それじゃダメじゃないですか」と発言してくれることを期待してのことです。もちろんそこで、釣り宣言をしてやーい釣られたバーカバーカとか言い出さないように。相手の指摘が適切であることをすぐに認め、自らのアイディアがいまいちだったことを認め、指摘に感謝し(相手の士気が高まり次回以降の発言が活発になります)、可能であればどうしてその指摘が適切だったかを掘り下げるのが得策です。</p><p>「確かに、Aさんの仰るとおり、このアイディアだとXXXがダメで上手くいかないね。まてよ、であれば、XXXが満たされればいいわけで、それならXXXXXXならどうかな?」</p><p>などと掘り下げのたたき台にできれば完璧でしょう。</p><p><br /></p><h3 style="text-align: left;">現状を振り返る</h3><p>とはいえ私は一休さんではないので、毎回毎回デタラメなアイディアが思いつくとは限りません。その場合に有用なのはこれです。現状を振り返ってみます。可能な限り細かく砕いて振り返るのが大事です。</p><p>「えーと待てよ、今困ってるのは、XXXが、XXXXになっちゃってるからXXXXXができないんだけど、それを解決するためには隣のCチームがXXXXを直してくれないとどうしようもないんだよな。でもCチームはスケジュール的に来月まで動けないとか言ってる。どうしよう。」</p><p>これは何も問題を解決していませんが、全員に現状のヒントを与えることができますので、そこから誰かの発言が出てくることを期待することもできますし、現状を振り返ることで自らがネタを思いつくことも期待できます。</p><p>「ああ、それならCチームのタスクのXXを我々が引き取って変わりにXXXXを進めてもらえばいいんじゃないかな?そういうバーターはやったことないけど、試して見る価値あるかも」</p><p><br /></p><h3 style="text-align: left;">質問を投げかける</h3><p>これは問題の解決策を誰かが持っていることが期待できるが発言が出てこない場合に使う作戦です。質問を投げます。</p><p>「Yさん、XXXXについては以前あたりのコード調べてた気がするけど、そこなにか知らない?」</p><p>質問に対して直接回答が得られれば良いですし、得られなくても追加の質問が行える場合もあります。</p><p>「わからないかー、そこ調べてもらったときに誰がそういう仕様にしたか見てもらった気がするけど、誰に聞いたらその辺の経緯分かる?」</p><p><br /></p><h3 style="text-align: left;">正しいか正しくないかではなく好きか嫌いかを問う</h3><p>これはある事項を決定しなければならないのだが、複数の選択肢の間でみんなが迷っているときに非常に有効な手だと思っています。つまり、こういうことです。</p><p>「Aさんはプラン1とプラン2どっちが好きかな?別に良いとか悪いとかじゃなくて、自分が好きだなーって思う方を気軽に教えてほしいんだけど」</p><p>「これだけいろいろ考えたけど、どっちもメリデメあって一長一短あるし、どちらに決めても後悔はないと思うので、それだったらよりみんなが好きでやりたいと思うプランを実施したほうがいいからね」</p><p>この作戦を取るときに最も大事なのは先ず隗より始めよ、まず自らが絶対に最初に言い出す、という点です。チームメンバーに意見を押し付けてはなりません。</p><p>「俺はプラン2のほうが好きかなーなんとなくだけど。1も嫌いじゃないんだけどどっちやったほうが楽しそうかなって言ったら2かな!」</p><p>この作戦を取るときには理由なんていりません。理由が付けられないから好きか嫌いかで選ぶんですから、理由をつけず、頭悪く好きか嫌いかでモノを語ってください。これで決まることもあるし、逆にちょっと待ったコールが上がることもあります。そうしたらシメたもので、ちょっと待ったと声を上げてくれた人はなにか違和感があるから好き嫌いで選びたくないと思っているのですから、そこを掘り下げれば解決策がでてくるかも知れません。</p><p><br /></p><h3 style="text-align: left;">最後は自分がやる</h3><p>あらゆる手段を用いて流れが止まらないように尽力したにもかかわらず上手く議論が進まなかったり、解決策が導かれなかった場合の奥の手はこれしかありません。自分でやりましょう。自分がその箇所を触ったことがあるとか無いとか、使ったことがある言語ではないとか、そういうのは関係ありません。誰もやらない、できないなら、自分が突撃するより他に手がないのです。ひょっとしたらチームメンバーの中には解決策を持っているが、発言したら自らがタスクを引き取らなくてはならなくなるからあえて答えを出さないという人もいるかも知れません。それに対して非難するのは誰でもできますし理屈では正しい事かもしれませんが、それによってメンバーの士気がダダ下がりになってしまうのであればマネージメントのゴールの達成という意味では大失敗です。上手く成功すれば自分のリスペクトも増えますから、決して悪い手ではありません。</p><p>こういう最後の手段が使えるように、普段の自らのタスクはやや少なめに拾っておくことをオススメします。やはりマネージメントが最も活躍しなければならない場面というのは問題が発生したときですから。</p><p><br /></p><h3 style="text-align: left;">相手の発言を細かく砕いて言い直す</h3><p>次に自らが聞きに回っているとき、すなわち守勢にいるときに明日から使えるテクをいくつか並べてみます。</p><p>会議のときなど、時々誰かが余りにも省略しすぎた発言をすることがないでしょうか?これは非常にありがちなのですが、全ての人間がしゃべるのが得意というわけではありませんので、自分には分かっているけど人には分かっていない前提条件だとか、コンテキストだとか、そういうのをついつい省略して喋ってしまうことがあるのです。そのようなときに、あえて細かく質問を投げかける形で聞き直すようなテクがあります。</p><p>例えばこういう感じです・・・「えっとごめん、今XXXXって言ってたのは、要するにXXXがXXXXでXXXXXだからXXXXだって話ってことであってる?念の為に確認したいんだけど」</p><p>これは特に発言者の発言が不用意で分かりづらかったときほど効果を発揮します。自らが分からなかったので教えて欲しい、という立場を取ることで、発言者を責めたり悪い気持ちにさせることもなく、周りのチームメンバーとの認識のズレを正すという効果を発揮しつつ、リスペクトも得られるからです。さらにここから流れを自分側に引き寄せて自らが追加で意見を述べることもできます。余りにも一人の人物が喋りすぎて場が独壇場になりそうな場合の有効な切り返し策となりえます。</p><p>もちろんあまりにも頻繁に聞き直したら嫌われますし、あまりにも見当違いな聞き直しをしたらただの馬鹿だと思われますのでくれぐれもご注意を。</p><p><br /></p><h3 style="text-align: left;">「他人を褒める時は大きな声で、けなす時はより大きな声で」</h3><p>言わずと知れた<a href="https://w.atwiki.jp/aniwotawiki/pages/29475.html" target="_blank">某銀河帝国の提督様</a>の家訓でありますが、少なくとも私にとってはこれほどしっくり来る人生哲学はありません。特に前半が極めて重要だと考えております。すなわち、人を褒めるときは大きな声で、すぐにその場で、どうして褒められるべきか理由を添えて、己の本心から褒める、これが最も大事です。正しい賞賛は味方の士気を大いに高め、モチベーションを作り、本人の成長と自立を促すことにつながります。</p><p>これも守勢のときに使える有効な作戦の一つです。</p><p>逆に後半部分ですが、これは敵に対して用いるべき方法で、チームメンバーの問題点を指摘する際には後から、人とは別の場所で、改善のための具体的な方法を添えて、私心を加えず指摘する、つまり褒めるときとはまるで逆のことをやるのが大事であると思います。</p><p><br /></p><h3 style="text-align: left;">他人の表情を見る</h3><p>自らが発言の主導権を持つときも、他人が主導権を持っていて聞き役に回るときも、どちらでも大事なことは、相手の表情を見ることです。今このご時世ですから会議はZoomでリモートでというのが当たり前だと思いますが、絶対に顔を出すようにしてください。少なくとも私は常に顔出し状態にしています。このほうが自らはきちんと会議に参加しているのだというリスペクトを得やすいです。裏で内職をしていないという証明みたいなものですね。それに自らが発言の主導権を持って進めているときに誰か他に発言したそうな状態の人を発見することもできますし、他の人が主導権を持って進めているときに相手に対して表情だけでも返事することができれば相手は進めやすく感じてくれるはずです。</p><p>これは会議に置けるマナーだとか言うつもりはないのですが、単純にそのような行動をするだけで他人から容易に一目置いてもらえるのに、カメラをOFFにしておく理由はないです。私に言わせればそれだけで大量のリスペクトを無駄にしています。</p><p>さらに他人が裏で内職をしている徴候をいち早く察することで、今の会議の状態を改める必要があることに素早く気づくことができます。内職をしている参加者は自らが参加している必要がないと思っているから裏で内職しているか、または他により大事なタスクがあったり締切に間に合わないから内職しているわけで、いずれの場合でも何かが上手く行っていない徴候です。会議の量を減らすとか、参加しなくていいものに呼ばないとか、タスクの割り振りが間違ってるとか、そういうマネージメントに必要な情報を得るのに相手の表情を見るのはとても適しています。</p><p><br /></p><h2 style="text-align: left;">最後に一番大事なこと</h2><p>チームリーダーだろうがマネージャーだろうが社長だろうがなんだろうが構いませんが、マネージメントのお仕事をしている人にとって最も大事なことは、先程までに述べた内容と矛盾しますが、実は自分の身を守ることではないかと考えています。要するに、もらっていない給料の分の仕事まで責任をとってはなりません。マネージャーは魔法使いでも神様でもチート能力を持った異世界転生主人公でもありませんので、無理な仕事は無理です。責任感があるのは素晴らしいことですが、できないことをできると言い張るのもそのためにチームを付き合わせるのも全てそれ以上に無責任なことです。とっとと逃げましょう。退職しましょう。あなた一人がチームから消えたところで人類全体の幸福にはほぼ何の影響もありません。むしろチームメンバーをそのような終わっているプロジェクトから守ってあげるために、さっさと退職させてあげたり、別の仕事を紹介してあげるほうが、人類のためになるのです。</p><p><br /></p><p><br /></p><p>長くなりましたが以上になります。何かしらの参考になれば幸いです。</p><p><br /></p><p>明日はtokibitoが担当です。お楽しみに。</p><p><br /></p>akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-85191611951850874142019-12-07T00:00:00.000+09:002019-12-07T17:11:16.772+09:00最近のクルマの話こんにちわ、<a href="https://adventar.org/calendars/3896">毎年恒例 pyspa Advent Calendar も今年は2019年となりました。</a>12月7日担当のakisuteです。明日の担当は <a href="https://adventar.org/calendars/3896#list-2019-12-07">@rokujyouhitoma</a> です。<br />
<br />
さて皆さん突然ですが、クルマは持ってますでしょうか?<br />
<br />
持ってない?当たり前ですね。馬鹿みたいに高い税金と車両代金と高速道路通行料を搾取されるだけで何一つメリットありませんから。<br />
<br />
それでは皆さん、スマホはお持ちでしょうか?<br />
<br />
持ってる?馬鹿にすんな?当たり前ですね。今どきスマホ無いと生活が困難ですね。10年ほど前はどうやって電車の車内で時間を潰していたのかすら、もはや思い出せそうもありません。<br />
<br />
そういうわけで、私のブログをわざわざ見に来てくださっている方はIT業界、それもスマホアプリ関連の業界の方が大多数でしょうから、スマホについてはバリバリ詳しいけど、クルマなんて持ってないから何一つ知らないし興味もないよ、という方が多数派を占めてらっしゃるのではないでしょうか。本日はそういう多数派の方向けのお話をさせていただきたいと思います。<br />
<br />
さて、皆さんがクルマに何の興味も知識もない前提でお話を進めさせていただこうと思うのですが、それでも平成以降に製造されたクルマにはカーナビとかいう地図とか表示するディスプレイが付いていたり、スマホをBluetoothで接続したりしてカーオーディオを聞いたりできるということぐらいはご存じの方が多い、と私は勝手に信じております。ところが今や元号も令和となりまして、どこもかしこもインフォメーションテクニック的なヤツが幅を利かせる世の中になっておりますゆえ、当然クルマに搭載されているカーナビですとかカーオーディオ的なやつもご多分に漏れず高機能化しています。おまけに最近は自動運転 (=カーナビの現在位置情報や道路情報を車両側が使用したい) ですとか、クルマの走行特性モードの設定 (=車両側の走行特性をUIから操作する必要がある) ですとか、クルマのハンドルについているスイッチからカーオーディオを直接操作したい (=車両側の入力をカーオーディオと連動させなければならない) ですとか、そういう昭和の時代では考えられないような高度な統合処理が必要な機能の搭載がクルマにとってもはや必須となっておりまして、昔のように後付で買ってきたカーナビやカーオーディオに配線すればOK、というわけにはいかなくなってきました。<br />
<br />
そこで、ここ2年ほど以内に発売された新車の真ん中に鎮座するディスプレイはカーナビではなく<a href="https://ja.wikipedia.org/wiki/%E3%83%9E%E3%83%AB%E3%83%81%E3%83%95%E3%82%A1%E3%83%B3%E3%82%AF%E3%82%B7%E3%83%A7%E3%83%B3%E3%83%87%E3%82%A3%E3%82%B9%E3%83%97%E3%83%AC%E3%82%A4">マルチファンクションディスプレイ</a>とかディスプレイオーディオとか呼ばれるようになってきておりまして、タッチパネルを装備し、カーナビ、オーディオ、エアコン、車両の設定などを直接タッチ操作で行う高度なシステムに変貌しています。<br />
<br />
それだけには留まらず、今年発売された新車に至っては4G接続用のSIMとアンテナが装備されており自立通信が可能で、電話は当然として緊急時の連絡サービスや、煽り運転されたときにボタンひとつで自動的に通報してくれるサービスもあり、カーナビの地図情報の自動更新やシステムアップデートもこなすことができます。なぜクルマにこのようなものが必要かといいますとこれまた将来的に重要になる自動運転が絡んできまして、最新の地図が常に使用されていないと自動運転は危険だからだとか、万一なにか問題があったときに自動運転中の車両を遠隔監視できていないと駄目だとかそういう理由があります。そこで先んじてこのような通信能力をクルマに持たせているというわけです (※おそらくは裏で匿名運転データの収集も行っているのではないかと推測していますが、そのへんは不明です) 。<br />
<br />
さらには昨今のスマホ社会を反映し、各社自動車メーカーが用意したスマホアプリと車両が連動して、スマホアプリから現在の自分のクルマの位置を調べるだの、走行距離とガソリンの残量を調べるだの、カーナビに目的地を送信するだの、果ては走行前に社外からエアコンをONにしたりクルマのドアロックをスマホアプリから開けてしまうだのといったことまで可能になっているのです。<br />
<br />
どうでしょう、まるでスマホみたいですね。<br />
そのとおり、最近のクルマにはちゃんとしたOSがいます。<br />
<br />
このクルマのOSは現在のところ基本的には各社自動車メーカーが内製で作っている (かまたは基本システムだけ買ってきて各社勝手にカスタマイズして使っている) ものが多いようで、iOSとAndroid以外のOSが事実上絶滅したスマホの世界とは異なり、どの会社からクルマを買うかによって大いに出来栄えと機能に差がある状態です。<br />
<br />
中にはスマホ世代の我々にはひと目見ただけで開発者をコンクリ詰めにして東京湾に投げ捨ててやりたくなるようなひどい代物も多数存在します。それどころかタッチしてから0.5秒遅れで反応するUIなどといった、初代iPhoneどころかAndroid 1.5 Cupcake世代の産業廃棄物スマホにすら劣るレベルの実装すら、世の中のクルマに存在します。<br />
<br />
ヤバいですね。何処のドイツでしょうか、そんなひどいブツを作るのは。実際に見てみたいと思いませんか?<br />
<br />
・・・ここで、ちょっと話は変わるのですが。<br />
<br />
メルセデスというクルマのブランドがあります。<br />
<br />
あまりクルマに詳しくない人でも、ベンツといえば分かるでしょうか。そう、ドイツの高級車メーカーです。最近はメルセデスという名前で名乗っていますので、メルセデスと呼ぶことにします。<br />
<br />
でそのメルセデスといえば、皆さん誰でもすぐに高級な外車で、金持ちが乗ってそうなイメージを浮かべると思います。当然、見た目も高級だし、内装も高級だし・・・<br />
<br />
・・・車載OSの動作だって高級で高品質を期待します。当たり前ですよね。下手すると1000万円とか払うクルマなわけですから、iPhone 11 Proみたいにきれいな見た目で高速にちゃんと動作することを期待したいじゃないですか。iPhoneなんて10万円かそこらしかしないんですから。<br />
<br />
じゃあ見てみましょうか。メルセデス。でもなぜだか知らないんですが日本仕様・右ハンドル・日本語表示のMBUX (Mercedes Benz User Experience) のUI、ネット上にほとんど落ちてないんですよね。しょうがないから実車の写真を撮ってきました。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://robotstart.info/wp-content/uploads/2018/02/mbux.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="400" data-original-width="735" height="174" src="https://robotstart.info/wp-content/uploads/2018/02/mbux.png" width="320" /></a></div>
ネットから拾ってこれた左ハンドルでの画像ですが、横一列につながった超ワイドなディスプレイ、一見なかなかカッコいいですね。高級感もあるし、これは期待できるのでは・・・<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjewuN72_ku808xyH5gVK7AofMj3vcsG-CjUi46Ru2QNMN7Qhyphenhyphen-SFUg9DEdjlodVFhtObx1JmJGK1OW_B0P6lzdChNbjLOCcCkiws80rcIMubD0NgXPLDHtNAzTAT8ltr5SWcII7DE8VE4/s1600/IMG_2866.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="1200" data-original-width="1600" height="240" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjewuN72_ku808xyH5gVK7AofMj3vcsG-CjUi46Ru2QNMN7Qhyphenhyphen-SFUg9DEdjlodVFhtObx1JmJGK1OW_B0P6lzdChNbjLOCcCkiws80rcIMubD0NgXPLDHtNAzTAT8ltr5SWcII7DE8VE4/s320/IMG_2866.jpg" width="320" /></a></div>
って思った?残念!半角カナちゃんです!<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEg9Nis3YY6CD4QgTGQKlaxfIOZipF92UTr4WIJarRA07YdwvPwKBfnZZ07_W0tcvjb1JZ5J6VQrsdwepbo__9fiKgFmUTi4CdasNE_Wr0PeX3LvHzv5Qt4MkYyxO5nz3PNa1wJC-9yoOH4/s1600/IMG_2874.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="1200" data-original-width="1600" height="240" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEg9Nis3YY6CD4QgTGQKlaxfIOZipF92UTr4WIJarRA07YdwvPwKBfnZZ07_W0tcvjb1JZ5J6VQrsdwepbo__9fiKgFmUTi4CdasNE_Wr0PeX3LvHzv5Qt4MkYyxO5nz3PNa1wJC-9yoOH4/s320/IMG_2874.jpg" width="320" /></a></div>
見ての通り、細部に至るまで半角カナです!<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhzSKQU5Wpo05rrI35MFBX-x25QiOQ3bhzUNuWPDq1jh6XGv9OizOiak7t5aIypGz_p5fbD_IGZmGiQViZOBSCCvRLHbi6DdfmZwopncI_vXavLCHaSc5esuOWyxhLw09I4R9xQwT3RISE/s1600/IMG_2877.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="1200" data-original-width="1600" height="240" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhzSKQU5Wpo05rrI35MFBX-x25QiOQ3bhzUNuWPDq1jh6XGv9OizOiak7t5aIypGz_p5fbD_IGZmGiQViZOBSCCvRLHbi6DdfmZwopncI_vXavLCHaSc5esuOWyxhLw09I4R9xQwT3RISE/s320/IMG_2877.jpg" width="320" /></a></div>
こっちは半角カナじゃありませんがガソリンスタン...になっちゃってます。<br />
<br />
幸いにしてタッチパネルの操作感度はまぁまぁ良好だし、タッチしてから1秒以上固まるみたいな絶望的な遅さは無いのですが、どう見ても初代iPhone 3Gとドッコイドッコイ程度の動作速度しかありません。そもそも走行中にカーナビを操作したくなったら、身を乗り出してこのタッチパネルを手で触れというのでしょうか?まことに素晴らしいUXだと思います。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://www.elasticfeed.com/wp-content/uploads/a572ac23bd762104748a0fd68fa08d68.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="600" data-original-width="800" height="240" src="https://www.elasticfeed.com/wp-content/uploads/a572ac23bd762104748a0fd68fa08d68.jpg" width="320" /></a></div>
引用元: <a href="https://www.elasticfeed.com/44db3e868b3b3bfe82835c65441ea3ca/">https://www.elasticfeed.com/44db3e868b3b3bfe82835c65441ea3ca/</a><br />
<br />
もちろんそのような危険なことをしないで済むように、こうして中央手元にMagic Trackpadもどきが付いているのですが、手触りはともかくこいつの反応が10年前のWindowsノートパソコンのほうがマシなのではという次元でして、自分のmacに触れた後にこいつに触ると引きちぎって車外に投げ飛ばしてやりたい気分に駆られること間違いありません。設定で感度を調整したりもできますがどう調整してもイライラするので、結局中央のタッチパネルを身を乗り出して触るほうがマシだったりします。<br />
<br />
その他、ハンズフリー操作のために音声アシスタント、いわゆるSiriもどきがついてるのですが、社内での会話中にうっかり「メルセデス」という単語を口にした瞬間「なにか御用でしょうか」と起き上がってきて大変邪魔です。名前を呼んではいけないあの人扱いみたいになっています。しかも初代のSiriより頭が悪く、基本的に何お願いしてもマトモに対応してくれないので使えません。お願いだからOK Googleと席を変わってほしくなります。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjdIDvDraWmfteDxWyyvGrTuuS6viOlSECpTMvAUabK9cHptiggh9wuyCZrxZnyOFpFbxT-STvAHLsUknuwvAGU35GvLiLBeBycokAa1lftTsKbPsK4SKb0pm-gW_evrY8r5ZhujUyyiHc/s1600/IMG_2876.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="1200" data-original-width="1600" height="240" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjdIDvDraWmfteDxWyyvGrTuuS6viOlSECpTMvAUabK9cHptiggh9wuyCZrxZnyOFpFbxT-STvAHLsUknuwvAGU35GvLiLBeBycokAa1lftTsKbPsK4SKb0pm-gW_evrY8r5ZhujUyyiHc/s320/IMG_2876.jpg" width="320" /></a></div>
<br />
<br />
極めつけはカーナビです。なんとこちら1年近く道路情報が更新されていないようで、設定画面から最新状態に更新しているにもかかわらず半年前に開通した道路が地図上に存在しません。当然カーナビはそこを迂回しようとします。全く使えません。メルセデスの人に聞いてみたところ、現在地図データを頑張って作ってるらしいです。頑張ってくださいね。<br />
<br />
ちなみにこちら、お見せしましたのはAクラスといいます一番安物のメルセデスですが、実は2000万円以上する<a href="https://carsmeet.jp/2019/11/25/131025/gallery/13/">メルセデス・マイバッハと言われる最上位車種でも全く同様のMBUXが使用されています。</a>それどころかAクラスは今年モデルチェンジされたため、下手するとそれらよりも最新のOSが搭載されています。要するに2000万円以上する超高級車でも1年前の道路しか存在しないカーナビを使わされるというわけです。まぁ、そんなクルマを買うような人は自分で運転なんかしないでしょうから、何の問題もないのかもしれませんね。<br />
<br />
じゃあ、次はメルセデスのスマホアプリを見てみましょうか。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjI6UmPoV7v2SRe2p2fjBcHRE8UXydxmWLO0qu_QwLN09u03bcLE5t7G3WAR-HO1GVWozU1OcJSv-N2UOc2YRbjsZNxKQn7lOxQggTXg0MhMTj2jdeS3i_rCIKjz6SvhvcK3rhdTMVnxhg/s1600/IMG_2857%25E3%2581%25AE%25E3%2582%25B3%25E3%2583%2592%25E3%2582%259A%25E3%2583%25BC.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjI6UmPoV7v2SRe2p2fjBcHRE8UXydxmWLO0qu_QwLN09u03bcLE5t7G3WAR-HO1GVWozU1OcJSv-N2UOc2YRbjsZNxKQn7lOxQggTXg0MhMTj2jdeS3i_rCIKjz6SvhvcK3rhdTMVnxhg/s320/IMG_2857%25E3%2581%25AE%25E3%2582%25B3%25E3%2583%2592%25E3%2582%259A%25E3%2583%25BC.jpg" width="148" /></a></div>
まぁ悪くはないですね。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh8M9rG_WUFzRlUDoP4RJTsjqN0V3PEgykNnm_OZYav_EP7Q91MSgtck_zVnlFYsd-tK1pDCLboF8Yvs6H9tCqfd7T_wNRKNgxsadpnWyojRTVPgfokZsETrWLC7pTBo8C34yxLTpPyYSI/s1600/IMG_2864.PNG" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh8M9rG_WUFzRlUDoP4RJTsjqN0V3PEgykNnm_OZYav_EP7Q91MSgtck_zVnlFYsd-tK1pDCLboF8Yvs6H9tCqfd7T_wNRKNgxsadpnWyojRTVPgfokZsETrWLC7pTBo8C34yxLTpPyYSI/s320/IMG_2864.PNG" width="148" /></a></div>
なんかWebViewっぽい匂いがしますが・・・<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjvG1wd0eugL0-_BztVrIhRxRhTfPaTRmwIyDvUx31nYNaGdqhNGjdENrleOu3qTKHbkq0D7M_8PkdzU6G7DWgHf7sAiCesaSJqTcdMu-5PNp2N_sI5KixVUZUYu89s6Z84xqqlSPZdIs4/s1600/IMG_2865.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjvG1wd0eugL0-_BztVrIhRxRhTfPaTRmwIyDvUx31nYNaGdqhNGjdENrleOu3qTKHbkq0D7M_8PkdzU6G7DWgHf7sAiCesaSJqTcdMu-5PNp2N_sI5KixVUZUYu89s6Z84xqqlSPZdIs4/s320/IMG_2865.jpg" width="148" /></a></div>
一応ネイティブアプリのようです。FOSS (Free/OpenSource Softwares)をちゃんと列挙しているのは好感が持てますね。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEg26OhhlrR1LR2p-HMEiM2R8dCcWA9TMzM1zGHhtRQPV9GVE0QpGeJvYtlIWJySKaAFsk3o0HWkkCtS1LwjlUH0t05pOIzBK6GvEArVb4vXQm40N1vAZFnDK_89rluLgtU0T8PNP7-RzLY/s1600/IMG_2859.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEg26OhhlrR1LR2p-HMEiM2R8dCcWA9TMzM1zGHhtRQPV9GVE0QpGeJvYtlIWJySKaAFsk3o0HWkkCtS1LwjlUH0t05pOIzBK6GvEArVb4vXQm40N1vAZFnDK_89rluLgtU0T8PNP7-RzLY/s320/IMG_2859.jpg" width="148" /></a></div>
一応機能は充実してますね。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhe4aKz66INg_hhzdBSU7a0dzyys4t-F_luoWqcBnvojeudWIrjQ3yahmAKCyKs-YV76H9HweVh76lVJtNfcugLH3HARPKxDdbNuGQWmZUg5bwJuovJj-wIJYVBLxzfEYXSudhPvHsRxxo/s1600/IMG_2860.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhe4aKz66INg_hhzdBSU7a0dzyys4t-F_luoWqcBnvojeudWIrjQ3yahmAKCyKs-YV76H9HweVh76lVJtNfcugLH3HARPKxDdbNuGQWmZUg5bwJuovJj-wIJYVBLxzfEYXSudhPvHsRxxo/s320/IMG_2860.jpg" width="148" /></a></div>
見ての通りクルマのドアロックをこのアプリから開けることもできます。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjQjTynsenzb6K3_i4bY-iNV0jUwnA6V0pik7DAMIBxcqTA1-2Bs3uRAMrzYMrOiFhonf89kZ-u5sSimi7QcghixM0rLGBNAuEaryWMbhiWLEScqaQIs1GUqAdvmO7YNu77QKd9T8AcQis/s1600/IMG_2863.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjQjTynsenzb6K3_i4bY-iNV0jUwnA6V0pik7DAMIBxcqTA1-2Bs3uRAMrzYMrOiFhonf89kZ-u5sSimi7QcghixM0rLGBNAuEaryWMbhiWLEScqaQIs1GUqAdvmO7YNu77QKd9T8AcQis/s320/IMG_2863.jpg" width="148" /></a></div>
カーナビ連動もありまして、この画面から目的地を検索してクルマのカーナビに飛ばすことも可能なのですが、<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgNinVH16qcsAPgjarG0vrr652f7qJ-7hmm7AEONAXYwzRx0J0k-wMXTLvWrp7-hhRk5kaF_Skw7yy1JEcz36O3tOwO2KUfFrzmVi3xxjATmOU3TRZRJqMkWMeEHR0UNeWf7adDGYjGXk4/s1600/IMG_2862.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgNinVH16qcsAPgjarG0vrr652f7qJ-7hmm7AEONAXYwzRx0J0k-wMXTLvWrp7-hhRk5kaF_Skw7yy1JEcz36O3tOwO2KUfFrzmVi3xxjATmOU3TRZRJqMkWMeEHR0UNeWf7adDGYjGXk4/s320/IMG_2862.jpg" width="148" /></a></div>
見ての通りこの程度の検索すらロクにできませんので使い物になりません。まぁよしんば目的地がクルマに送信されたところで1年前の日本地図に基づいてナビゲーションされてしまうので、どっちにしろ使い物にならないと思いますが。<br />
<br />
いかがでしたでしょうか。まぁ、車載MBUXに比べると幾ばくかマシかもしれませんが、2000万円以上払った人専用アプリなんてものはございませんので、最廉価のAクラスユーザーでも最上位のVIPユーザーでもこのアプリを使わされることになります。もうちょっと頑張れよと思わないでも無い気がします。<br />
<br />
一応、弁護させていただきますと英語版やドイツ語版は半角カナではないのでフォントはマシですし、多分ドイツにサーバがあるのでドイツで使ったほうがアプリの動きも多少はサクサクだと思いますし、ドイツ国内のカーナビは多分1年遅れで更新されるとか言うことも無いはずです。要するに日本版が飛び抜けてショボいだけかもしれません。あとクルマ自体は普通にとんでもなく良かったので、アプリとか気にしない人にとっては最高のクルマだと思います。<br />
<br />
で。<br />
また閑話休題で申し訳ないのですが。<br />
<br />
テスラというクルマのブランドがあります。<br />
<br />
最近話題になることが多い電気自動車のメーカーです。あのイーロン・マスクの会社というと我々IT業界人には通じやすいのではないでしょうか。何かやってくれそうな気がしませんか?<br />
<br />
そのテスラなんですが、こんな感じなんです。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://response.jp/imgs/thumb_h2/1440767.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="499" data-original-width="800" height="199" src="https://response.jp/imgs/thumb_h2/1440767.jpg" width="320" /></a></div>
iPad Proより巨大なタブレットのようなサムシングが鎮座しています。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://response.jp/imgs/p/HyHJUDsvwEJjLfURx1fZfoRN5UBBQkNERUZH/1444935.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="542" data-original-width="800" height="216" src="https://response.jp/imgs/p/HyHJUDsvwEJjLfURx1fZfoRN5UBBQkNERUZH/1444935.jpg" width="320" /></a></div>
引用元: <a href="https://response.jp/article/2019/08/19/325526.html">https://response.jp/article/2019/08/19/325526.html</a><br />
<br />
見てください、これ。まるでiPadのアプリにしか見えません。このテスラ モデル3の社内にはハンドル周り以外に一切の物理ボタンが存在せず、ディスプレイもダッシュボードのメーターもありません。速度計の確認も、エアコンの操作も、ハンドルとアクセルとブレーキとギアの選択とウィンカー操作以外、何もかもすべてこのiPadで行うことになります。<br />
<br />
普通にメチャクチャ綺麗です。カーナビに至ってはおそらくこれGoogle Mapsそのものです。当然、Google Mapsですから、1年前の日本地図が表示されることもなければ、<a href="https://ja.wikipedia.org/wiki/%E3%83%91%E3%83%81%E3%83%B3%E3%82%B3%E3%82%AC%E3%83%B3%E3%83%80%E3%83%A0%E9%A7%85">パチンコガンダム駅</a>が存在することもありません。完璧に最新最先端の地図が無料でいつでも使えます。素晴らしいですね。<br />
<br />
では、テスラのアプリはいかがでしょうか?<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh1X_xlB-te6ZJ8d1SLrcfKUykFoslBDebSfQS4IiHZlNe-_O18eFzvpEXCKs_z6_GiVweBJDfsVtTT6oXIOi-34jyUhL0856KRjFHW6J1pDulfixVX8jOn9CRS5BsY2cTfr1sDvG2AvVw/s1600/73195525_484637758811972_1805617561342050304_n.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh1X_xlB-te6ZJ8d1SLrcfKUykFoslBDebSfQS4IiHZlNe-_O18eFzvpEXCKs_z6_GiVweBJDfsVtTT6oXIOi-34jyUhL0856KRjFHW6J1pDulfixVX8jOn9CRS5BsY2cTfr1sDvG2AvVw/s320/73195525_484637758811972_1805617561342050304_n.jpg" width="148" /></a></div>
当然のようにネイティブです。まぁReact NativeかもしれませんしFlutterかもしれませんが。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhYfpFCtFbcudbFgoCGkWhPCXbizPp-KP64Ahgot7UiR9ogGvT5SO8UcXTm_AoACll8Zy-Fnh_qHR6cJlzj1RzQHWr7zTRK1t3x8hHDhbI08gdfRZhgvw4wn17cRy62leH9fTggrgWPIjM/s1600/74179154_2559232957500902_1426170171541684224_n.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhYfpFCtFbcudbFgoCGkWhPCXbizPp-KP64Ahgot7UiR9ogGvT5SO8UcXTm_AoACll8Zy-Fnh_qHR6cJlzj1RzQHWr7zTRK1t3x8hHDhbI08gdfRZhgvw4wn17cRy62leH9fTggrgWPIjM/s320/74179154_2559232957500902_1426170171541684224_n.jpg" width="148" /></a></div>
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEigkfS2805pUxZe0ay08bcBlDGZ079zP4QfZMbmU3LzLCo3WtY2GG8-MtHUBz6T6jr230Y_SUCE8WF4_7Nhfff2sZfMloTpo3_jngFx30MnmJjXQz5tX_0_1qraP2GcR8uqNmsDTNVWxLc/s1600/74163965_919019355136797_1397420399196635136_n.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEigkfS2805pUxZe0ay08bcBlDGZ079zP4QfZMbmU3LzLCo3WtY2GG8-MtHUBz6T6jr230Y_SUCE8WF4_7Nhfff2sZfMloTpo3_jngFx30MnmJjXQz5tX_0_1qraP2GcR8uqNmsDTNVWxLc/s320/74163965_919019355136797_1397420399196635136_n.jpg" width="148" /></a></div>
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEj61gVXbgo05DPTeptniR5jbFQyLA9gUNs-jYY8biZyGo_Wtgt6CkEaaaxy1izdQcrpM_me9R8ikjoePFgfipRQeiKu1Fr16zy19Q3noLIP-Rle7k_OrzCM6x0coUQONK343Qmu4ra10XM/s1600/74188738_1214044035650364_4135543796509179904_n.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEj61gVXbgo05DPTeptniR5jbFQyLA9gUNs-jYY8biZyGo_Wtgt6CkEaaaxy1izdQcrpM_me9R8ikjoePFgfipRQeiKu1Fr16zy19Q3noLIP-Rle7k_OrzCM6x0coUQONK343Qmu4ra10XM/s320/74188738_1214044035650364_4135543796509179904_n.jpg" width="148" /></a></div>
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgGxruDc0EGs1w5K24TSMlY7ZLIDgx1ncczO5SxwOWvdVQFJ6aaVG7oTTDKfcGS4XxzSb8U4EI0XWeVJDDOMnsh7qRRS6k6shSSzY2M1joNKwAgkToCTOpKOnsXo9PtY2pq6u6idLpwx-o/s1600/75026083_1063728357306814_3471907925495119872_n.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="320" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgGxruDc0EGs1w5K24TSMlY7ZLIDgx1ncczO5SxwOWvdVQFJ6aaVG7oTTDKfcGS4XxzSb8U4EI0XWeVJDDOMnsh7qRRS6k6shSSzY2M1joNKwAgkToCTOpKOnsXo9PtY2pq6u6idLpwx-o/s320/75026083_1063728357306814_3471907925495119872_n.jpg" width="148" /></a></div>
どうみても出来が良いです。半角カナなんて汚物は当然存在しません。<br />
<br />
これで、テスラ モデル3の車両価格は最高でも717万円、最安値ですと511万円で収まります。<br />
<br />
以上で大体私の申し上げたいことはなんとなく伝わってきたのではないかと思います。つまり、我々スマホ開発者界隈が買うべきなのはテスラってことです!さぁ、このテスラの新車を今すぐ予約してあなたも未来を体験しましょう!<br />
<br />
<iframe allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen="" frameborder="0" height="315" src="https://www.youtube.com/embed/J2U9Hmmpqhc" width="560"></iframe><br />
<br />
・・・やっぱりメルセデスのほうがいいですね!メルセデス最高!<br />
<br />
<hr />
<br />
最後に余談になりますが、Apple CarPlayとAndroid Autoについてちょっと話します。この2つはスマホ開発者界隈の皆様も少しくらいは聞いたことがある名前ではないでしょうか。これまで見てきましたとおり、最近のクルマには車載OSが存在するのですが、Apple CarPlayやAndroid Autoは車載OSのアプリの一つとして動作します。要するに車載OSのナビアプリを開いたりオーディオアプリを開く感覚で、CarPlayアプリを開いて、その中でCarPlayのUIの中で更にiOSのアプリを操作する、みたいな作りになっています。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://www.elasticfeed.com/wp-content/uploads/a075b1455336f2439652c9a283054ef5.jpg" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="600" data-original-width="800" height="240" src="https://www.elasticfeed.com/wp-content/uploads/a075b1455336f2439652c9a283054ef5.jpg" width="320" /></a></div>
引用元: <a href="https://www.elasticfeed.com/44db3e868b3b3bfe82835c65441ea3ca/">https://www.elasticfeed.com/44db3e868b3b3bfe82835c65441ea3ca/</a><br />
<br />
そのせいで、見てのとおりに画面が狭いです。どうやらApple Carplayは7インチの画面を前提に作ってあるようで、10インチのワイド画面であるメルセデスのディスプレイオーディオにはフィットしません。しかしながらこの中は完全にiOSであり、マトモなGoogle Mapsが機能します。おまけにスマホと車体が連携しており、車体側から車載GPSやジャイロコンパスなどのより正確な情報を提供してもらえるため、スマホ上で使うGoogle Mapsより段違いに自車の現在位置精度が高くなります。<br />
<br />
参考: <a href="https://developer.apple.com/documentation/carplay">https://developer.apple.com/documentation/carplay</a><br />
参考: <a href="https://developer.apple.com/design/human-interface-guidelines/carplay/interaction/car-data/">https://developer.apple.com/design/human-interface-guidelines/carplay/interaction/car-data/</a><br />
<br />
なんかクルマ向けにアプリ作ってみてえなと言う人がいらっしゃいましたら今がチャンスかも知れませんのでぜひぜひやってみていただければと思います。まぁ、デバッグに実車が必要・・・かもしれませんけど・・・<br />
<br />
最近は国産車でもトヨタ・カローラという昭和のジジイ以外だれも知らないようなオワコンカーがびっっっっっっっっっっくりするぐらいモダンで素晴らしいクルマに生まれ変わりまして、<a href="https://response.jp/article/2019/09/18/326603.html">そちらの新型トヨタ・カローラには標準でカーナビがついておらず、Apple CarPlay / Andriod Autoやトヨタ独自のSDLという規格を利用してスマホをナビにすることを前提の作りになっています。</a>まさに今のスマホ時代にふさわしい素晴らしい方針だと思います。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-34002588240467972342019-09-24T15:18:00.002+09:002019-09-24T15:18:27.147+09:00SwiftUI で ScrollView / Listの現在のスクロール位置 (UIKitにおけるcontentOffset) に相当する値を参照する/コード上から指定する方法は iOS SDK 13.0 地点では存在しないタイトルの通りですが一応補足しておきます。<br />
<br />
先日、奇しくもAppleのSwift UIを実際に手掛けている開発者に直接質問する機会に恵まれましたので以前から気になっていた内容を質問してみたのですが、やはり現段階のリリースではcontentOffsetに相当するものの変更タイミングを取得したり、またプログラム的にcontentOffsetを調整することもできないと明言されてしまいました。<br />
<br />
ただしcontentOffsetの変化したタイミングを取得するだけであれば、以下の方法で擬似的に再現することができるとアドバイスを頂きました。<br />
<br />
<br />
<ol>
<li>ScrollView / List 上の特定の位置に、ダミーの隠しViewを配置する。</li>
<li>隠しViewのonAppear() / onDisappear() を利用する。</li>
</ol>
<br />
<br />
彼曰く、コミュニティの誰かが作ったサンプルのVideoPlayerアプリがこのテクを使って動画が画面内に入ってきたときに自動再生を開始する挙動を実現していると言っていたのですが、具体的にどのサンプルアプリなのかは教えてもらえなかったので発見できず。残念。<br />
<br />
あとは例によっていつものごとく、機能追加が欲しい場合は <a href="https://developer.apple.com/bug-reporting/">https://developer.apple.com/bug-reporting/</a> 経由で要望を上げてくれると対応できるよと言っていましたので皆さんガンガン書けばいいと思います。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-88356485724995692622019-06-08T19:28:00.001+09:002019-06-08T19:33:03.874+09:00SwiftUI 未解決問題まとめ一通り試してほぼほぼ理解できたのですが、現状どこをどのように調べてもわからなかった内容がいくつかあるので未解決問題としておいておきます。誰か教えて\(^o^)/<br />
<br />
<h3>
Transition</h3>
type-eraseされたAnyTransitionはSwiftUI.frameworkに居るのですが、元のProtocolが見えない上にドキュメンテーションも一切ありません。おそらくはNavigationViewの中で使われているのだと思いますが詳細不明。<br />
<br />
<h3>
ScrollView / Listの現在のスクロール位置 (UIKitにおけるcontentOffset) に相当する値を参照する/コード上から指定する方法</h3>
今の所、一番怪しいのが以下のpreferenceの仕組みではないかと睨んでいるのですが、<br />
<ul>
<li><a href="https://developer.apple.com/documentation/swiftui/view/3278633-preference">preference(key:value:)</a></li>
<li><a href="https://developer.apple.com/documentation/swiftui/view/3278618-onpreferencechange">onPreferenceChange(_:perform:)</a></li>
<li><a href="https://developer.apple.com/documentation/swiftui/preferencekey">PreferenceKey</a></li>
</ul>
問題は肝心要のPreferenceKeyの具体実装がSwiftUI.framework上には一切見つからず、したがってキーが指定できないためonPreferenceChange(_:perform:)がうまく利用できません。特定位置までスクロールしたら発火、とかビューが50%スクロールして隠れたら発火、とか普通に使いたいのですが、困りました。それとかあとはenvironment経由なりイニシャライザ経由なりでinitialScrollPositionのようなプロパティを用意してScrollViewのスクロール初期位置を与える、みたいなテクも使いたいですし、普通に必要だと思うんですけど(´・_・`)<br />
<br />
ちなみにView.offsetではない・・・と思います、たぶん。一応念のためにList.offset()で試してみましたが、ドキュメンテーションにもある通り、全く違う挙動になります。akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-18241487978660307172019-06-08T19:12:00.002+09:002019-06-08T19:30:53.178+09:00SwiftUI チュートリアル ヘルプ ドキュメント FAQ 困ったらとりあえずここ見ればOK<a href="https://github.com/Juanpe/About-SwiftUI">https://github.com/Juanpe/About-SwiftUI</a><br />
<br />
いろいろ調べたのですが、これよりよくまとまっているドキュメントを現状発見できませんでしたので、2019/06/08現在では上記のドキュメントを参照するのがベストだと思います。<br />
<br />
特に以下の記事は役立ち度が高かったです。この2つだけで問題の9割ぐらいは解決できると思います。<br />
<br />
SwiftUI by Example<br />
<a href="https://www.hackingwithswift.com/quick-start/swiftui">https://www.hackingwithswift.com/quick-start/swiftui</a><br />
<br />
Answers to the most common questions about SwiftUI<br />
<a href="https://wwdcbysundell.com/2019/swiftui-common-questions/">https://wwdcbysundell.com/2019/swiftui-common-questions/</a><br />
<br />
あとは直接SwiftUI.frameworkの中身を見るのが良いと思います。正直Appleのドキュメンテーションは未だにSwiftのExtensionベースの実装をきれいにドキュメントに起こす事ができておらず、複数のドキュメントに重複した記載が見られたり、どこで定義されているfunc/varなのかを正しく表現できていなかったりします。なのでシグネチャの名前さえわかっていればSwiftUI.frameworkの中を自分で検索したほうが正確にどういう定義になっているか判断できて便利です。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-32611121446928018862018-12-19T21:29:00.000+09:002018-12-19T21:43:03.787+09:00HTML5 video / audioがiOSデバイスの消音スイッチの状態に従うようにする方法 (UIWebView編 / WKWebView編)いろいろ調べていたのですが、<a href="http://akisute.com/2015/10/ios-youtube.html">遥か大昔に私が書いたブログ記事</a>が長い年月を経て今や大間違いになっていたのでここに訂正させていただきたいと思います。<br />
<br />
今回ご紹介するのはUIWebViewまたはWKWebViewで表示しているHTML5のvideo要素やaudio要素が音声を出すときに、iOSデバイスについている消音スイッチ (ミュートスイッチ, mute switch, silent switch) の状態を無視してしまう問題を解決する方法です。相変わらず紹介内容がとてもニッチですね。<br />
<br />
ちなみにSafariはデフォルトでちゃんと消音スイッチの状態を反映してくれるので、SafariでYouTubeの動画を見ててもいきなり音が流れ出すことはありません。素晴らしい!というわけで我々のアプリもぜひそのようにしたいと思います。<br />
<br />
<h3>
消音スイッチの挙動について</h3>
まず基本的なおさらいとして、消音スイッチがどのような挙動を示すかについてこちらにまとめます。<br />
<ul>
<li>各プロセスごとに、AVAudioSessionが消音スイッチに対してどのように振る舞うかを定義している。</li>
<li>具体的には、AVAudioSession.Categoryの値に応じて挙動が変化する。ドキュメントにも明記されている。</li>
<ul>
<li>AVAudioSession.Category.ambientやAVAudioSession.Category.soloAmbientを指定すると、消音スイッチがONのときは音が出ないようになる。</li>
<li>逆にAVAudioSession.Category.playbackを指定すると消音スイッチを一切無視するようになる。</li>
</ul>
<li>消音スイッチの現在の物理的な状態を取得するAPIは一切ない。Private APIで以前は可能だったが、穴が塞がれたためその方法も利用できない。そもそもアプリが提出時の審査で蹴られる。当然JS経由でHTMLコンテンツ上から状態を取得するのも不可能。</li>
</ul>
<br />
<h3>
解決方法・UIWebView編</h3>
UIWebViewはWebKit1を利用しており、したがってUIWebViewのエンジンは我々のアプリ内のプロセスで動作します。そこでUIWebViewのインスタンスを生成するより先に、先述の通りAVAudioSessionのcategoryを変更して消音スイッチの状態を反映してやるように示してやるとうまくいきます。<br />
<pre>try? AVAudioSession.sharedInstance().setCategory(.soloAmbient, mode: .default, options: [])
try? AVAudioSession.sharedInstance().setActive(true)</pre>
ただしご存知の通り、すでにUIWebViewはdeprecated扱いとなっており、WKWebViewへの移行が推奨されています。そこでWKWebViewでも同様の方法が使えないかやってみましょう。<br />
<br />
<h3>
解決方法・WKWebView編</h3>
ありません。<br />
<br />
繰り返しますが、ありません。WebKit2の設計上のバグです。<br />
<br />
今後修正される可能性もほぼ間違いなくありません。<a href="https://forums.developer.apple.com/thread/24464">諦めてください。</a><br />
<br />
一応、順を追って説明します。<br />
<ol>
<li>WKWebViewはWebKit2で実装されています。</li>
<li>WebKit2は我々のアプリ内のプロセスで動作するのではなく、「共用の」WebCoreプロセス上で動作し、その結果が我々のアプリ内に転送されてくるような実装になっています。</li>
<li>「共用の」WebCoreプロセスは(これは推測ですが、audioやvideoを最大限に活かすため)AVAudioSession.Category.playbackとAVAudioSession.Mode.moviePlaybackで動作するように設定されている用に見えます。したがって消音スイッチは無視されます。</li>
<li>最初にご説明したとおり、AVAudioSessionによる消音スイッチに対する振る舞いは「プロセス単位」で制御されており、これは遥か大昔のiOS 2.0どころか下手するとNeXTSTEPの時代からCarbonレイヤーで決められている挙動だったりします。要するに今からの変更はほぼ不可能です。</li>
<li>もうおわかりだと思いますが、「共用の」WebCoreプロセスの挙動を我々個別のアプリがAVAudioSession経由で勝手に変更することは不可能ですし、これを可能にするのはiOSとWebKitの設計上実質不可能なため、問題は解決されません。</li>
<ol>
<li>WebCoreプロセスを各個別のアプリごとに立ち上げて対応しろよ!と思うかもしれませんが、WebCoreプロセスは近年AppleのOSに組み込まれている特権階級モードで動作するプロセスとなっており(そのためメモリに無制限なアクセスが可能で、JSをJITコンパイルして高速に動作させる事が可能)、最近のセキュリティとバッテリーライフにうるさいAppleがそれを個別のアプリごとに立ち上げさせるなどということは現経営陣が全滅しない限りありえないと断言できます。</li>
<li>共用のWebCoreプロセスが使用するAVAudioSession.Categoryを変えてしまえばいいだろ!とも思いますが、実はAVAudioSession.Categoryをplaybackに指定しない限りバックグラウンドで音楽を流し続けることができません。したがって今度はWebViewで音楽プレイヤーを作ったりPicture in Pictureで動画を流し続けるアプリが全滅するため、これも不可能です。</li>
<li>だったら消音スイッチの状態を自分で調べて自分で動画をmuteにすればよいのでは、と思いますが、これも先述の通り消音スイッチの状態を調べる方法は一切ないため、一律でmuteにしてしまうなどの乱暴な方法を用いない限り対応できません。</li>
</ol>
</ol>
やばいですね☆<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-64218450069043140212018-12-19T20:54:00.001+09:002018-12-19T20:55:24.157+09:00iOS開発でエラーコードを調べるときはOSStatus.comを使おう久しぶりに書く価値のあるネタが見つかったのでご紹介します。<br />
<br />
iOSのフレームワークがNSErrorを返してきたとき、そのcodeが何を意味するのかを調べる必要が出てくることが往々にして発生します。新し目のフレームワークは比較的簡単に調べられるようにまとまっているのですが、厄介なのがOSStatusなどの大昔から存在するエラーコードの場合です。<br />
<br />
<a href="http://d.hatena.ne.jp/watanata2000/20110920/1316517342">NSError コードの調べ方</a> こちらのブログ記事で紹介されている通りに一つ一つヘッダファイル内を探す方法でもいいのですが、もはや平成の終わりが目の前に迫っている2018年の年の瀬にもなってこのような原始的な方法を使っているようではよろしくありません。<br />
<br />
そこで <a href="https://www.osstatus.com/">OSStatus.com</a> の出番です。こちらの検索欄にcodeをコピペして検索ボタンを押すだけであらゆるエラーの詳細が一発で帰ってきます。実際にやってみましょう。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEiDpVDipRSxXltdOeb_rXQWNpPy-aovtX7o_QTZyvckzdnZW_lmG-xECLkn0PA-GLhWNHjqsrpjcQY6X13WnRddVVX_5lP5ddP9Q-d5fjy1dasYhve05e4tXFb_8uvjsOd21ObPB0gxiA8/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2018-12-19+20.49.56.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="829" data-original-width="1600" height="206" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEiDpVDipRSxXltdOeb_rXQWNpPy-aovtX7o_QTZyvckzdnZW_lmG-xECLkn0PA-GLhWNHjqsrpjcQY6X13WnRddVVX_5lP5ddP9Q-d5fjy1dasYhve05e4tXFb_8uvjsOd21ObPB0gxiA8/s400/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2018-12-19+20.49.56.png" width="400" /></a></div>
<br />
一発ですね。素晴らしい。<br />
<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjW1UTevoAknAO4aGpbqgP45No6w-_i7vvauIl7_ZOiWOB7upkIRvnPHXnOcVX1o5kIc8nhy_FitXLdXZHlf6ty4C37RQ8YPbiQVojWLFQmrj5CNQVOdW6Ot552MBVPgFInaYD29UQWifs/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2018-12-19+20.53.52.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="940" data-original-width="1600" height="235" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjW1UTevoAknAO4aGpbqgP45No6w-_i7vvauIl7_ZOiWOB7upkIRvnPHXnOcVX1o5kIc8nhy_FitXLdXZHlf6ty4C37RQ8YPbiQVojWLFQmrj5CNQVOdW6Ot552MBVPgFInaYD29UQWifs/s400/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2018-12-19+20.53.52.png" width="400" /></a></div>
<br />
名前での検索もバッチリです。<br />
<br />
偶然見つけたのですが、はてブもほとんどついてないし、ネット上で紹介されている記事も全く見つからなかったので書いてみました。便利です!<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-48435850536898996532018-12-01T00:00:00.000+09:002018-12-01T00:00:10.493+09:002018年のしめくくり<a href="https://adventar.org/calendars/3018" style="caret-color: rgb(68, 68, 68); font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">pyspa Advent Calendar 2018 12/1です。</a><br />
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
早いもので、あっというまに2018年も終わりを迎えそうな時期になってまいりました。皆様ご無沙汰しております、akisuteです。今年もまとめだけ書いて終わりにしたいと思います。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<h3>
今年やったこと</h3>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
JavaとかSwiftとかKotlinとかNode.jsとかReact Nativeとかやってました。Node.jsはなかなかいいですね。食わず嫌いで最近まで手を出しておらず、昔はお手軽になにか作りたいときはもっぱらPythonだったのですが、今ではついついNode.jsを使うようになってしまいました。特にTypeScriptが大変気に入っております。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
細かいところですとチームのドキュメンテーションの問題を考えている時間が増えてきました。というのも最近ようやく気づきを得たのですが、プログラマの中にはドキュメンテーションができるタイプの人と一切できないタイプの人の二種類がいるように思えます。後者のタイプの人がドキュメンテーションが不得手な理由を自分なりに考えていますが、</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<ul>
<li>「自分が理解しているので、相手も理解しているだろう、相手も理解できているだろう」という考えがあり、そのため本来必要であるドキュメントが不用と判断されて欠落してしまう。</li>
<li>思考が直接コードで行われているために、わざわざ日本語なり英語なりの言語に変換されない。またはその変換コストがその人にとって高い。</li>
<li>間違った情報ということは認識しているのだが、既存のWikiやコードコメントを書き換えるのを遠慮してしまう。</li>
</ul>
</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
これらが原因なのかどうか、いずれも定かではありません。テンプレートを整備してみたり、OpenAPI+ReDocのような解決策に頼ってみたりしているのですが、なかなか成果が上がっておらず。ドキュメントが不足しがちな人をあえてドキュメントが必要な状況におく(普段とは違うプロジェクトをアサインして、ドキュメントを読まなければにっちもさっちもいかない状態にすることで、どのような情報が欠落すると自分が困るのかを理解できる)とかはいいアイデアなのでは?と思っています。ドキュメンテーションの得意な人と組ませて、得意な人がレビュープロセスなどで書き加えるというアイデアなどもなんとなく回りそうな気はしています。なかなか難しいです。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<h3>
今年のGotY</h3>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<a href="https://forzamotorsport.net/en-us/games/fh4">Forza Horizon 4</a>以外にありえないですね。これより優れたクルマゲームは地球上に存在しないと断言していいレベルで素晴らしいです。RDR2は正直雰囲気と世界観のためにゲーム性を犠牲にしすぎている気がします。1ほど楽しめませんでした。どうやら私もおっさんになったようです。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<h3>
今年のマイブーム</h3>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
Forza Horizon 4のせいか急にクルマがマイブームになってきました。と言っても別に今すぐクルマを持ちたいとは思っていません。なにせ現実世界のクルマはカネがかかり、乗っても楽しく乗れる場所はなく、ただひたすら渋滞とマナーの悪いドライバーに憤慨し、事故とスピード違反に怯えながらほそぼそとアクセルを踏まなければならないような代物です。全く執着するに値しません。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
しかしながらゲームの中となると、たかだか1万円で何百車種もの好きなクルマに好きなだけ乗ることができ、キレイな風景や楽しいコースを好き放題乗り回し、渋滞もなく、マナーの悪いドライバーは遠慮なくポルシェ・カイエンターボで粉々に踏み潰すことができ、事故ったところで現実世界の私はかすり傷一つ受けることもないため何のリスクもなくクルマの限界に挑戦できます。まさにクルマの楽しい箇所だけをすべて味わい尽くすことができるのです。うーん、仮想世界って素晴らしいですね。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
<br /></div>
<h3>
まとめ</h3>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
今年も平和でいい年が過ごせて100%満足しています。よかったです。</div>
<div style="caret-color: rgb(68, 68, 68); color: #444444; font-family: "Trebuchet MS", Trebuchet, Verdana, sans-serif; font-size: 14px;">
来年も平和で良い年でありますように。</div>
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-80599479942200954192017-12-04T00:00:00.000+09:002017-12-04T14:01:20.624+09:002017年のしめくくり<a href="https://adventar.org/calendars/2258">pyspa Advent Calendar 2017 12/4です。</a><br />
<div>
<br /></div>
<div>
どうもお久しぶりです。オンライン上では一切活動しておりませんが、おかげさまで大変元気にやってます。というわけで2017年のまとめを書いておこうと思います。</div>
<div>
<br /></div>
<h3>
そもそもなんで最近ブログ書いてないの</h3>
<div>
書くことがないからです。というのも私は最近iOS/Androidの開発ではなくジャバでサーバサイドアプリケーションを書くみたいな仕事をしていて、別にSpring Bootの話なんかここに書いてもあまり面白くもなければ、すでにありとあらゆる先人が地雷を踏み尽くしていて調べればだいたいなんでもわからないことは分かる状態になっているので、わざわざ私が書く必要もないと思われるためです。</div>
<div>
<br /></div>
<h3>
なんでiOS/Androidやってないの</h3>
<div>
端的に言えば愛想が尽きました。iPhone Xは買っていません。SwiftとKotlinは大好きで今でも仕事で使ったり他人のコードをレビューしたりする毎日ですが、もはやiOSもAndroidもどちらも執着するには値しません。</div>
<div>
<br /></div>
<h3>
なんか技術的に新しいネタないの</h3>
<div>
正直今ネイティブアプリに大きな流れはないので(ARとクライアントサイド機械学習程度しか新規性のあるネタがなく、どちらも適用分野が限られ、ほとんどのアプリケーションでは縁がない)、後数年の間はモバイルネイティブアプリ一択だった流れが半分ぐらいウェブに返ってくると思っていて、ゲームなどではすでにその兆候もありますし、React NativeとかモバイルフロントエンドJavaScriptとかを見ておけばいいんじゃないのかなぁと思っています。個人的にはあんまりどちらも好きじゃないんですが、悪いものではないと思います。</div>
<div>
また、Rxはもはや使えて当然と思っていますが、しかしながらRxを限界まで酷使するようなアプリはごく少数の人間しかメンテできなくなり、そのうちメンテがおぼつかなくなり、Rxのシグナルやオブザーバを使うのではなく昔ながらのdelegateやcallbackで手っ取り早くコードを修正するような事態が訪れ、ゆっくりと破綻していくような事態が周囲で散見されています。正直人間のほうが技術側に追いついていないというか。使えるやつは使えるんだけどそれじゃメンテ回らないよというか。</div>
<div>
<br /></div>
<h3>
なんか注目のビジネスネタないの</h3>
<div>
フィンテックはガチだと思います。儲かるので。ただしビットコイン、というかブロックチェーン技術を神聖視するのは正直イケてないと思います。</div>
<div>
動画配信系もガチだと思います。これもカネの匂いがします。ただしこっちはそもそもプレイヤーになれる存在が非常に限られており、ドワンゴすらそろそろもうついていけないかなぁという状態ですので、皆様がんばってくださいという感じです。</div>
<div>
AIが〜とか機械学習が〜とかはやらなくてはならないのは間違いありませんが、ぶっちゃけ現状ではまだそんな言うほど大したことないです。どっかのタイミングで私みたいな何もわからないド素人でも簡単に放置してデータ食わせるだけで勝手に最良の状態に学習する事ができるような仕組みが誕生すると思っていて、そうなったら本当に爆発的に世界が変わる気がしています。それまでは自動運転みたいなカネが突っ込まれまくる分野でのみ世界が変わるほどの進展が見られるんじゃないかと勝手に思っています。</div>
<div>
スマートスピーカーは正直使ってみて要らないと思います。不便とは思いませんが、現状無くてもほとんど全く困らないので要らないです。無いと困るようになったら爆発的に売れるのではないでしょうか。</div>
<div>
<br /></div>
<h3>
今年のゲームはどうだったの</h3>
<div>
ゼルダがGotY 2017なのは間違いありませんが、その他も非常に豊作な一年だったと思います。個人的な推しは<a href="http://store.steampowered.com/app/518790/theHunter_Call_of_the_Wild/">theHunter: Call of the Wild</a>です。</div>
<div>
<br />
<h3>
家どうなったの</h3>
<a href="http://wozozo.hatenadiary.jp/entry/2017/12/01/002206">友人が家を探しているというので色々物件を紹介していたら、まさかの我がマンションに引っ越してきました。</a>おかげさまで大変楽しい毎日です。近所付き合いってのも悪いもんじゃないと思います。<br />
あとはベッドを買い替えました。こちらもiPhone Xを買うより100倍ぐらい幸福になりました。良いお金の使い方をしたと思います。<br />
<br />
<h3>
婚活どうなったの</h3>
退会しました。1年間、のべ数百人単位の女性を紹介してもらって、数十人単位と付き合って、数人とじっくり時間を過ごしてみた得た結論は、正直私は女性と付き合っても煩わしいばかりで楽しくないし、相手も私といても満足できないし、結婚してもいいことはないというものでしたので、そのようにしようと思います。1年前はちょっと焦りとかもありましたが、今では大変気分も落ち着いて平穏が得られました。自らの人生について考える良い機会だったと思います。<br />
<br />
<h3>
今どうなの</h3>
おかげさまで非常に幸福な人生を送っていると思います \(^o^)/<br />
来年もいい年になりますように!<br />
<br /></div>
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-7492911682966992162017-06-19T18:04:00.001+09:002017-06-19T18:04:04.335+09:00UIWebView, WKWebView 等において Drag and Drop を禁止する方法<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjT26tEV1YNTLq20t1Q-ZJn9IDaWZLZURDPncKjyq8pqQKRAdtwzeSOYc6qQJxEctreeP9b3RlLQF6jkATRjeS1VkEPqjLDr16idLWXSJufLO5kbojrB7Hwd_ljQ_2HQO9h2qlZnE0zevg/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2017-06-16+18.28.27.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" data-original-height="1600" data-original-width="897" height="400" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjT26tEV1YNTLq20t1Q-ZJn9IDaWZLZURDPncKjyq8pqQKRAdtwzeSOYc6qQJxEctreeP9b3RlLQF6jkATRjeS1VkEPqjLDr16idLWXSJufLO5kbojrB7Hwd_ljQ_2HQO9h2qlZnE0zevg/s400/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2017-06-16+18.28.27.png" width="223" /></a></div>
<br />
iOS 11よりDrag and Drop APIがUIKitに追加され、UITextView / UITableView / UICollectionViewに簡単にDrag and Dropを実現するためのdelegateが用意されたり、それ以外のUIViewにもDrag and Dropをハンドリングするための仕組みが用意されました。このDrag and Dropは基本的にはアプリをまたいでデータを受け渡しすることを前提として作られていますが、一応は自分のアプリ内でデータを移動したりコピーしたりするためにも使えるようになっているので、積極的に活用していきたいところです。<br />
<br />
ところでこのDrag and Drop、基本的にはアプリ側が対応しない限り自動的には対応してくれないのですが、例外があります。それがUIWebView / WKWebView / SFSafariViewController要するにWebViewのたぐいの皆様です。これらについては最初からDragも可能ですしDropも可能なように作られているようです。便利ですね!<br />
<br />
ですがここをご覧になっている皆様の中には、大人の事情によりどうしてもそのような便利な機能をユーザーさんにわざと提供したくないと言うケースがある方もいらっしゃるかと思います。そこでUIWebView / WKWebViewのデフォルトのDrag and Dropを無効化したり、頑張って自分のコードでハンドリングしたりすることができるようにする方法をご紹介します。なおSFSafariViewControllerに対しては手も足も出ないのでご了承ください。<br />
<pre>if #available(iOS 11.0, *) {
webView.scrollView.subviews.first?.interactions = []
}</pre>
たったのこれだけです。簡単ですね!<br />
<br />
なんのこっちゃなので詳しく説明します。<br />
<br />
まずはじめにiOS 11よりUIView.interactionsプロパティが追加になっています。このinteractionsは[UIInteraction]型となっていて、公式のドキュメントによると「ジェスチャベースの挙動をビューに定義する」ことができるらしいです。まさにドラッグ・アンド・ドロップ的なやつですね。<br />
で、Appleによって最初からドラッグ・アンド・ドロップ用のUIInteractionすなわちUIDragInteractionとUIDropInteractionが用意されています。これらのInteractionは内部的にdelegateを持っていて、これらのInteractionがインタラクションを検知したらdelegateにメッセージングを行い、そいつらが実際のドラッグ・アンド・ドロップ処理を行うというような仕組みになっているようです。<br />
調べてみたところUITableViewのような最初からDrag and Drop用のdelegateを追加されているViewについても、このinteractionsに何らかのUIDragInteraction / UIDropInteractionが追加されていて、それが一旦イベントを拾ってから我々が定義したUITableViewDragDelegate / UITableViewDropDelegateに問い合わせを行うという仕組みで動作しているということがわかりました。<br />
<br />
ここまで分かれば後は簡単で、UIWebView / WKWebViewも同様にinteractionsで一旦イベントを受け取った後に、内部的に処理を行っていると考えられるのでUIWebViewの中身をどったんばったん探してみたところwebview.scrollView.subviews.firstの中にinteractionsが刺さっているのを発見したと言う次第です。<br />
<br />
先程の無効化する例では空配列を渡すことで完全にinteractionsを無効化していますが、ここで自分が作ったカスタムのUIDragInteraction / UIDropInteractionを渡してやればうまい具合に自分でWebViewのDrag and Drop挙動をカスタマイズすることができるかもしれません。また両方を消すのでなくて片方だけを消せばDragによってデータを持ち出すのは禁止するがDropによってデータを持ち込むのは許可するみたいなこともできます。akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-53623883902594598612017-05-24T20:03:00.002+09:002017-05-25T20:55:18.991+09:00Xcode 8.3 & Swift 3.1 環境で LLDB を使う Tips以前も似たような記事書いた気がするんですが、定期的にSwiftのコード上でLLDBを使ってて困ることがあるので定期的に書くことにします。ほとんど <a href="https://stackoverflow.com/questions/29441418/lldb-swift-casting-raw-address-into-usable-type">https://stackoverflow.com/questions/29441418/lldb-swift-casting-raw-address-into-usable-type</a> からの転載です。<br />
<br />
<h3>LLDBの言語を指定して実行する</h3>標準ではbreakpointを挟んでbreakした行がSwiftであればSwift、Objective-CであればObjective-CでLLDBの言語が選択されるようなのですが、これでは毎回毎回breakした地点に応じてデバッグの仕方が変わって大変で仕方がないので、Swiftに統一してしまうのが良いと思っています。<br />
<pre>expr -l swift -- {Swift Command}
e -l swift -- {Swift Command}</pre>とすると常にSwiftでLLDBに対してコマンドを送る事が可能になります。逆にObjective-C側に寄せたい場合は<br />
<pre>expr -l objc++ -O -- {Objective-C Command}
e -l objc++ -O -- {Objective-C Command}</pre>とすれば良いです。-Oオプションがないと結果がいい感じにObjective-Cのオブジェクトとして扱われないので注意。<br />
<br />
<h3>importを忘れずに</h3>以前も書いた気がしますがLLDBでbreakした環境はそのままだとFoundation.frameworkすらまともに使えない(シンボルを認識しない)状態なので、importを忘れずにしましょう。<br />
<pre>e -l swift -- import UIKit
e -l swift -- import MyApp
e -l swift -- import MyAppLib</pre>という具合でしょうか。他必要なものがあればその都度。注意点として自分のアプリ自体に含まれるシンボル、例えばAppDelegateだとかViewControllerのたぐいだとかもimportしないと触れないので忘れないように。<br />
<br />
<h3>unsafeBitCast(_:to:)を使ってポインタからオブジェクトを取得する</h3>これも以前書いた気がしますが、Swift 3でまたシグネチャが変わったらしいので念のため。<br />
<pre>e -l swift -- let $obj = unsafeBitCast(0x00000000, to: MyClass.self)
e -l swift -- print($obj)</pre>とかやるといい具合になります。<br />
<br />
<h3>エイリアスを作ろう</h3>毎回<code>e -l swift -- </code>とかタイプするの面倒ですし、poが<code>e -O -- </code>のエイリアスなのと同様に、eswiftとかpswiftとかそういうエイリアスを作ってしまえばいいと思います。<code>~/.lldbinit</code>の中に、<br />
<pre>command alias eswift e -l swift -- </pre>とか書いておけば良い気がします。akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-65860337186937731582017-03-07T21:51:00.001+09:002017-03-07T21:54:35.373+09:00NSAttributedString に lineSpace を付与したとき、1行であるにも関わらず lineSpace が下端に付与されて困っている人はこれを読んだら直ります<div><div class="separator" style="clear: both; text-align: center;"><a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh-6FultmZOa9EcfDAeYT0DuctulSA4tW5bMutUIBjBY_y23jajbB76qznaxk7UZZSHDO7p4JUXLQwdiy0hy3JXK0qGa2LqQrIRAvJdPQTZvsRJJ0vpLxovLdckTPENCLeqkl0Stjl7-hk/s1600/x8gVt.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="400" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEh-6FultmZOa9EcfDAeYT0DuctulSA4tW5bMutUIBjBY_y23jajbB76qznaxk7UZZSHDO7p4JUXLQwdiy0hy3JXK0qGa2LqQrIRAvJdPQTZvsRJJ0vpLxovLdckTPENCLeqkl0Stjl7-hk/s400/x8gVt.png" width="333" /></a></div><br />
</div><div><br />
</div>突然ですが皆さん、NSAttributedStringには2017/03/07付のiOS 10.3現在でも致命的にバグっている箇所があります。<br />
<br />
<h4>症状:</h4><div><ul><li>NSAttributedStringにNSParagraphStyleAttributeNameを利用してNSParagraphStyle経由でlineSpaceが付与されている。</li>
<li>NSAttributedStringに複数の「区間」が存在する。例えばNSAttributedString全体が単一のAttributeによって構成されているときはこの問題は発生しません。2つ以上の異なるAttributeの区間が必要です。</li>
<li>NSAttributedStringを描画するときに、横幅及び文字列の長さの都合で、1行で描画される。複数行になるときにはこの問題は発生しません。</li>
<li>NSAttributedStringにNSBackgroundColorAttributeNameが付与されていない。または、NSBackgroundColorAttributeNameとNSKerningAttributeNameの両方が付与されている。</li>
</ul></div><div><br />
</div><div>上記の条件を全て満たすとき、</div><div>本来、1行の文字列はレンダリングしたときに下端にlineSpaceが付与されてはいけませんが、この条件が満たされていると下端にlineSpaceが付与されてしまいます。先頭の画像の左上のケースがこの問題に相当します。</div><div><br />
</div><div>具体的な問題としては、</div><div><ul><li>問題が発生するNSAttributedStringを使用したUILabel, UITextField, UITextViewの描画内容が思いっきり上にずれます(下に無駄なスペースが発生するので)</li>
<li>NSAttributedStringの地点で壊れているので、Core Textを利用したり、<code>boundingRect(with size: CGSize, options: NSStringDrawingOptions = [], context: NSStringDrawingContext?) -> CGRect</code>を使っても一切回避できません</li>
</ul></div><div><br />
<h4>解決策: </h4>この問題に対する正しいワークアラウンドの方法は一つしかありません。</div><br />
NSAttributedStringの全Attribute区間にNSBackgroundColorAttributeNameを付与し、かつどの区間にもNSKerningAttributeNameを付与しない。<br />
<div><br />
<div>背景色が不要の場合はUIColor.clearでも付与してごまかしてください。</div><div>以上です。よろしくお願いします。</div><div><br />
</div><h4>参考:</h4><div><ul><li>この問題の発端、一番わかり易い図説がある、誰も正しい回答を示せていない</li>
<ul><li><a href="http://stackoverflow.com/questions/23575958/uilabel-size-incorrect-for-single-line-of-text-with-linespacing-and-multiple-col">http://stackoverflow.com/questions/23575958/uilabel-size-incorrect-for-single-line-of-text-with-linespacing-and-multiple-col</a></li>
</ul><li>もう一つの例、カスタムフォントが悪いということにされているが、カスタムフォントは一切関係ない、システムフォントでも発生する</li>
<ul><li><a href="http://stackoverflow.com/questions/32291105/line-spacing-for-uilabel-with-a-single-line-of-text">http://stackoverflow.com/questions/32291105/line-spacing-for-uilabel-with-a-single-line-of-text</a></li>
</ul><li>最後の例、この例によって「背景色があれば問題が発生しない」「カーニングを付与すると背景色があっても問題が再発する」ことが示され、この問題の正しい解決策が得られた</li>
<ul><li><a href="http://stackoverflow.com/questions/32216700/uilabel-with-paragraph-style-line-spacing">http://stackoverflow.com/questions/32216700/uilabel-with-paragraph-style-line-spacing</a></li>
</ul></ul></div></div>akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-49976148992055834512016-12-13T20:51:00.002+09:002016-12-13T20:51:20.424+09:00Instruments を使っていてシンボルが見えないときの対処法Instruments を久しぶりに起動して Time Profiler を使おうとしたら、びっくり仰天シンボルが全部見えない状態になっているじゃあありませんか。 Swift化したのがあかんのか!?と思って調べてみました。<br />
<br />
結論から言うと、先日ビルドの軽量化のためにDebugビルド時にdSYMの出力を止めていたのが原因でした(´・_・`)<br />
<a href="http://stackoverflow.com/questions/36882788/how-to-create-dsym-file-in-xcode-7-0">http://stackoverflow</a><a href="http://stackoverflow.com/questions/36882788/how-to-create-dsym-file-in-xcode-7-0">.com/questions/36882788/how-to-create-dsym-file-in-xcode-7-0</a><br />
Debug Information FormatをDWARF with dSYM Fileに戻して、dSYMの出力を再開したら問題なくシンボルが出力されるようになりました。なんじゃそりゃ。<br />
<br />
それでもだめな場合はInstruments側のFile -> Symbols... を選択すると、直接モジュールのdSYMを指定する事が可能になります。覚えておくと便利かもです。<br />
<a href="http://stackoverflow.com/questions/33683690/apple-instruments-has-mangled-symbols-and-greyed-out-symbol-names-when-profiling">http://stackoverflow.com/questions/33683690/apple-instruments-has-mangled-symbols-and-greyed-out-symbol-names-when-profiling</a><br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-56105936579023796722016-12-01T02:48:00.000+09:002016-12-01T11:45:54.364+09:00pyspa Advent Calendar 2016 1日目: 今年の話去年に引き続きまして <a href="http://www.adventar.org/calendars/1435">pyspa Advent Calendar 2016</a> 一番槍担当の<strike>斧さん</strike>akisuteです。よろしくお願いします。<br />
<br />
去年はゲームの話をしましたが、今年は人生に一度しかやらなさそうなイベントを大量に発生させましたので、主に今年何をやっていたのかについてご報告させていただきたいと思います。<br />
<br />
<h3>
1. 家を買いました</h3>
ずっとワンルームのアパートにしか住んだことがない男だったのですが、30過ぎたので人生一度きりだし家でも買ってみるかと思い、半年ほど探して思い切ってタワーマンションの部屋を買ってしまいました。<br />
<br />
正直ちょっと前まで家を買うとか単に負債を抱えるだけの大アホだと思っていたのですが、実際に買ってみると案外悪くなかったです。売ればいいとか資産価値があるとか賃貸より買ったほうが割安とかそういうことはさんざん世間様がおっしゃってるかと思いますので、ここではあんまり世の中では言われていない気づきを書いてみようかと思います。<br />
<br />
<h4>
買うと(当たり前だけど)自分のものになる</h4>
賃貸で引っ越しするのとはまるで違う感覚です。まず引き渡しですが鍵と権利書をもらうだけです。クリーニングすらされてません。売主は貸主様じゃないので当たり前です。そこでハウスクリーニングを自分で呼んで掃除してもらい、あとは業者を呼んで鍵を付け替えてもらったりします。なんというか、自分のものなんだな、今まで家を借りてた相手の家主さんというのはこういうことをやってたのかな、という感じがします。<br />
<br />
当然(マンションの管理規約の範囲内で)部屋は自分のものなので、壁に穴を開けようがぶち破ろうが工事してみようが全て自由・自己責任となります。逆に老朽化した設備を保守点検するのも自分の責任となります。とにかくなんというか、守るものができたなぁという感じを大いに受けます。<br />
<br />
<h4>
売買は意外と大雑把</h4>
私の場合売主さんが全ての鍵を渡すのを忘れてましたというのが後から発覚して大騒動になりました。最終的には無事全部私の手元に来たのですが、大手の不動産仲介屋というのも案外雑なもんだなという気持ちになりました。<br />
<br />
あと管理費・修繕積立金の将来の計画(当然年度が上がるごとに高くなる、大雑把に言って5年おきに値上がりで、30年で2倍に上昇すると見積もればだいたいOK)ですとか、固定資産税の額ですとか(これは実際に買うまで見積もりもなかなかできない、年間20万円ぐらいすることもある)、家が広くなるので水道光熱費が上がるとか、そういう諸費用は契約直前までなかなか教えてくれないので注意した方がいいです。大体不動産屋も銀行もローンと最初の管理費・修繕積立金ぐらいまでしか計算に入れずに大丈夫ですよとか言って勧めてくるので目に唾を付けて聞いてください。私の場合はお願いした不動産屋さんが知り合いのツテで、大変親切に説明していただいたお陰で難を逃れました。<br />
<br />
<h4>
買うと銀行が身近になる</h4>
<div>
家を買わない限り銀行なんてただの財布の延長程度の扱いでしかない、という感覚だと思うのですが(特に私と同年代以下の方は)、家を買うとなると銀行が本格的にお客さんというかパートナーというか商売相手になるというか・・・とにかく全く別の相手になります。銀行の担当の人とは仲良くしましょう。</div>
<div>
それから自分の口座から頭金として3桁万円の現金札束を引き出したり、4桁万円の融資が飛び込んできたりするのが目の前で見られてちょっと社長にでもになった気持ちが味わえます。でもできればあんまり味わいたくないです(´・_・`)</div>
<div>
<br /></div>
<h4>
買うと強制的に勉強せざるを得ない状態になる</h4>
やれ保険があるだのいざとなれば売ればいいだの言われますが、借金は借金です。返せなければ死ぬしかありません。今は金利も安いですが将来はわかりませんし、何より管理費・修繕積立金・自宅設備の更新など、将来必要になるお金は増える一方で減ることは絶対にないと悲観的に考えたほうがよいぐらいです。そして我々の給料はほっといても絶対に上がりません。退職金も年金もありません。<br />
<br />
よく家を買うと保守的になるだの守りに入るだのというネガティブな考えを持ちそうになりますが、そんなことはありません。むしろこの激動の時代において、守りに入ったら即座に借金が返せなくなって死にます。死にたくなければ勉強し続け変化し続け攻め続けるしかありません。最大の防御は攻撃です。<br />
<br />
そういった理由で最近はゲームをする時間もめっきり減って、代わりに株式投資だの投資信託だのの勉強をしてみたりとか、経済のニュースを見て将来の住宅ローンの金利がどうなるか調べてみたりだとか、仕事もiOSアプリばっかりいじってないでDockerだのansibleだのに手を染めてみたりだとか、筋トレする時間を毎日取って体調を維持してみたりだとか、そういうおっさんっぽい規則正しい生活を強制的に余儀なくされております。<br />
<br />
家を買うまではそれまでの努力の成果に甘えてダラダラ過ごせてしまっていたので、家を買って強制的に借金を持って自分を追い込んだのは大正解だったと思っています。<br />
<br />
<h4>
買うと視野が広くなる</h4>
生活が強制的に変わるので当然といえば当然なのですが、その他家が広くなるので今まで置けなかった家具が置けるようになったり、筋トレ器具を置けるようになったり、最新のIoTっぽいアイテムを買って試してみたり、カーシェアリングサービスを使ってみたりと、試せるものの範囲が広がります。これは今まで生まれてきてプログラマーしかやったことがない私には大変刺激的です。特にルンバや食洗機には本当に感動していて、私が作ってきたiPhoneのアプリなんてこのルンバの1/100も人の役に立たないと反省することしきりです。<br />
<br />
<h4>
近所付き合いの話</h4>
<div>
よくタワーマンションを買うと上下格差が〜とか近所付き合いが〜とか言われますが、今私が住んでいるところでは特に上下格差が〜みたいなものは一切見られません。その代わりと言っては何ですが管理組合の人が良くも悪くも恐ろしく有能で厳しい人で、とにかく「マンションの資産価値を守る」という点において凄まじいです。引っ越しの日に私が使った業者の養生がショボかったのを見つけて廊下が割れたらどうするつもりだとものすごい勢いで業者の人に怒鳴ってました。後で菓子折り持って謝りに行きました(´・_・`)</div>
<div>
<br /></div>
<div>
逆に防災訓練を定期的に開催したり、子供向けのハロウィンパーティを開催したり、設備に不備があったらゼネコン建設会社や販売会社のアフターサービス担当に殴り込みに行って無償修理させるなどの活躍をしているようで、敵に回したら怖いけど味方にしたら頼りになるなぁと思って見てます。</div>
<div>
<br /></div>
<div>
近所付き合いに関してはやはり多少はあって、ちゃんと出会った人には挨拶して、お子さんには愛嬌よくして、お隣さんには菓子折り持ってよろしくお願いしますと挨拶に行って、ぐらいは当然必要になります。ですが慣れればなかなか楽しいものです\(^o^)/</div>
<br />
<h3>
2. 婚活をしました</h3>
30過ぎたのでいっちょ結婚ぐらいした方がいいだろと思い、ツヴァイというところに申し込んで婚活というやつをしてみましたのでご報告いたします。興味のある方は参考にしていただければと思います。<br />
<br />
<h4>
金の話</h4>
大体1年間で30万円払えばOKです。そうすると年間72人だか48人は紹介してもらえることがシステム上保証されます。私はと言うと大体300人弱ぐらい紹介を受けることができました。<br />
なお2年目以降はまた30万円が必要になりますので、短期決戦をおすすめします。なお後述しますがどっちにしろ2年以降続けることはあまりないと思います。<br />
<br />
<h4>
ツヴァイの話</h4>
個人的にイオンが好きなのでツヴァイにしたのですが、どうやら婚活業界的には以下のようなヒエラルキーが構成されているようです。<br />
<br />
<pre>無料アプリ
↓
Omiai, Match, ゼクシィなどの有料アプリ(1万円〜)
↓
楽天オーネット(15万円〜)
↓
ツヴァイ(30万円〜)
↓
パートナーエージェント(40万円〜)</pre>
<br />
大変身も蓋もない暴言を申しますと、上のものほど婚活とは名ばかりの出会い系アプリで、下の方になるほど今度はアラフォーの行き遅れが滞留する阿鼻叫喚地獄となっております。このことは各社の資料を取り寄せてみて平均会員年齢などを調査すれば分かるかと思います。ツヴァイはだいたい35~40歳、パートナーエージェントになると40〜50歳がボリュームゾーンとなっていました。より上位のサービスで駄目だった人が下に流れてくるという構造があるのかもしれません。<br />
<br />
大変な暴言を申し上げてしまいましたが、やはり高い額を払うだけあって、下のサービスになればなるほど入会審査も厳格で(市町村が発行する独身証明書や2年分の源泉徴収票、大学卒業証書を提出する必要があります)、きちんとした担当の方がついてくださってアドバイスを受けたり、相手の女性の方も本気度が高いです。実際に1年間やってみてツヴァイは本当に大満足でしたのでおすすめしておきます。<br />
<br />
<h4>
婚活の話</h4>
皆さんおそらく「本当に出会えるのか?」とか「居たとしてもアラフォーの嫁ぎ遅れとサクラばかりなのでは?」とか「自分の趣味に合う人なんているわけがない」とか色々不安があるかと思いますが、断言します。<b><span style="color: red;">そんなことは一切ありません。普通に魅力的な方ばっかりです。</span></b><br />
<br />
私のような海外のPCゲームばっかりやってて普通の女の子の話題なんぞ何一つわからないおっさんですら、1年間で300人近く紹介してもらえまして、1/5以上の方には普通にお付き合いの連絡を申し込み、連絡先を交換して実際にお会いしたのが1/10程度で、最終的に趣味がバッチリ合う理想的な女性の方と付き合う事ができました。私自身ビックリです。年間30万円出せて本気で結婚したい方は即座に試す価値があると断言します。<br />
<br />
あとはもう一つ気になる、どういう男が人気があるか、ですが、周りの様子を見てみたところ、汚くなくて臭くなくて笑顔が気持ち悪くなければ見た目に関してはどんな人でも全く何の心配もありません。ただし汚いのと臭いのと笑顔が気持ち悪いのだけは即座にアウトで門前払いだったのでそこだけは徹底してください。あとやっぱり身長が高くて体格が良くて声が低い人のほうが(誤差の範疇程度ですが)人気があった気がします。<br />
<br />
職業はやはり医者と公務員から順番に売り切れます。付き合ってた彼女もおっしゃってましたが、女性は驚くほど現実を見ます。ただ当たり前ですが医者と公務員じゃないと駄目ということは一切ないのでご安心ください。<br />
<br />
むしろそんなことより大事なのは相手を思いやる気持ちをいついかなる瞬間も忘れないことだなぁと思う次第です。<br />
<br />
<h4>
結果の話</h4>
そして皆さん気になる結果の方はですが、四ヶ月ほど付き合った彼女に私の不徳のなすところから<b><span style="color: red;">見事にフラれまして</span></b>、私が伴侶を持つに値しない人徳のない男であるということが完全に証明された形となりました(´・_・`)<br />
これに懲りて来年以降は変な欲を持たずに良いカルマを積んでいきたいと思います(´・_・`)<br />
<br />
<h3>
3. 焼肉をおごりました</h3>
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjuItEea3MKoL2a1v3bjnoSOHheUjUcg9-VGgjPI6lsVkbt6yjdA3fX80KuoUHblRLXgQI4hfxLQ39GJES44TQZ3GmhNvOEQJe3ZTG5D2A7p6bWbS8xFpdO_94s4w_00fcu9yLroHytk_M/s1600/IMG_9591.jpg" imageanchor="1"><img border="0" height="480" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjuItEea3MKoL2a1v3bjnoSOHheUjUcg9-VGgjPI6lsVkbt6yjdA3fX80KuoUHblRLXgQI4hfxLQ39GJES44TQZ3GmhNvOEQJe3ZTG5D2A7p6bWbS8xFpdO_94s4w_00fcu9yLroHytk_M/s640/IMG_9591.jpg" width="640" /></a><br />
<br />
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEia7IaXZhOhiY4lR3okznwvJVBfBqOyN9dTPIp2Qpa4qGyqgaPTbS0yOrmO2MOma-xbp_QXlQuuVViotTWxiiv3F4LLXT1OMTH6P3JOPidv3J8XuOJYby236yTOl0Cuj7EbPUxIRNH2jk0/s1600/IMG_9593.jpg" imageanchor="1"><img border="0" height="480" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEia7IaXZhOhiY4lR3okznwvJVBfBqOyN9dTPIp2Qpa4qGyqgaPTbS0yOrmO2MOma-xbp_QXlQuuVViotTWxiiv3F4LLXT1OMTH6P3JOPidv3J8XuOJYby236yTOl0Cuj7EbPUxIRNH2jk0/s640/IMG_9593.jpg" width="640" /></a><br />
<br />
シャトーブリアン、大変美味しゅうございました。<br />
<br />
以上、よろしくお願いします。<br />
明日の担当は「お前、誰よ」で一世を風靡したイアン・ルイスさんです。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-15029109757913441512016-11-17T19:03:00.003+09:002016-11-17T19:04:08.063+09:00UITouchGestureRecognizer をやめて UILongPressGestureRecognizer や UIButton を使ってみる適当なUIViewにUITouchGestureRecognizerを貼ってタッチアクション可能にするという実装を行うことがありますが、UITouchGestureRecognizerはUIButtonと違ってタッチして離すタイミングではなくタッチした瞬間に反応してしまうためユーザビリティ的に望ましくない場合が結構あります。<br />
<br />
そんなときはUILongPressGestureRecognizerを代わりに使ってみましょうと言う話です。<br />
<a href="http://stackoverflow.com/questions/12830547/how-to-implement-touch-up-inside-in-touchesbegan-touchesended">http://stackoverflow.com/questions/12830547/how-to-implement-touch-up-inside-in-touchesbegan-touchesended</a><br />
<br />
UILongPressGestureRecognizerというと、長押しして何かコンテキストメニューを表示する際に使うイメージが強いものですが、minimumPressDurationを0.01など十分に短く取ると、事実上タップとほとんど変わらないユーザビリティを得ることができます。さらにUITouchGestureRecognizerとは異なりタッチを認識した後に離すまでの間のイベントをUILongPressGestureRecognizer.stateプロパティ経由で監視することができます。その為例えば<br />
<ul>
<li>タッチされている間は見た目を変える</li>
<li>タッチを話したときにアクションを実行する</li>
</ul>
といった味付けが簡単にできます。<br />
<br />
しかしもっと簡単に透明なUIButtonを作って目的のUIViewの上に重ね、そちらにタッチをハンドリングさせたほうがより綺麗で簡単にボタンらしい挙動を再現できたので別にUILongPressGestureRecognizerを使う必要はなかったかもしれません(´・_・`) まぁこういう使い方もあるんだよという参考になれば幸いです。akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-77127789767506259142016-11-17T18:51:00.003+09:002016-12-01T11:37:26.195+09:00ビルドが遅いのをどうにかこうにかやりくりする方法すでに皆様痛感されていると思いますが、Swiftはビルドがとんでもなく遅いです。正確に言うと、Swiftのビルドが遅いのに加え、clang moduleのビルドが遅いため、frameworkの数が増えてくるとビルドにかかる時間がどんどん破滅的なことになってきます。<br />
<br />
Swift & dynamic framework導入前はアーカイブ込みで10分以下だったビルド時間が、現在は20分を超えてしまっており、更に今後もSwiftコードの増加ととframeworkの増加に従ってビルド時間が伸びることが予想されます。そこでなんとかしてこの時間を短縮しようと思って調べてみました。<br />
<br />
<h3>
そもそもビルド時間が伸びる原因</h3>
<br />
<ul>
<li>Swiftのビルドが遅い、特に型推論が遅い</li>
<li>frameworkのビルドが遅い</li>
<li>Xcode上でcleanを実行するとプロジェクト内でビルドしているのframeworkが全てビルドし直しになり分割ビルドの恩恵が得られない</li>
</ul>
<br />
それぞれ詳しく見ていきます。<br />
<br />
<h4>
Swiftのビルドが遅い、特に型推論が遅い</h4>
以下の記事が大変参考になります。<br />
<a href="https://thatthinginswift.com/debug-long-compile-times-swift/">https://thatthinginswift.com/debug-long-compile-times-swift/</a><br />
Swift 3.0になって随分と高速化したため、型推論のせいで異常に時間がかかるという自体は減りましたが、それでもObjective-Cのコードと比べると10倍以上時間がかかることが多々あります。型を明示すれば高速化に寄与できますが、そうするとSwiftのせっかくの旨味が減るため悩ましいところです。<br />
<br />
<h4>
frameworkのビルドが遅い</h4>
<div>
dynamic framework = clang moduleのビルドはコードのコンパイル以上の作業を多分に含むため、分割すれば分割するほどそのオーバーヘッドが増えて無駄に時間がかかることになります。<br />
<div>
<br />
<h4>
Xcode上でcleanを実行するとプロジェクト内でビルドしているのframeworkが全てビルドし直しになり分割ビルドの恩恵が得られない</h4>
</div>
</div>
<div>
個人的に最悪なのがこれだと思います・・・せめて特定のコードに対してのみcleanできれば良いのですが、Xcodeで各種トラブルを避けるためにはcleanは必須、しかしながらcleanを実行すると全てのframeworkがビルドし直しとなって分割ビルドの恩恵が得られません。</div>
<div>
<br /></div>
<div>
特にCocoaPodsを使っていると全てのpod管理下のframeworkが再ビルドとなるため大変効率が悪いです。</div>
<div>
<br /></div>
<div>
これらを踏まえて対処法を考えてみました。と言っても大したことは出来ないのですが、</div>
<div>
<br /></div>
<div>
<ul>
<li>定期的にビルドタイムをチェックする環境を用意する</li>
<li>CocoaPodsをやめてcarthageを使用する</li>
<li>microframeworkをやめる、どうしてもmicroframeworkを行うのであればcarthageと併用する</li>
</ul>
</div>
<div>
<br /></div>
<div>
ビルドタイムのチェック方法は <a href="https://thatthinginswift.com/debug-long-compile-times-swift/">https://thatthinginswift.com/debug-long-compile-times-swift/</a> に記載がありますが、<code>Other Swift Flags</code>に<code>-Xfrontend -debug-time-function-bodies</code>を追加してビルドし、ビルドレポートを見ればOKです。結果は<code>pbpaste | egrep '\.[0-9]ms' | sort -t "." -k 1 -n | tail -10</code>などで整形すると見やすくなります。100msを超えている関数があれば手を打つ必要があるかもしれません。</div>
<div>
<br /></div>
<div>
CocoaPodsですが、手元で全てのframeworkをビルドしようとするためcleanした際の無駄が多いです。carthageを使えば最初にビルドを一回行って後はそれを参照して使いまわすだけ、という風にできます。ただしcarthageは最初のビルドがやたら長く、CocoaPodsを使った場合より5倍以上遅い?と言う問題があるのでその点は注意が必要です。</div>
<div>
<br /></div>
<div>
microframeworkを辞める論についてはおそらくいちばん賛否両論議論があるところかと思いますが、少なくとも殆どの場合、開発中はコードを使いまわす回数と時間よりXcodeをcleanしてビルドし直す回数と時間のほうが支配的であると信じています。Xcodeがちゃんとしてさえいればベストプラクティスに従ったほうが良いのでしょうが、そのようなことは伝統的にないので、ベストプラクティスにこだわらずビルドを高速化するために一つのframeworkにまとめるのも悪くはないのではと思っています。</div>
<div>
ただしcarthageと併用できるのであれば、一度ビルドしたmicroframeworkを使いまわすメリットが享受できるので良いかもしれません。その場合、microframework側はコードベースがある程度安定している必要があります。さもなければ何度も何度もcarthage updateを走らせ直して結局時間がかかることになります。</div>
<div>
<br /></div>
<h3>
結論</h3>
<div>
いろいろ考えてみましたが、頑張ってビルド高速化するより、新しいビルドマシンを増設したほうが結果的に良いかと思います\(^o^)/<br />
<br />
<h3>
2016/12/01 追記</h3>
</div>
<div>
Qiitaにより詳細で素晴らしいまとめがあったのでそっちを見ておいたほうが良い気がします(´・_・`)</div>
<div>
<a href="http://qiita.com/you_matz/items/e95f30023eccc8d96357">Xcode8でビルドが遅すぎるのを解消した件(Swift パフォーマンス改善)</a></div>
<div>
<br /></div>
<div>
全体的に見て型キャストとかワンライナー(とそれを誘発する演算子等)による途中の型推論が劇的に遅い原因のようですね。まぁ来年ぐらいまでにはclangが賢くなって直ってるのではないでしょうか・・・多分・・・</div>
<div>
<br /></div>
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-88496885682516469672016-11-14T20:23:00.003+09:002016-11-14T20:28:41.041+09:00UNNotificationAttachment.init(identifier: String, url: URL, options: [AnyHashable : Any]? = nil)に拡張子のないURLを渡す際の注意iOS 10で導入されたUserNotifications.frameworkは大変便利ですが、落とし穴が早速幾つか見つかっておりますのでご紹介いたします。<br />
<a href="http://qiita.com/himara2/items/dcfcc30b550c3304d86a">参考: iOS 10で画像つきのNotificationを配信する - Qiita</a><br />
<br />
表題にあります<code>UNNotificationAttachment.init(identifier: String, url: URL, options: [AnyHashable : Any]? = nil)</code>ですが、第二引数のurlに画像や動画などのメディアのURLを渡してUNNotificationContentに付与すると言う使い方をするのが一般的かと思います。<br />
<br />
このときどうやらUserNotifications.frameworkはurlのpathExtension、要するに拡張子の情報を利用してメディアの種類を判別しているようで、以下のような状況が発生すると<span style="color: red;"><b>実機でのみ</b></span>UNNotificationAttachment.initの実行が失敗してnilが返却されます。<br />
<br />
<ul><li>第二引数のurlに拡張子が付与されておらず、かつoptionsに<code>[UNNotificationAttachmentOptionsTypeHintKey: "<当該URLのメディアの適切なMIME Typeを表す文字列>"]</code>が指定されていない場合。</li>
<li>または間違った拡張子やMIME Typeが指定されている場合。</li>
</ul><br />
なおシミュレータでは問題ありません。いきなり実機に持っていって困るということが有るかと思いますので十分ご注意ください。<br />
<br />
対処法は2つあります。<br />
<br />
<ol><li>URLに適切な拡張子を付与する</li>
<li>optionsに<code>[UNNotificationAttachmentOptionsTypeHintKey: "<当該URLのメディアの適切なMIME Typeを表す文字列>"]</code>を付与する</li>
</ol><br />
何れにせよURLが指し示すメディアの種類が正確にわかっていないと問題になるため、その点は注意が必要です。<br />
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-56364563291875992262016-10-04T15:34:00.000+09:002016-10-04T15:34:51.435+09:00Q. ATS を Debug ビルドでだけ無効にしたいのですが...<h3>
A. こちらの内容に従えば一発です。</h3>
<div>
<a href="http://stackoverflow.com/a/32704702">http://stackoverflow.com/a/32704702</a></div>
<div>
要するにBuild Phaseにビルドスクリプトを追加してそこでPlistBuddyを使って値を書き換えましょうと言う作戦です。</div>
<div>
<br /></div>
<h3>
もうちょっとマシな解説</h3>
<div>
<a href="http://stackoverflow.com/questions/1832157/setting-a-boolean-property-in-info-plist-from-a-user-defined-setting">http://stackoverflow.com/questions/1832157/setting-a-boolean-property-in-info-plist-from-a-user-defined-setting</a></div>
<div>
普段、Info.plistの設定内容をConfiguration毎に書き換えたい場合は、たいていよくやるのがxcconfigファイルを以下のように用意して、</div>
<code>APP_BUNDLE_NAME = MyApp_Dev</code><br />
<div>
でもってInfo.plistに対して</div>
<code>Bundle Display Name = ${APP_BUNDLE_NAME}</code><br />
<div>
こんな風に書いておけばビルド時に自動的にXcodeが環境変数による置換処理を行ってくれる、というのを使うのですが、まいったことにATSの設定を司るNSAllowArbitraryLoadsはString値ではなくBool値でして、このInfo.plistに対するXcodeの環境変数置換処理はString値(と、確かNumber値)にしか使えないというオチがあります。仕方がないので最初に説明した方法が最善になるかと思います。Info.plistファイルをConfiguration毎に用意する方法もありますが、大概そっちのほうが設定のメンテナンスが面倒になるのでオススメしません。</div>
<div>
<br /></div>
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-74833482628874390542016-10-03T11:44:00.000+09:002016-10-03T16:47:16.065+09:00ATS の AVFoundationに対する 例外条項の注意点2016/10/03段階でのATSを有効にする際の覚書です。基本的には全てTLS 1.2以上をサポートするhttpsにしてしまえばOKですが、例外条項としてAVFoundationのStreamについてはhttpでもよいと例外条項が定められています。<br />
<br />
<a href="https://developer.apple.com/videos/play/wwdc2016/706/?time=324">https://developer.apple.com/videos/play/wwdc2016/706/?time=324</a><br />
<br />
これについては設定不要で自動的に適用されるようにWWDCのビデオ上では見えますが、実は落とし穴があり、この例外条項はiOS 10以降でのみしか適用されません。iOS 9については実はAVFoundationのStreamについてATSが完全に適用されているようで、httpの動画をStreamingしようとしても失敗してしまうので注意が必要です。<br />
<br />
2016/10/03 16:45 追記<br />
こちらXcode 7.3.1でビルドしたバイナリにて調査した内容ですので、SDKのバージョンが異なるXcode 8.0以降でビルドしたバイナリであればiOS 9.x系でもAVFoundationのStreamについてATSの例外条項が適用されてhttpで通信できるようになるかもしれません。ただ紹介した参照元のWWDCのビデオを見ても分かる通り、基本は例外を考えず全てhttps化するのが良いとされているため、ビデオストリームについても可能であれば全てhttps化するほうがトラブルがないかと思います。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-27388863464179257962016-09-11T21:21:00.002+09:002016-09-11T21:22:26.809+09:00Parse.com が潰れたので新しい mBaaS を探す旅に出た定番のmBaaSとして人気だったParse.comが終わってしまうことが確定してしまった2016/09/11現在、次はどのようなmBaaSを選ぶのが良いのか調べてみました。なおどのプロダクトも全然試せていないので実戦での詳細な評価についてはできかねます。むしろこれを読んだ皆さんに果敢にトライしていただきたい!\(^o^)/<br />
<br />
<h3>
Niftyクラウド mobile backend</h3>
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgWkX2I5Iu0ezJ_5MJXF4KnVIRX1WRRShBiIaTi2NE1-gHS-w5dP9JSM3ptIO7MQh0C3TMa7qq7Xc8DkQpidGiNi1gy-gS6g9zM2Vs2Rz7b5jEoF8umD3aVCiC64BV_lNyjDD2PLII47ls/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.15.56.png" imageanchor="1"><img border="0" height="255" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgWkX2I5Iu0ezJ_5MJXF4KnVIRX1WRRShBiIaTi2NE1-gHS-w5dP9JSM3ptIO7MQh0C3TMa7qq7Xc8DkQpidGiNi1gy-gS6g9zM2Vs2Rz7b5jEoF8umD3aVCiC64BV_lNyjDD2PLII47ls/s400/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.15.56.png" width="400" /></a><br />
<a href="http://mb.cloud.nifty.com/">http://mb.cloud.nifty.com</a><br />
<br />
<ul>
<li>国内だとここ</li>
<li>ちょっと前にひとりぼっち惑星がここ使ってるといって有名になった</li>
<li>正直機能的にはダメだと思ってる</li>
<ul>
<li>その話題になったブログ記事の内容が、データストアにトランザクションに相当する処理を新機能として導入しました、とかDBキーによる高速ルックアップを実装しました、とかそういう次元の、あまりにも最初から存在して当然と思われる機能を新規実装として誇っているような内容だったため</li>
<li>逆に言えば今後もこの調子で継続的に改善してくれるかも</li>
</ul>
<li>Niftyが社命賭けてるっぽいしそう簡単には潰さないと信じてる</li>
</ul>
<br />
<h3>
Google Firebase</h3>
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjgXYdo6ii06nu0rpwPeQF322MfUSBSfdMsoAMvdFmRM4qei1PMsXbYo_vGgrBLijQ2UZEUZnGF9y8XJyPQSXeBZLYp0I63L3m1TLl2bOtqOk47OiiBT8jph8VUFcCEF1nqBGADzYxHulc/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.07.21.png" imageanchor="1"><img border="0" height="250" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjgXYdo6ii06nu0rpwPeQF322MfUSBSfdMsoAMvdFmRM4qei1PMsXbYo_vGgrBLijQ2UZEUZnGF9y8XJyPQSXeBZLYp0I63L3m1TLl2bOtqOk47OiiBT8jph8VUFcCEF1nqBGADzYxHulc/s400/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.07.21.png" width="400" /></a><br />
<a href="https://firebase.google.com/">https://firebase.google.com</a><br />
<br />
<ul>
<li>なんといってもGoogle様ブランドがある</li>
<li>Analyticsが非常に素晴らしい、GAをリプレイスして、Analyticsのためだけに導入する価値がある</li>
<li>一方でiOS/Android/Webプラットフォーム全てで同一の概念を利用できるように、各プラットフォーム独自の箇所をラップして激しく潰しているように見える</li>
<ul>
<li>特にFCMと呼ばれるPush通知に相当する箇所についてはAPNSが元の影も形も見えないぐらい激しくラップされていて操作できないため、例えば今の時期だとiOS 10向けのUserNotifications.frameworkに対応させるのが実質不可能とかそういうデメリットがある</li>
<li>iOS側が割を食いそうな不安はある</li>
<li>とはいえ調べてみたところiOS 9からのUniversal LinkはApple仕様に従ってきちんと対応するようになっていて、Firebase側は全く無視する気はなく、対応が遅れるのを許容できるのであれば大丈夫そう</li>
</ul>
<li>Googleだし気合入れてるみたいだしそう簡単には潰れない・・・はず・・・</li>
<ul>
<li>とはいえ最近のGoogleは不採算プロダクトを平然と潰すので油断ならない</li>
</ul>
</ul>
<br />
<h3>
backendless</h3>
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgTN5tW0LSZGHax1_lM2j3udpB9h1gHVh49LSfLjfpZS6_Nraw9F07UMd3XQ7N3-EBTIO73xhYGIPrKrebk-qzfeOfUX2LhyphenhyphenUapyLWi7azsT0zHr1Zo6Ij7ejOBJd5Eexg9YYDC0Qrqbzs/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.06.44.png" imageanchor="1"><img border="0" height="167" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEgTN5tW0LSZGHax1_lM2j3udpB9h1gHVh49LSfLjfpZS6_Nraw9F07UMd3XQ7N3-EBTIO73xhYGIPrKrebk-qzfeOfUX2LhyphenhyphenUapyLWi7azsT0zHr1Zo6Ij7ejOBJd5Eexg9YYDC0Qrqbzs/s320/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-09-11+21.06.44.png" width="320" /></a><br />
<a href="https://backendless.com/">https://backendless.com</a><br />
<br />
<ul>
<li>パッと見一番Parse.comっぽい</li>
<li>トップページからして「Migrating from Parse? Welcome to Backendless!」だし</li>
<li>一番Parseっぽくて下手にラップするようなことをしておらず、かつ必要な物は全部あるので最も個人的には好感</li>
<ul>
<li>Firebaseみたいに激しくラップされすぎる構造は、例えるならRuby on RailsやPython Djangoのような感覚があってどうも個人的には好きになれない</li>
</ul>
<li>ただしParseの二の舞いのように潰れる可能性もまた否定出来ない</li>
</ul>
<br />
<br />
<h3>
まとめ</h3>
どこも潰れるリスクがワンチャンありそうですね(´・_・`)<br />
ご利用はご計画的に\(^o^)/<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-76496798408535315522016-09-11T20:56:00.001+09:002016-09-11T21:32:24.905+09:00iOSのフォントのお話最近フォント周りについて深く掘り下げる機会がありましたので、その際のメモを残しておこうと思います。かなり読む人置いてけぼりな中身になってますが、フォントを詳しく触り始めるとなるほどーとためになる(と思う)のでどうかご了承ください(´・_・`)<br />
<br />
<h3>
UIFontのプロパティについて</h3>
UIFontにはフォントに関する数値を表すプロパティが存在します。いろいろありますが、もっとも重要なのは以下に列挙するプロパティです。<br />
<br />
<ul>
<li>pointSize</li>
<li>lineHeight</li>
<li>ascender</li>
<li>descender</li>
<li>leading</li>
<li>capHeight</li>
</ul>
<br />
以下の画像を見ると非常にわかりやすいかと思います。<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://developer.apple.com/library/ios/documentation/StringsTextFonts/Conceptual/TextAndWebiPhoneOS/Art/textpg_intro_2x.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="329" src="https://developer.apple.com/library/ios/documentation/StringsTextFonts/Conceptual/TextAndWebiPhoneOS/Art/textpg_intro_2x.png" width="640" /></a></div>
<br />
<br />
参照元: <a href="https://developer.apple.com/library/ios/documentation/StringsTextFonts/Conceptual/TextAndWebiPhoneOS/CustomTextProcessing/CustomTextProcessing.html">https://developer.apple.com/library/ios/documentation/StringsTextFonts/Conceptual/TextAndWebiPhoneOS/CustomTextProcessing/CustomTextProcessing.html</a> および <a href="http://stackoverflow.com/a/35922853">http://stackoverflow.com/a/35922853</a><br />
<br />
これを踏まえて各プロパティを解説すると以下のとおりになります。<br />
<ul>
<li>pointSize</li>
<ul>
<li>フォントのサイズを表します。</li>
<li>ほとんどすべてのケースでpointSize = ascender + descenderとなり、すなわちpointSizeは上端と下端に空白を含まない純粋なフォントの高さだけを表す数値であると言っても良いと思います。</li>
<ul>
<li>上下に空白を作らない高さでフォントを描画したい場合はこの高さを使うと良い、と言い切って間違いないと思います、ごく一部の変なフォント以外。</li>
</ul>
</ul>
<li>lineHeight</li>
<ul>
<li>フォントの上部末端(上端空白が存在する場合はそれも)から下部末端(line gapも存在するなら含む)までの完全な高さを表します。</li>
<li>飾り文字などではないまっとうなフォントであれば、必ずこのlineHeightの中に文字の高さが収まるようになっています。</li>
<ul>
<li>一部飾り文字のあるフォントについてはfont leadingを使って下部末端のline gapを表現していることがあります。</li>
</ul>
<li>pointSizeとの最も大きな違いは、上端と下端にline gapに相当する空白を持っています。特にシステムフォントで顕著にそれが見られます。詳しくは後述。</li>
<li>実際にフォントが画面上にレンダリングされるときはこのlineHeightがすべての基準になっているケースが多いです。</li>
</ul>
<li>ascender</li>
<ul>
<li>全てのアルファベットの文字の最大高さを表します。</li>
<li>この高さまで描画されるのはごく一部の文字だけで、殆どの文字はcapHeightまでの高さしか使いません。詳細はこの後のcapHeightを参照してください。</li>
</ul>
<li>descender</li>
<ul>
<li>アルファベットのBaselineより下に描画される文字、例えばj p q yなどの下付き部分の高さを表します。</li>
<li>日本語の文字でこの領域に描画される文字は私が知っている限りでは存在しません。</li>
<li>UIFontが返す値は常にマイナスになるので気をつけてください。</li>
<li>CTFontが返す値は常にプラスです。</li>
<ul>
<li>めんどくさいので常にfabs()を通すといいと思います。</li>
</ul>
</ul>
<li>leading</li>
<ul>
<li>Baselineからフォントの下部末端(line gapを含む)までの高さを表します。</li>
<li>上記の図ですとline gapの部分だけがleadingとなっていますがヒラギノフォントで試した限りは誤りです。Baselineから下全てが対象になっているようです。</li>
<li>lineHeightに含まれている下部末端のline gapとは「全く別の値が付与されていることがあります」。</li>
<ul>
<li>例えば飾り文字のあるフォントについて、lineHeightよりもフォントの描画が大きくなるぐらい下に飾りが伸びていたとしても、leadingの値はそこまで考慮して付与してあるので、usesFontLeadingのオプションを付与すると下の行にはみでなくなる、みたいなケースが存在します。</li>
</ul>
</ul>
<li>capHeight</li>
<ul>
<li>アルファベットの大文字の最大高さを表します。</li>
<li>日本語の文字は(特に漢字は)基本的にすべてcapHeightの高さに描画されます。</li>
<li>capHeightより上に描画されるのは、例えばÅ(Aウムラウト)のように修飾子が付いているアルファベットや、日本語だと濁点および【】隅付き括弧のような一部の文字が、capHeightよりも上に描画されます。</li>
</ul>
</ul>
特に覚えておきたいのがleadingというプロパティの働きです。UIKit, TextKit, Core TextにusesFontLeadingというような名前のオプションが存在するケースが多々ありますが(例えばNSAttributedString.boundingRect(with:options:context:)のoptionsとかですね)、これは何かというと、<br />
<blockquote class="tr_bq">
usesFontLeadingが指定されている場合、lineHeightの計算の際に常にUIFont.leadingの値が使用される。従って、もしattributed stringにparagraph styleを付与してlineSpacingに0を付与していたとしても、フォントにleadingの値が付与されていたら、それ以上に行の高さが低くならず必ず行間が開いてしまう。</blockquote>
というオプションになります。paragraph styleでlineSpacingやparagraphSpacingを0に指定しているにもかかわらずどうしても行間がゼロにならないとか、指定するより広がってしまうみたいな状況になった場合は、このfont leadingを疑ってみてください。<br />
<br />
ちなみにiOS 8以上については以下のとおりです。<br />
<br />
<ul>
<li>UILabelは常にfont leadingを使用しません。</li>
<li>UITextFieldは未調査ですがおそらく常にfont leadingを使用しません。</li>
<li>UITextViewはデフォルトfont leadingを使用しますが、UITextView.layoutManager.usesFontLeadingの値をfalseにすることで回避可能です。</li>
<li>その他のラベルパーツ(UIButton.titleLabelなど)については完全には調査できていませんが、UIButton.titleLabelに関してのみいうとiOS 9の地点ではfont leadingが常に「使用されてしまいます」。これはおそらくiOS 6以前のUILabelの挙動がそのまま残っているためではないかと思われます。同じUILabelクラスですが中身と挙動がUILabelと異なるため気をつけてください。このようなケースは他にもあると思われます。</li>
</ul>
<br />
<h3>
iOSビルトインフォントの問題点</h3>
<div>
幾つかのiOSビルトインフォントについては、先述のUIFontのプロパティの値に疑問がある数値が設定されているケースが有るようです。具体的にご紹介します。</div>
<h4>
1. ヒラギノフォントの場合</h4>
<div>
ヒラギノフォント、すなわちヒラギノ角ゴシックW3/W6、それからヒラギノ明朝W3/W6については以下の様なフォントの設定の特徴があります。</div>
<div>
<ul>
<li>leadingの値がきちっと付与されています。</li>
<li>常にpointSize == lineHeightになります。</li>
<li>おそらくバグだと思うのですが、descenderの値が本来必要な量の半分しか設定されていません。そのためヒラギノフォントを明示的に付与したラベルにg, j, y, qなどの下付き部分があるアルファベット文字を描画すると下が途中で千切れてしまいます。</li>
<ul>
<li>対処法としてはfont leadingを使用するか、または自力でdescender分だけラベルの高さを高くするか、明示的に指定してヒラギノフォントなんて使うのをやめてシステムフォントを使用してください。</li>
</ul>
</ul>
</div>
<h4>
2. システムフォントの場合</h4>
<div>
次にシステムフォントはどうなっているのかというと、以下の様な特徴があります。</div>
<div>
<ul>
<li>常にleadingの値がゼロになります。従ってfont leadingを使うモードで描画しても使わないモードで描画しても全く同じ結果になります。</li>
<ul>
<li>iOS 8はHelvetica Neue, iOS 9以上はSan Franciscoが採用されていますが、いずれのケースでも全く同様の設定になっています。</li>
</ul>
<li>pointSizeよりlineHeightが高く、ちょうどその差分の分だけ自動的にline gapとして行間が開くような設定になっています。</li>
<ul>
<li>要するに簡単にいえば、font leadingを使わないで、かつattributed stringでlineSpacingやparagraphSpacingをゼロに設定したとしても、必ず行間がちょっとだけ空いてしまいます。まずやることはないと思いますが、正確に行間ゼロを作るのは著しく困難です。</li>
</ul>
</ul>
</div>
<h3>
UIFontのpointSizeについて</h3>
<div>
UIFont.pointSizeについて、みなさん普段から何気なしに17とか14とか値を指定されているかと思いますが、このpointSizeに指定する値の単位って気にされていますでしょうか?このフォントのpointSizeとはPostScriptポイントないしDTPポイントと呼ばれるものらしいです。一般的に1PostScriptポイント=1/72インチとされています。</div>
<div>
<br /></div>
<div>
ではこの単位はiOS上で使われている論理座標系のpt単位とは全く違うものなのかというと、どうやら大昔のMacintoshのころからの名残で、macOS/iOS上の論理座標系の1pt=1PostScriptポイントとなるように定められていて、これは現在に至るまで常に維持されているということらしいです。</div>
<div>
<br /></div>
<div>
要するに、UIFontのpointSize=iOS上での1ptと完全に一致します。このことはlineHeightが16のフォントを画面上にレンダリングすると論理単位16pt(2倍Retinaなら32pxになる)ことからも確認ができます。便利ですね。</div>
<div>
<br /></div>
<div>
参考元: <a href="http://qiita.com/usagimaru/items/9d39d105ad57bc73ea24">http://qiita.com/usagimaru/items/9d39d105ad57bc73ea24</a></div>
<br />
<h3>
UIFont.leadingのiOS 8以下での不具合</h3>
かいつまんで言うと、UIFont.leadingの値は、iOS 8以下のときに常にUIFont.lineHeightと同じ値を返します。これは"font leading"という単語が使われている文脈や文化圏によって意味が異なり、font leading = line heightとなる文化圏があったので誤用されてしまった、ということらしいです。しかしながら実際のiOSでは先述の通り「font leading = BaselineからフォントのLine gapを含む下端までの距離」と定義されていますから、このような値では困ります。<br />
<br />
対処法として、iOS 8以下の場合はCTFontを使ってleadingの値を取得してください。こちらにコードを用意しました。<br />
<a href="https://gist.github.com/akisute/3c3d162da73abd784525c9ed7859cda2">https://gist.github.com/akisute/3c3d162da73abd784525c9ed7859cda2</a><br />
<br />
なお、iOS 9以上はこの問題は発生しません。iOS 8なんてそろそろサポート切れるんで忘れていいかもしれませんが、念のため。<br />
<br />
<h3>
まとめ</h3>
デザイナーさんにフォント周りで行間調整がおかしいぞーとかベースラインがずれてるぞーとか突っ込まれまくっていて必死こいて修正方法を調べている時に、こんな記事を見つけました。<br />
<br />
<a href="https://www.raizlabs.com/dev/2015/08/advanced-ios-typography/">https://www.raizlabs.com/dev/2015/08/advanced-ios-typography/</a><br />
<br />
Stop Saying "No" to Designers.<br />
このサイトには英字フォントをBaselineではなくcapHeightの位置で上揃えにするべく奮闘した一人の漢の話が書かれています。おお、海外にも私と同じ境遇の漢が居たのです。<br />
<br />
我々もデザイナーにNOと言わないタイポグラフィを実現させたいですね\(^o^)/akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-85545303500078766522016-06-30T23:36:00.002+09:002016-06-30T23:40:11.085+09:00Auto Layout と Manual Layout を混載させるときに役立つ UIView.translatesAutoresizingMaskIntoConstraints プロパティの話Auto LayoutがiOS 6で導入されてはや4年、未だによく理解していなかった挙動に UIView.translatesAutoresizingMaskIntoConstraints があります。このプロパティは自分がプログラムコード上で生成したviewをAuto Layoutするときにfalseにする必要があるものということで皆様記憶されているかと思うのですが、具体的にこのプロパティは何をやっているのかが個人的に全く謎でした。それが今日一つ謎が解けましたのでここに共有させていただきたいと思います。<br />
<br />
UIView.translatesAutoresizingMaskIntoConstraintsの値がtrueのときとfalseのときの違いについて以下に記載します(iOS 8以上で確認しています)。<br />
<br />
<ul>
<li>trueのとき</li>
<ul>
<li>対象のviewのframe、すなわちx, y, width, heightの4つの要素をview.frame, view.bounds および view.center プロパティから直接操作することが可能になります。これはAuto Layoutが導入される以前のiOSの世界と同じ状態です。この挙動をAuto Layoutとマッチさせるため、対象のviewのx, y, width, heightの4要素を指定された値に固定するようなAuto Layout Constraintsが自動的にシステムによってviewに挿入されます。この自動的に挿入されるAuto Layout Constraintsのpriorityは常に1000 (Required)になります。</li>
</ul>
<li>falseのとき</li>
<ul>
<li>対象のviewのframe、すなわちx, y, width, heightの4つの要素はすべてAuto Layoutエンジンが管理するようになり、view.frame, view.bounds および view.centerの値を直接書き換えても一切無視されるようになります。Auto Layout Constraintsが設定されていない場合、viewのframeはCGRect.zeroになります。</li>
</ul>
</ul>
<div>
<br />
プロパティの名前にAutoresizing Maskとか入っているのでてっきりAutoresizingの仕組みに影響している用に見えますが、実際には全く関係ありません。その証拠にAutoresizingMaskの値をどのように変化させても勝手にAuto Layout Constraintsが挿入されてしまいます。このプロパティはあくまで当該viewのframeを自動的に操作するようなAuto Layout Constraintsを挿入するか否かを決めるフラグとして覚えると良いでしょう。</div>
<div>
<br /></div>
<div>
さてこの挙動を覚えると何が嬉しいかと申しますと、Auto Layoutと非Auto Layoutを混載させるときに非常に役立ちます。こうすることで、特定のviewだけをframe手動操作で設定し、他のviewはAuto Layoutに任せるというような荒業が自由自在に可能になります。</div>
<div>
<br /></div>
<div>
具体例を見てみましょう。例えば以下の様なニュースを表示する画面を作ってみようと思います。</div>
<div>
<br /></div>
<div>
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEirVjdj6TG37qWNeNVIwfGFLA8KaN7MGIaV8fbEXF3t_ZnN9TJIsZmsgJCXA35G_Hi4FDWeU4ELpcfWzWOjxOTriT3aIBFMHuc9gh4I_FqHmEVrjomJzkq3X0Z8zVzwagJi1pmCFjtpGNg/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-06-30+22.42.07.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="640" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEirVjdj6TG37qWNeNVIwfGFLA8KaN7MGIaV8fbEXF3t_ZnN9TJIsZmsgJCXA35G_Hi4FDWeU4ELpcfWzWOjxOTriT3aIBFMHuc9gh4I_FqHmEVrjomJzkq3X0Z8zVzwagJi1pmCFjtpGNg/s640/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-06-30+22.42.07.png" width="385" /></a></div>
<br /></div>
<div>
<br /></div>
<div>
ここでこのnewsを表示するviewのframeは複雑なアニメーションをさせたいなどの理由で外部からマニュアルで設定したいが、viewの中身はauto layoutに任せたいというようなケースがあるかと思います。</div>
<div>
<br /></div>
<div>
というわけで普通にAuto Layoutで作ってみましょう。</div>
<pre>private func commonInitialize() {
self.translatesAutoresizingMaskIntoConstraints = true
self.backgroundColor = UIColor.white()
self.imageView = UIView()
self.imageView.translatesAutoresizingMaskIntoConstraints = false
self.imageView.backgroundColor = UIColor.green()
self.addSubview(self.imageView)
self.titleLabel = UILabel()
self.titleLabel.translatesAutoresizingMaskIntoConstraints = false
self.titleLabel.text = "factorio alpha 0.13 has been released!"
self.titleLabel.font = UIFont.preferredFont(forTextStyle: UIFontTextStyleTitle1)
self.titleLabel.numberOfLines = 2
self.addSubview(self.titleLabel)
self.articleLabel = UILabel()
self.articleLabel.translatesAutoresizingMaskIntoConstraints = false
self.articleLabel.text = "In 0.13 we have the new multiplayer matching server and server browser. This will let you find games of people online join your friends and other stuff. Server games are published to the server and clients can browse existing games. The first thing you will notice is the new multiplayer menu. When you click on 'Browse Public games' you will be asked to log in to your factorio account."
self.articleLabel.font = UIFont.preferredFont(forTextStyle: UIFontTextStyleBody)
self.articleLabel.numberOfLines = 0
self.addSubview(self.articleLabel)
let views: [String: AnyObject] = ["imageView": self.imageView,
"titleLabel": self.titleLabel,
"articleLabel": self.articleLabel]
self.addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "H:|[imageView]|", options: [], metrics: nil, views: views))
self.addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "H:|-10-[titleLabel]-10-|", options: [], metrics: nil, views: views))
self.addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "H:|-10-[articleLabel]-10-|", options: [], metrics: nil, views: views))
self.addConstraints(NSLayoutConstraint.constraints(withVisualFormat: "V:|[imageView]-10-[titleLabel]-10-[articleLabel]|", options: [], metrics: nil, views: views))
self.imageView.addConstraint(NSLayoutConstraint.init(item: self.imageView, attribute: .height, relatedBy: .equal, toItem: self.imageView, attribute: .width, multiplier: 0.66, constant: 0))
}</pre>
<div>
<br /></div>
<div>
<br />
しかしながらこのコードはAuto Layout Warningが発生してしまいます。<br />
<br /></div>
<pre>2016-06-30 23:24:12.050906 AutoLayout[1725:80607] [LayoutConstraints] Unable to simultaneously satisfy constraints.
Probably at least one of the constraints in the following list is one you don't want.
Try this:
(1) look at each constraint and try to figure out which you don't expect;
(2) find the code that added the unwanted constraint or constraints and fix it.
(Note: If you're seeing NSAutoresizingMaskLayoutConstraints that you don't understand, refer to the documentation for the UIView property translatesAutoresizingMaskIntoConstraints)
(
"NSAutoresizingMaskLayoutConstraint:0x7fd60ae1db90 h=--& v=--& AutoLayout.Example2View:0x7fd60ad35b70.width == (active)",
"NSLayoutConstraint:0x7fd60ac360a0 H:|-(10)-[UILabel:0x7fd60ad3c6b0'factorio alpha 0.13 has b...'] (active, names: '|':AutoLayout.Example2View:0x7fd60ad35b70 )",
"NSLayoutConstraint:0x7fd60ac36380 H:[UILabel:0x7fd60ad3c6b0'factorio alpha 0.13 has b...']-(10)-| (active, names: '|':AutoLayout.Example2View:0x7fd60ad35b70 )"
)</pre>
<div>
<br />
これは先程のUIView.translatesAutoresizingMaskIntoConstraintsについての説明を元にすると以下のように解釈できます。<br />
<br /></div>
<div>
<ol>
<li>UIView.translatesAutoresizingMaskIntoConstraintsがtrueに設定されていることにより、このviewにはwidth=frame.size.widthになるようなAuto Layout Constraintsが自動的に設定されている。</li>
<li>このviewにはさらに "H:|[imageView]|"となるようなAuto Layout Constraintsが設定されている。これはimageViewを横幅いっぱいに表示するため。</li>
<li>しかしながらこのような設定を行うと、imageViewが親となるviewの横幅を自分の横幅に合わせて引っ張ろうとするConstraintsが定義されてしまうので、1. で自動的に設定されたConstraintsと衝突してしまう。</li>
<li>結果としてwarningが発生する。</li>
</ol>
<div>
これを回避してやるにはいくつか方法があります。<br />
<br />
<ol>
<li>"H:|[imageView]-(0@999)-|"のように設定することで、右側ないし下側のpriorityを999に下げる。こうすることによってUIView.translatesAutoresizingMaskIntoConstraintsによって設定されるConstraintsのpriorityが勝つためワーニングは発生しなくなる。</li>
<li>両側を引っ張るようにvisual formatを使って設定するのをやめて、view.x=imageView.x, view.width=imageView.widthとなるようにConstraintsを付与する。</li>
</ol>
<div>
例えば2. のケースはiOS 9以降であればNSLayoutAnchorを使って簡単に設定ができます。<br />
<br /></div>
<pre>self.imageView.leadingAnchor.constraint(equalTo: self.leadingAnchor).isActive = true
self.imageView.widthAnchor.constraint(equalTo: self.widthAnchor, multiplier: 1.0).isActive = true</pre>
</div>
</div>
<div>
<br />
これでAuto LayoutとManual Layoutをより自由自在に混載させることが可能になると思います。</div>
akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-33864328531607147762016-06-19T20:58:00.001+09:002016-06-19T20:58:14.791+09:00ゼロから始めるマインクラフトのサーバー構築の話突然なのですが、最近いまさらながらマインクラフトに大ハマリしておりまして。きっかけは友人のマルチプレイサーバーでプレイしたことだったのですが、やっぱり皆で遊ぶゲームは面白いなと再確認している次第です。マインクラフトはリリース直後、βが取れたぐらいに遊んで面倒でやめてしまったのですが、近頃はFeed The Beastと呼ばれるオールインワンのランチャーアプリが登場しており、こいつがMODの導入や管理も全て一発でやってくれるため面倒やストレスが一切なく楽しめます。<br />
<br />
で、しばらく遊んでいたのですが、友人が他のゲームにハマってしまったためマインクラフトのサーバーを立てなくなってしまいました。そこで私のほうで自分でマインクラフトのサーバーを立ててやろうと思い立ちまして早速今週末チャレンジしてみました。<br />
<br />
ちなみに私インフラに関しては大の苦手で、サーバーをゼロから構築するのは初めてだったりします。大昔新卒の頃にSIerの仕事でちょっと触ったりとか、その後の仕事でちょっと触ったりとかした程度。アプリケーションなら分かるんですけど(´・_・`)<br />
<br />
前置きはこのぐらいにして早速やってみましょう。<br />
<br />
<h3>
ホスティングサービスを選定</h3>
パッと思いつくところだとさくらVPSとAWSが有名ですが、今回は友人のおすすめでConoHaというVPSホスティングサービスを選んでみました。ほとんどさくらVPSと大差ないように見えたのですが、ConoHaのほうがさくらVPSと違い初期費用がかからないらしいです。新しいもの好きですしせっかくなのでConoHaを選択。<br />
<br />
サーバーのスペック等は以下のようにしました。CentOS 7.2に3コア2GBメモリで月1750円。マインクラフトのサーバであればこのぐらいの性能で大丈夫なはず。<br />
<div class="separator" style="clear: both; text-align: center;">
<a href="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhMsMVp7MNLB1Tq7qLmYJbo16AqJoTs3HbEFQKZFyHnKqn7Q8GWUGXyYkJLGJi0F30E4lw9oMwzfASBID8RHTWaKEv-CnN8JwNxIytW-TrRx3d6pZjmAo_0g9eNsgTCSn3TkgDbjpfTMuA/s1600/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-06-19+18.44.06.png" imageanchor="1" style="margin-left: 1em; margin-right: 1em;"><img border="0" height="417" src="https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEhMsMVp7MNLB1Tq7qLmYJbo16AqJoTs3HbEFQKZFyHnKqn7Q8GWUGXyYkJLGJi0F30E4lw9oMwzfASBID8RHTWaKEv-CnN8JwNxIytW-TrRx3d6pZjmAo_0g9eNsgTCSn3TkgDbjpfTMuA/s640/%25E3%2582%25B9%25E3%2582%25AF%25E3%2583%25AA%25E3%2583%25BC%25E3%2583%25B3%25E3%2582%25B7%25E3%2583%25A7%25E3%2583%2583%25E3%2583%2588+2016-06-19+18.44.06.png" width="640" /></a></div>
<br />
<br />
<h3>
とりあえずサーバーを起動してsshまで</h3>
構築したらすぐにインスタンスが起動します。ConoHaでは自動的に固定Public IPが1インスタンスに割り当てられるようで、この点固定IPを用意したりするのが何かと面倒なAWSに比べて楽でした。管理画面もすっきりまとまっていて使いやすく好印象です。キーペアもその場で同時にすぐ作成してくれて手間がかかりません。<br />
<br />
生成したキーペアをローカルの ./ssh 以下に保存して ./ssh/config を適当に設定してログイン。root@を付け忘れてssh接続要求に失敗するみたいなよくあるお約束のミスをしましたが、特に問題なくログインできました。<br />
<br />
ログインに成功したのを確認してから手持ちのドメイン (akisute.com) のサブドメインを用意してDNSを設定してやります。疎通して問題なし。<br />
<br />
<h3>
nginxを立てる</h3>
サーバーの情報などをウェブに公開したいので、nginxを利用してstatic htmlファイルをホスティングできるようにします。nginxのCentOS 7.2に対するインストール自体はこちらのページに従えばすぐにできます。<br />
<a href="http://nginx.org/en/linux_packages.html">http://nginx.org/en/linux_packages.html</a><br />
後は以下のビギナーズガイドに従って設定を行えば良いのですが、基本的には最初から全て必要そうな設定は用意されているので、/usr/share/nginx/html以下のindex.htmlを適当に編集してやるだけでオッケーです。<br />
<a href="http://nginx.org/en/docs/beginners_guide.html">http://nginx.org/en/docs/beginners_guide.html</a><br />
<br />
<h3>
firewalldを設定する</h3>
しかしながらnginxを立ち上げただけでは外からアクセスしてもつながりません。これはCentOS(に限らないですが、ほとんどのLinuxのディストリビューションについて)は何らかのファイアウォールがデフォルトで入っているためです。一般的にはiptablesが使用されているのですが、CentOS 7ではiptablesではなくfirewalldというやつが使われているらしいので、そちらを設定することにしました。<br />
<a href="http://urashita.com/archives/963">http://urashita.com/archives/963</a><br />
<a href="http://www.unix-power.net/centos7/firewalld.html">http://www.unix-power.net/centos7/firewalld.html</a><br />
<br />
httpサービスをpublicゾーンの設定に追加するだけで簡単に外からアクセスできるようになります。iptablesより簡単でいいですね。<br />
<br />
<h3>
Javaをインストールする</h3>
マインクラフトはJavaで動作するので最新版のJavaが必要です。Javaには大きくOpenJDKとOracle Javaがありますが、個人的にはOracle Javaのほうが速そうなのでそちらを使うことにしました。以下のページで紹介されているとおりにすれば簡単にインストールできるかと思います。<br />
<a href="https://www.digitalocean.com/community/tutorials/how-to-install-java-on-centos-and-fedora">https://www.digitalocean.com/community/tutorials/how-to-install-java-on-centos-and-fedora</a><br />
<br />
<h3>
minecraft-serverを立てる</h3>
いよいよ本題のminecraft-serverです。これについてはFTBのmodpackであればFTBのクライアントからDownload Serverボタンを押すだけでzip形式で最初から用意されたサーバーセットアップ一式がダウンロードできますので、ダウンロードURLをコピーしておいてサーバー側からwgetして取得しunzipするだけで準備完了です。<br />
<a href="http://www.feed-the-beast.com/servers">http://www.feed-the-beast.com/servers</a><br />
<br />
unzipして展開した中身のServerStart.batをServerStart.shにして中身を適当に書き換えてやり(主にJVMの起動オプションからヒープに割り当てる量を調整するなど)実行すれば簡単にサーバが立ち上がります。ただし普通に立ち上げるとシェルを専有されてしまうので、screenなど経由でServerStart.shを実行するようにすればいいでしょう。<br />
<a href="http://www.minecraftforum.net/forums/support/server-support/server-administration/1897274-how-to-use-screen-on-your-linux-minecraft-server">http://www.minecraftforum.net/forums/support/server-support/server-administration/1897274-how-to-use-screen-on-your-linux-minecraft-server</a><br />
<br />
あとはサーバーコマンドで自分をopにでも追加しておけば良いと思います。起動したサーバーのコマンドラインから /op 自分の名前 とコマンドを実行すればopを付与できます。その他必要に応じてminecraft-serverの設定ファイルを操作したりgameruleを調整すれば良いと思います。<br />
<br />
最後に先ほど設定したfirewalldにminecraftサービスを追加して、外からアクセスできるようにすればおしまいです。minecraftはhttpと違ってデフォルトでサービスが用意されていないので、自分でサービスを追加する必要があります。サービスの追加は/etc/firewalld/servicesにxml設定ファイルを追加すればよいです。サンプル代わりにデフォルトのサービスのxml設定ファイルが/usr/lib/firewalld/services以下に配置されているので適当なものをコピーして改造して使えばよいです。minecraftはtcp/udpの25565がデフォルトポートなのでここを開くように設定してfirewalldを再起動すれば外から繋がるようになるかと思います。<br />
<br />
<h3>
その他の設定</h3>
ログローテーションを設定しておかないとディスクがログで溢れて死ぬかと思ったんですが、最初からnginxについてはログローテーションが設定されているため気にしなくてOKです。minecraft-serverについてはどうなるかわかりませんので、必要に応じてlogrotateを設定して対応したほうが良いと思います。<br />
<a href="https://genchan.net/server/4907">https://genchan.net/server/4907</a><br />
<br />
<h3>
まとめ</h3>
いろいろ調べながらで時間がかかりましたが、3時間ほどでゼロ知識の状態からでもマインクラフトのマルチプレイサーバーを立てることができました。現在のところ無事快適に遊べています\(^o^)/<br />
<br />
こういう自分の趣味嗜好のために技術を勉強すると非常に捗るのでオススメですね。みなさんもゲームを楽しみながら技術を勉強して一石二鳥になりませんか\(^o^)/<br />
<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.comtag:blogger.com,1999:blog-42067579805640535.post-29502743536300442922016-01-07T21:56:00.000+09:002016-01-07T21:56:02.160+09:00Asset Catalog を使用しているときは [UIImage imageNamed:] が遅くなることがあるタイトルのとおりですが、本日発見してひどくパフォーマンスに影響が出たため注意喚起を兼ねて共有いたします。<br />
<br />
最近のiOSプロジェクトは全て画像をxcassetsすなわちAsset Catalog経由で管理することが多いかと思いますが、このAsset Catalogを使用している際に<code>[UIImage imageNamed:]</code>経由で画像を取得するとiOS 8以前とは異なり大変画像の取得が遅くなることがあるようです。<br />
<br />
詳細はこちら。<br />
<a href="https://forums.developer.apple.com/thread/17888">https://forums.developer.apple.com/thread/17888</a><br />
<br />
実際に私の環境では目に見えて遅くなりました。Instruments経由で計測したところ最大で10倍ほど差が出ているように見えました。iOS 9.1以降は修正されているとのことですが、それでも<code>[UIImage imageNamed:inBundle:compatibleWithTraitCollection:]</code>を使用したほうがパフォーマンスへの悪影響が少ないとの情報があります。実際Instruments上でもTrait Collectionを検索するのに無駄なパワーを使っているように見えたのでパフォーマンスがタイトな箇所では自分で非同期ロードするなりキャッシュするなりして補ってあげたほうが良い気がします。<br />
<br />
こちらからは以上です。<br />
<br />akisute (Masashi Ono)http://www.blogger.com/profile/14509295337098903238noreply@blogger.com