scala.meta.internal.classifiers.Macros.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of common_2.11 Show documentation
Show all versions of common_2.11 Show documentation
Bag of private and public helpers used in scala.meta's APIs and implementations
package scala.meta
package internal
package classifiers
import org.scalameta.internal.MacroHelpers
import scala.reflect.macros.whitebox.Context
// NOTE: It really sucks that `@foo object Bar` can't produce `class Bar`.
// Because of that, we have to uglify our classifier DSL.
class ClassifierMacros(val c: Context) extends MacroHelpers {
import c.universe._
lazy val ClassifierClass = tq"_root_.scala.meta.classifiers.Classifier"
def classifier(annottees: Tree*): Tree = annottees.transformAnnottees(new ImplTransformer {
override def transformTrait(cdef: ClassDef, mdef: ModuleDef): List[ImplDef] = {
val q"$mods trait $name[..$tparams] extends { ..$earlydefns } with ..$parents { $self => ..$stats }" =
cdef
val q"$mmods object $mname extends { ..$mearlydefns } with ..$mparents { $mself => ..$mstats }" =
mdef
def collectUnapplySig(stats: List[Tree]): Option[(Tree, Tree)] = stats
.collectFirst { case q"$_ def unapply[..$tparams]($_: $t): $ret = $_" => (t, ret) }
val unapplySig = collectUnapplySig(mstats).orElse(collectUnapplySig(stats))
val t = unapplySig match {
case Some((t, _)) => t
case None => c
.abort(c.enclosingPosition, "only traits with an unapply method can be @classifier")
}
val evidence = q"""
implicit def classifier[T <: $t]: $ClassifierClass[T, $name] = new $ClassifierClass[T, $name] {
def apply(x: T): $BooleanClass = $mname.unapply(x)
}
"""
val mstats1 = mstats ++ stats ++ List(evidence)
val cdef1 =
q"$mods trait $name[..$tparams] extends { ..$earlydefns } with ..$parents { $self => }"
val mdef1 =
q"$mmods object $mname extends { ..$mearlydefns } with ..$mparents { $mself => ..$mstats1 }"
List(cdef1, mdef1)
}
})
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy