petitviolet_blog

@petitviolet blog

classをcase classに変換するscala.metaなライブラリ

練習を兼ねて、scala.metaを使って普通のclasscase classに進化させるライブラリを作った。

case classにすると自動で生成される以下のメソッド群をscalametaで模倣している。

  • toString
  • copy
  • equals
  • apply
  • unapply

hashCodeめんどくさいのでスキップ…。

@Case class Hoge(val n: Int)

とか書くとだいたい

case class Hoge(n: Int)

と同じ感じになる、ということ。

使い方

mavenにあげているので、依存の追加は簡単。

libraryDependencies += "net.petitviolet" %% "acase" % "<latest-version>"

でもマクロを有効にするための設定がいろいろ必要となる。

手順はここに書いた。setup
これがどれくらい必要なのかわからないが、とりあえずこれを書いていれば動く。

もうちょっと詳しく

publicなコンストラクタフィールド(?)しか持たないclassに@Caseをつける。

@Case class CaseApp(val n: Int, val s: String)

これが、おおよそ以下のように展開される。

class CaseApp(val n: Int, val s: String) {
  override def toString: String = {
    "CaseApp" + "(" + ("n" + ": " + n.toString + ", " + "s" + ": " + s.toString) + ")"
  }

  override def equals(obj: Any): Boolean = {
    if (!obj.isInstanceOf[CaseApp]) false else {
      val other = obj.asInstanceOf[CaseApp]
      this.n == other.n && this.s == other.s
    }
  }
  def copy(n = this.n, s = this.s) = new CaseApp(n, s)
}
object CaseApp {
  def unapply(arg: CaseApp): Option[(Int, String)] = {
    Some((arg.n, arg.s))
  }

  def apply(n: Int, s: String): CaseApp = new CaseApp(n, s)
}

とはいえ

case classにしない理由があるのにとりあえず全部生やすのも意味ないので、別々にアノテーションとして実装してある。

  • @ToString
  • @Copy
  • @Equals
  • @Apply
  • @Unapply

コンストラクタを隠したい、みたいな場合には欲しいアノテーションだけをごてごてとつければ良い。

@Equals @ToString @Copy @Unapply
class Hoge(val n: Int, val s: String)

privateなコンストラクタがあるとUnapplyとかを実装出来ないので全てpublicである必要がある。
もちろんリフレクション使えば出来るんだけど…。

感想

インスタンスメソッドを差し込むパターンとコンパニオンオブジェクトに差し込むパターンもあり、 何となくscalametaに慣れてきたところ。

とはいえアノテーション書きまくるの、Javaっぽいしちょっと嫌だなぁというのがまだ拭えないのでdef macro来てくれるの待ってる。

ScalaMatsuri2017に参加してきた

2017/02/25-26で開催されたScalaMatsuri2017に参加してきました。

2017.scalamatsuri.org

petitviolet.hatenablog.com

CFPに出したら通って400人の会場で話すことになって、この一ヶ月くらいずっとどきどきしてた。

発表資料はこちら

speakerdeck.com

なんだかんだでそこそこ人も入っていただいて、Twitterとかでも感想が少しpostされていて嬉しかったです。

登壇者の特権で壇上から写真も撮れた。

2日目のお昼は寿司職人が来て寿司Bar。テンション上がったし美味しかった。

2日間通して色々刺激を受けて非常に楽しかったです。
運営の皆様ありがとうございました。

来年も出来ればスピーカーとして参加出来るといいな。

Vimperatorでページのurl/titleをMarkdown的にcopyするプラグイン作った

久々に検索ショートカット以外のプラグイン

github.com

できること

とはいっても出来ることはめっちゃ少なくて、vimperatorでyankmarkdown|ymdコマンドを実行すると、
[petitviolet_blog](http://petitviolet.hatenablog.com/)
みたいにmarkdown形式でurlとtitleをclipboardに入れてくれるだけ。

github特別対応

ちょっとこだわったのがgithub上での挙動で、普通にやると
petitviolet/vimp-plugins: My Vimperator plugins
みたいにリポジトリのdescriptionまでタイトルに入ってしまう。

なので<user>/<repository>だけをタイトルとして扱えるようにした。
こんな感じ。
[petitviolet/vimp-plugins](https://github.com/petitviolet/vimp-plugins)

ちなみにファイルを開いている場合は<user>/<repository>/<filename>みたいな感じにした。
こんな感じ。
[petitviolet/vimp-plugins/yankMarkdown.js](https://github.com/petitviolet/vimp-plugins/blob/master/yankMarkdown.js)

特にgithubリポジトリへのリンクをmarkdown的に加工するのがめんどくさかったのが解消されて個人的に便利。