Sublime Text 2でHTML要素のクラス名やIDの選択を少しだけ楽にする設定

Sublime Text 2は、デフォルトの設定では「-」(ハイフン)が単語の区切り文字に設定されています。なので、HTML要素のクラス名やIDなどのハイフン区切りの識別子をCtrl-dの単語選択で選択しようとしても一部分しか選択できなくて残念な気持ちになります。

しかし、この振る舞いはもちろん簡単に修正することができます。Sublime Text 2の設定項目のうちのひとつ、word_separatorsの内容を変更するだけです。

word_separatorsはデフォルトでは次のような設定になっています。

"word_separators": "./\\()\"'-:,.;<>~!@#$%^&*|+=[]{}`~?",

見ての通りハイフンが単語の区切り文字リストに含まれています。これを取り除いてユーザー設定ファイルにコピペすればOKです。

備考

この設定はユーザー設定ファイルに直接書いておくだけでも良いですが、言語固有の設定ファイルに置いておくのもよさそうです。僕はとりあえずHTMLとCSSJavaScript系の設定ファイルに置いておくことにしました。(言語固有の設定ファイルは、設定したい言語のファイルを開いて、メニューのPreferences > Settings - More > Syntax Specific - User と辿っていけば簡単に見つかります)

他にも、

  • JavaScript:「$」も単語の区切り文字リストから外しておく
  • Ruby: 「!, ?」を(略)
  • Lisp系: 「-, !, ?, *, などいろいろ」を(略)

などを設定しておくとよさそうですね:)

CoffeeScriptを利用したPhantom.jsのevaluateで「ReferenceError: Can't find variable: __slice」となる問題

原因

Phantom.jsのpage.evaluateに渡す関数は外側のスコープの変数を利用できないのに、CoffeeScriptのコンパイラが外側のスコープの変数を利用する関数を生成してしまうため。

対策

次のような関数を利用することでpage.evaluateに渡す関数が変数を自身のスコープ内のみで解決できるようにする。

toBeExecutableInBrowser = (func) ->
  str = func.toString().replace /\n/, '\n    var __slice = [].slice;\n'
  eval "(#{str})"

あるいは次のようにしてpage.evaluateの動作を変更し、ブラウザ内に__sliceという変数を事前に定義するようにしておく。

do () ->
  evaluate = page.evaluate
  page.evaluate = (func) ->
    evaluate.call @, `function() {
      window.__slice = [ ].slice;
    }`
    evaluate.call @, func

経緯みたいなもの(蛇足)

例えば、はてなブックマークホッテントリのタイトル一覧を取得しようとした場合、次のようなコードを書けば上手くいきます。

page = require('webpage').create()
page.open 'http://b.hatena.ne.jp/hotentry', (status) ->
  result = page.evaluate () ->
    links = document.getElementsByClassName 'entry-link'
    links = [ ].slice.call links
    JSON.stringify(links.map (link) -> link.innerText)
  titles = JSON.parse result
  console.log titles.join('\n')
  phantom.exit()

ブラウザ内でJavaScriptを実行するところの

links = document.getElementsByClassName 'entry-link'
links = [ ].slice.call links

という部分でdocument.getElementsByClassNameで取得したリンク一覧(配列のようなオブジェクト)をきちんとした配列に変換しています。

この部分をCoffeeScriptの文法を利用して次のように書き直しました。

[links...] = document.getElementsByClassName 'entry-link'

...これがGood PracticeかBad Practiceかどうかはどうかはさておくとして、一応これでも動くと思ったんですが、実行してみると次のようなエラーになってしまいました。

ReferenceError: Can't find variable: __slice

  phantomjs://webpage.evaluate():3
  phantomjs://webpage.evaluate():7
  phantomjs://webpage.evaluate():7
TypeError: 'null' is not an object (evaluating 'titles.join')

  example.coffee:17

CoffeeScriptのコンパイラが余計な気を利かして配列化の関数__slicepage.evaluateに渡す関数の外側で定義していたことが原因でした。page.evaluateは渡された関数の外側にある変数を利用できない(というかPhantom側からブラウザ側に関数を渡す過程でスコープチェーンを保持しておくことができない)ので、ブラウザ側で__sliceという変数の解決に失敗しているということです。

それで、なるべく関数のコードを変えないような形でこの問題を解決できないかといろいろ試したんですが、最終的に次のようにするのがベターかなと思いました。

toBeExecutableInBrowser = (func) ->
  str = func.toString().replace /\n/, '\n    var __slice = [ ].slice;\n'
  eval "(#{str})"

getTitles = () ->
  [links...] = document.getElementsByClassName 'entry-link'
  JSON.stringify(links.map (link) -> link.innerText)

browserGetTitles = toBeExecutableInBrowser getTitles

page = require('webpage').create()
page.open 'http://b.hatena.ne.jp/hotentry', (status) ->
  result = page.evaluate browserGetTitles
  titles = JSON.parse result
  console.log titles.join('\n')
  phantom.exit()

toBeExecutableInBrowserという関数を利用して、page.evaluateに渡したい関数の中身に無理やり__sliceの宣言を追加してしまうという感じです。

一応これで動くようになりました。まあ今回の場合はべつに[links...]という記法を使わずに[ ].slice.callを使ったり、そもそも[ ].map.callを使えばよかったわけですが、他にも色々と同じような問題が出てくるだろうということで書き残しておきました。CoffeeScriptの予約語リストを見るに'__hasProp', '__extends', '__slice', '__bind' '__indexOf'のあたりはまた同じような問題を起こしそうです。


もう一つのやり方として次のようなのも良い感じです。

browserGetTitles = () ->
  [links...] = document.getElementsByClassName 'entry-link'
  JSON.stringify(links.map (link) -> link.innerText)

page = require('webpage').create()

do () ->
  evaluate = page.evaluate
  page.evaluate = (func) ->
    evaluate.call @, `function() {
      window.__slice = [ ].slice;
    }`
    evaluate.call @, func

page.open 'http://b.hatena.ne.jp/hotentry', (status) ->
  result = page.evaluate browserGetTitles
  titles = JSON.parse result
  console.log titles.join('\n')
  phantom.exit()

page.evaluateの動作を変更して、先にグローバルに__sliceを定義するようにしています。CoffeeScriptでは__slice予約語になっているのでJavaScript埋め込み機能を利用しています。page.evaluateJavaScriptを使えばもう少し簡潔になるんですが、なぜか利用できないのでこういう形になりました。

あとはpage.includeJspage.injectJsなんかを利用しても解決できるかと思います。

ブラウザでBase64エンコードされた画像をデコードして閲覧する

(Chrome, Safari, Firefoxで確認)

ブラウザのアドレスバーにdata:image/jpeg;base64,{{Base64文字列}}みたいな感じで入力して移動すると表示できます。

とりあえずGooglefaviconで。下のリンクをクリックするか、テキストをトリプルクリックなんかでコピーしてアドレスバーに貼り付けて移動してみてください。画像が表示されるのが確認できると思います。

リンク形式

リンク

テキスト形式



MIMEタイプのところを変えればPNGやGIFといった他の形式の画像や、テキストファイルなんかも表示できそうですね。下は「Base64テスト」と書かれたテキストファイルをBase64エンコードした例です(もし文字化けしていればブラウザのエンコード指定をUTF-8に変更すれば正しく表示されると思います)。

リンク

data:text/plain;base64,QmFzZTY044OG44K544OICg==

Google画像検索結果のimgのsrc属性に使われているのを見て知りました:)

追記

こういうのはData URI schemeというらしい。

GitHub Pages利用メモ

Git力がないからこういうことはすぐハマってしまう:(
いろいろなサイトを参考にさせてもらいながらなんとか解決。

GitHubにmasterブランチは登録せずにgh-pagesブランチだけを登録してそのままgh-pagesで開発していく、というようなことがしたかった。

参考サイト

コード

masterをgh-pagesに改名する版

git init
echo "hello world" >> index.html
git add .
git commit -m "first commit"
git branch -m gh-pages
git remote add origin git@github.com:username/repositoryname.git
git push -u origin gh-pages

ローカルのmasterをリモートのgh-pagesにpushする版

git init
echo "hello world" >> index.html
git add .
git commit -m "first commit"
git remote add origin git@github.com:username/repositoryname.git
git push -u origin master:gh-pages

GitHubでの検索

ちょっと前の話題だけど、GitHubの検索機能の改良があったそうなのでリンクのメモ。あと、検索のトップページにいろいろなフィルターが載っていたのでこちらもメモ。

追記(2013/01/24)

GitHubにコードの全文検索機能が追加されてトップページのデザインが一新されてた。

JavaScript関連いろいろ

備忘録的に。

Backbone.js in Practice: Part I – Preventing Memory Leaks

Backbone.jsにおけるメモリリーク対策の解説。Backbone.jsにおけるメモリリークはイベントバインディングとネストしたビューが鬼門なんだけど、それらに対するメモリリーク対策のパターンが分かりやすく解説されている。

いままでメモリリークのことを気にしたことなかったのでためになった。まあメモリリークを気にするほどの大きなBackbone.jsアプリ作ってないけど:)

Marionette.js – A scalable and composite application architecture for Backbone.js

Backbone.jsの拡張的なもの?GitHubでのStar数が1600を超えていてかなり人気っぽい。

GitHub上のREADMEをちょこっと読んだだけだから詳しくは分からないけど、メモリ管理の機能を用意してくれたりビューやイベントの管理が楽にできたりするらしい?早く詳しく調べたい。

novus / nvd3

チャートライブラリ。グラフ描画系のライブラリといえば少し前にMorris.jsが話題になっていたけど、このNVD3はD3.jsを利用したもの。

まだまだ開発途中・発展途上って感じだけどなかなか良さげ。なによりD3.jsを利用してるってところが(・∀・)イイ!!

最近ちょっとしたお家騒動みたいなことがあったらしく、公式サイト謝罪文的なものが貼られてるけど、まあ結果的にこれからもずっとオープンソースプロジェクトとしてやっていくことになったらしいから一安心。