Category Archives: プログラミング

vim-autocloseの補完対象の追加と削除

前置き

lisp系言語を使っているとカッコの対応をとるのがつらぽよになるので, エディタの支援機能を使うのが一般的(らしい)です.
カッコの色付けにより対応をわかりやすくするのはよく見る(気がする)支援機能ですね. あとは, 左カッコと同時に右カッコが補完される機能が, 賛否両論みたいですが割とメジャーっぽいですね.

さて,僕がよく使うエディタはvimなのですが, vimにも支援機能は当然あります. 僕は色付けにkien/rainbow_parentheses.vimを, カッコの自動補完にTownk/vim-autocloseあたりを使ってます. (ぶっちゃけあまり良く調べずにとりあえず使ってる)

このvim-autoclose, 各種カッコに限らず, 各種クォートも補完してくれるのですが, lispではシングルクォートってシンボルとかquoteに使うし, バッククォートも準クォートとかシンタックスクオートとかに使うんで, 補完されるとぶっちゃけ邪魔です.
他にも, rubyでは|をブロックでよく使うので補完してほしいのですが, こちらは逆に補完してくれません.
filetypeに合わせて, 補完対象を変えてくれたら便利ですよね. と思ってググったところ日本語情報が見つからなかったので, vim-autocloseのドキュメントにあたったところ, 解決策を発見出来たので, 書き残す次第です.

本題

vim-autocloseのドキュメントによると,

AutoClose#DefaultPairsModified(pairsToAdd, openersToRemove)

という書式で補完対象の追加, 削除が出来るようです. さらに実際の利用例も載っています.

" auto-close || pair in ruby files
      autocmd FileType ruby
              \ let b:AutoClosePairs = AutoClose#DefaultPairsModified("|", "")

これで, filetypeがrubyだった場合, |も補完対象に含めてくれるようです. 他にも自分のほしい機能を.vimrcに設定してみました.

autocmd FileType lisp let b:AutoClosePairs = AutoClose#DefaultPairsModified("", "' `")
autocmd FileType clojure let b:AutoClosePairs = AutoClose#DefaultPairsModified("", "' `")

シングル, バッククォートを外してみました. こんな感じでいろいろ設定するといいと思います. (そんなに複雑な設定をするようなものでもない気がするけど)

では, 良きvimライフを

余談

久々に書いたらやけに丁寧口調になった
こういうのっってqiitaに書いたほうがいいんだろうか

プログラミングClojure 第2版 p236の誤植

タイトルの通りです.

先に結論

$ lein noir new projectname

ではなくて

$ lein new noir projectname

試行錯誤の記録

現在の環境ならば

$ lein plugin install lein-noir 1.2.0

を実行しようとする時点でおそらくうまく行かない. ~/.lein/profiles.cljを作成して, 設定ファイルに

{:user {:plugins [[lein-noir "1.2.0"]]}}

と書いてからやるとうまく行くらしいんだけど, 怒られたのか, それともうまくいったのかよくわからない(多分ダメそう)な感じに

$ lein plugin install lein-noir 1.2.0
Retrieving lein-noir/lein-noir/1.2.0/lein-noir-1.2.0.pom from clojars
Retrieving org/clojure/clojure/1.2.1/clojure-1.2.1.jar from central
Retrieving lein-noir/lein-noir/1.2.0/lein-noir-1.2.0.jar from clojars
The plugin task has been removed.
 
Please see the upgrade guide for instructions on how to use the user profile to
specify plugins instead: https://github.com/technomancy/leiningen/wiki/Upgrading

ほかにも, noirを使おうとするプロジェクトのproject.cljのdependenciesに

[noir "1.2.0"]

を追加してlein depsするといろいろ落としてくるけど,

Retrieving noir/noir/1.2.0/noir-1.2.0.jar from clojars
WARNING!!! version ranges found for:
[noir "1.2.0"] -> [org.clojure/clojure "[1.3.0,1.3.0],[1.2.1,1.2.1]"]
Consider using [noir "1.2.0" :exclusions [org.clojure/clojure]].

なんかこう言われる. ので:exclusionsのあたりを追記してlein depsすると何も出力しない.
とりあえずインストールできたと仮定して,

lein plugin install lein-noir 1.2.0

を再度実行. すると,

$ lein noir new clojurebreaker
Warning: *project-dir* not declared dynamic and thus is not dynamically rebindable, but its name suggests otherwise. Please either indicate ^:dynamic *project-dir* or change the name. (leiningen/noir/new.clj:5)
Warning: *project* not declared dynamic and thus is not dynamically rebindable, but its name suggests otherwise. Please either indicate ^:dynamic *project* or change the name. (leiningen/noir/new.clj:5)
Warning: *dirs* not declared dynamic and thus is not dynamically rebindable, but its name suggests otherwise. Please either indicate ^:dynamic *dirs* or change the name. (leiningen/noir/new.clj:5)
Create and manage noir projects.

Subtasks available:
new     

Run `lein help noir $SUBTASK` for subtask details.

Arguments: ([new])

といわれる. 正直良くわからなくなってしまったのでGoogle先生に聞いてみると,

【Lisp】プログラミング言語 Clojure #3【JVM】

191 :デフォルトの名無しさん:2014/02/14(金) 20:03:15.49 .net
プログラミングClojure第二版の10.4の

lein noir new clojurebreaker

って

lein new noir clojurebreaker

の間違いじゃね?

とのことなので, やってみる.

$ lein new noir clojurebreaker
Generating a lovely new Noir project named clojurebreaker...

できた.

ほかにも

このclojurebreakerは完全に新しいディレクトを作るべきなのか, それともすでにあるclojurebreakerのディレクトリ内に作るべきなのかとかわからないし, このページの下半分だけすっごく不親切.
どうやら, 今まで使ってきたclojurebreakerプロジェクトとは別に, noirを使ってできたprojectにコード類は移植するってことらしい.

ただ, lein-noir自体が今では開発が終わってしまっていることは本の訳注にもあるし, 実質参考にならないページだと言っていいのかもしれない.

追記

noirのバージョンが, 1.2.0だとlein runで例外を吐く.
lein new noir projectnameしたときにproject.cljに書いてあった1.3.0-beta3でないとダメみたい.

更に追記
Retrieving noir/noir/1.2.0/noir-1.2.0.jar from clojars
WARNING!!! version ranges found for:
[noir "1.2.0"] -> [org.clojure/clojure "[1.3.0,1.3.0],[1.2.1,1.2.1]"]
Consider using [noir "1.2.0" :exclusions [org.clojure/clojure]].

どう見てもclojureのバージョンについての警告なのになぜかnoirのバージョンで怒られているものだと勘違いしていた
まあ, だからどうということでもないけど

Clojureのmapのちょっとした気付き

mapってリストを複数受け取れたんだなって

知らなかった

user=> (doc map)
-------------------------
clojure.core/map
([f coll] [f c1 c2] [f c1 c2 c3] [f c1 c2 c3 & colls])
  Returns a lazy sequence consisting of the result of applying f to the
  set of first items of each coll, followed by applying f to the set
  of second items in each coll, until any one of the colls is
  exhausted.  Any remaining items in other colls are ignored. Function
  f should accept number-of-colls arguments.
nil
user=> (map + [1 2 3] [2 3 4] [3 4 5] [4 5 6])
(10 14 18)
-------------------------

複数のリストの同じ位置の要素に, それぞれ関数を適用した結果のリスト(あやしい日本語)を返していることがわかる

渡したリストの長さが違う場合

user=> (map + [1 3] [2])
(3)
user=> (map + [1 3] [2 3 4])
(3 6)

短い方に合わせられている

Clojureでx::xsとか[X|Y]みたいなやつ

何したか

F#では, リストをx::xsみたいに先頭とそれ以外に分けれるっぽい
Prologも[X|Y]みたいにできた気がする
Clojureでもできないかと思ってしらべた

(defn f [[x & xs]]
  (str x "::" xs))
(f [1 2 3]) ; -> "1::(2 3)"

どうなるのか

関数fの引数にリストを渡すと, 分配束縛によってxにfirst, xsにrestが入る.

感想

関数の引数としてしか機能しないし, 予期しない制限多そう
ほかにはletつかうとか?

追伸

こっそりClojure始めたので今後書いていきたい