6

scalacプラグインを複数のファイルに分割したいと考えています。これは簡単に聞こえますが、行に起因するパス依存のタイプの問題により、うまくいきませんでしたimport global._

Lex Spoon のサンプル プラグインは次のとおりです。

package localhost

import scala.tools.nsc
import nsc.Global
import nsc.Phase
import nsc.plugins.Plugin
import nsc.plugins.PluginComponent

class DivByZero(val global: Global) extends Plugin {
  import global._

  val name = "divbyzero"
  val description = "checks for division by zero"
  val components = List[PluginComponent](Component)

  private object Component extends PluginComponent {
    val global: DivByZero.this.global.type = DivByZero.this.global
    val runsAfter = "refchecks"
    // Using the Scala Compiler 2.8.x the runsAfter should be written as below
    // val runsAfter = List[String]("refchecks");
    val phaseName = DivByZero.this.name
    def newPhase(_prev: Phase) = new DivByZeroPhase(_prev)    

    class DivByZeroPhase(prev: Phase) extends StdPhase(prev) {
      override def name = DivByZero.this.name
      def apply(unit: CompilationUnit) {
        for ( tree @ Apply(Select(rcvr, nme.DIV), List(Literal(Constant(0)))) <- unit.body;
             if rcvr.tpe <:< definitions.IntClass.tpe) 
          {
            unit.error(tree.pos, "definitely division by zero")
          }
      }
    }
  }
}

スコープ内になくても、独自のファイルにComponentどのように配置できますか?DivByZeroPhaseimport global._

4

3 に答える 3

4

これは私が同じことをした本当に古いプロジェクトです:

https://github.com/jsuereth/osgi-scalac-plugin/blob/master/src/main/scala/scala/osgi/compiler/OsgiPlugin.scala

グローバルからパス依存型を渡す必要がない場合は、その「this.global」部分を適切に保つことを心配する必要はありません。

于 2011-04-12T01:00:40.467 に答える
3

Scala Refactoring ライブラリでは、トレイト CompilerAccess を使用して解決しました。

trait CompilerAccess { 
  val global: tools.nsc.Global
}

アクセスする必要がある他のすべての特性は、依存関係としてglobal宣言するだけです。CompilerAccess

trait TreeTraverser {
  this: CompilerAccess =>
  import global._

  ...
}

そして、これらすべての特性を混ぜ合わせて global のインスタンスを提供するクラスがあります。

trait SomeRefactoring extends TreeTraverser with OtherTrait with MoreTraits {
  val global = //wherever you get your global from
}

このスキームは私にとって非常にうまくいきました。

于 2011-04-12T08:42:21.407 に答える
2

コンポーネント用に別のクラスを作成し、グローバルを渡すことができます:

class TemplateComponent(val global: Global) extends PluginComponent {

  import global._

  val runsAfter = List[String]("refchecks")

  val phaseName = "plugintemplate"

  def newPhase(prev: Phase) = new StdPhase(prev) {

    override def name = phaseName

    def apply(unit:CompilationUnit) = {
    }    
  }
}
于 2011-04-12T01:09:52.003 に答える