All Downloads are FREE. Search and download functionalities are using the official Maven repository.

dotty.tools.dotc.printing.RefinedPrinter.scala Maven / Gradle / Ivy

There is a newer version: 3.6.4-RC1-bin-20241220-0bfa1af-NIGHTLY
Show newest version
package dotty.tools
package dotc
package printing

import core.*
import Constants.*
import Texts.*
import Types.*
import Flags.*
import Names.*
import Symbols.*
import NameOps.*
import TypeErasure.ErasedValueType
import Contexts.*
import Annotations.Annotation
import Denotations.*
import SymDenotations.*
import StdNames.{nme, tpnme}
import ast.{Trees, tpd, untpd}
import tpd.closureDef
import typer.{Implicits, Namer, Applications}
import typer.ProtoTypes.*
import Trees.*
import TypeApplications.*
import NameKinds.{WildcardParamName, DefaultGetterName}
import util.Chars.isOperatorPart
import config.{Config, Feature}
import config.Feature.sourceVersion
import config.SourceVersion.*

import dotty.tools.dotc.util.SourcePosition
import dotty.tools.dotc.ast.untpd.{MemberDef, Modifiers, PackageDef, RefTree, Template, TypeDef, ValOrDefDef}
import cc.{CaptureSet, CapturingType, toCaptureSet, IllegalCaptureRef, isRetains, ReachCapability, MaybeCapability}
import dotty.tools.dotc.parsing.JavaParsers

class RefinedPrinter(_ctx: Context) extends PlainPrinter(_ctx) {

  /** A stack of enclosing DefDef, TypeDef, or ClassDef, or ModuleDefs nodes */
  private var enclosingDef: untpd.Tree = untpd.EmptyTree
  private var myCtx: Context = super.printerContext
  private var printPos = ctx.settings.YprintPos.value
  private val printLines = ctx.settings.printLines.value

  override def printerContext: Context = myCtx

  def withEnclosingDef(enclDef: Tree[?])(op: => Text): Text = {
    val savedCtx = myCtx
    if (enclDef.hasType && enclDef.symbol.exists)
      myCtx = ctx.withOwner(enclDef.symbol)
    val savedDef = enclosingDef
    enclosingDef = enclDef
    try op finally {
      myCtx = savedCtx
      enclosingDef = savedDef
    }
  }

  def inPattern(op: => Text): Text = {
    val savedCtx = myCtx
    myCtx = ctx.addMode(Mode.Pattern)
    try op finally myCtx = savedCtx
  }

  inline def inContextBracket(inline op: Text): Text =
    val savedCtx = myCtx
    try op finally myCtx = savedCtx

  def withoutPos(op: => Text): Text = {
    val savedPrintPos = printPos
    printPos = false
    try op finally printPos = savedPrintPos
  }

  protected def enclDefIsClass: Boolean = enclosingDef match {
    case owner: TypeDef => owner.isClassDef
    case owner: untpd.ModuleDef => true
    case _ => false
  }

  protected def PrintableFlags(isType: Boolean): FlagSet = {
    val fs =
      if (isType) TypeSourceModifierFlags | Module | Local  // DOTTY problem: cannot merge these two statements
      else TermSourceModifierFlags | Module | Local
    fs.toCommonFlags
  }

  override def nameString(name: Name): String =
    def strippedName = if printDebug then name else name.stripModuleClassSuffix
    if ctx.settings.YdebugNames.value then strippedName.debugString
    else if name.isTypeName && name.is(WildcardParamName) && !printDebug then "_"
    else super.nameString(strippedName)

  override protected def simpleNameString(sym: Symbol): String =
    nameString(if (ctx.property(XprintMode).isEmpty) sym.initial.name else sym.name)

  override def fullNameString(sym: Symbol): String =
    if !sym.exists
      || isEmptyPrefix(sym.effectiveOwner)
      || !homogenizedView && !sym.is(Package) && isOmittablePrefix(sym.effectiveOwner)
    then nameString(sym)
    else super.fullNameString(sym)

  override protected def fullNameOwner(sym: Symbol): Symbol = {
    val owner = super.fullNameOwner(sym)
    if (owner.is(ModuleClass)) owner.sourceModule else owner
  }

  override def toTextRef(tp: SingletonType): Text = controlled {
    tp match {
      case tp: ThisType if !printDebug =>
        if (tp.cls.isAnonymousClass) keywordStr("this")
        if (tp.cls.is(ModuleClass)) fullNameString(tp.cls.sourceModule)
        else super.toTextRef(tp)
      case tp: TermRef if !printDebug =>
        if tp.symbol.is(Package) then fullNameString(tp.symbol)
        else super.toTextRef(tp)
      case _ =>
        super.toTextRef(tp)
    }
  }

  override def toTextPrefixOf(tp: NamedType): Text = controlled {
    def isOmittable(sym: Symbol) =
      if printDebug then false
      else if homogenizedView then isEmptyPrefix(sym) // drop  and anonymous classes, but not scala, Predef.
      else if sym.isPackageObject then isOmittablePrefix(sym.owner)
      else isOmittablePrefix(sym)
    def isSkippedPackageObject(sym: Symbol) =
      sym.isPackageObject && !homogenizedView && !printDebug

    tp.prefix match {
      case thisType: ThisType =>
        val sym = thisType.cls
        if isSkippedPackageObject(sym) then toTextPrefixOf(sym.typeRef)
        else if isOmittable(sym) then ""
        else super.toTextPrefixOf(tp)
      case termRef: TermRef =>
        val sym = termRef.symbol
        if isSkippedPackageObject(sym) then toTextPrefixOf(termRef)
        else if isOmittable(sym) then ""
        else super.toTextPrefixOf(tp)
      case _ => super.toTextPrefixOf(tp)
    }
  }

  override protected def toTextParents(parents: List[Type]): Text =
    Text(parents.map(toTextLocal).map(typeText), keywordStr(" with "))

  override protected def refinementNameString(tp: RefinedType): String =
    if (tp.parent.isInstanceOf[WildcardType] || tp.refinedName == nme.WILDCARD)
      super.refinementNameString(tp)
    else {
      val tsym = tp.parent.member(tp.refinedName).symbol
      if (!tsym.exists) super.refinementNameString(tp)
      else simpleNameString(tsym)
    }

  private def arrow(isGiven: Boolean, isPure: Boolean): String =
    (if isGiven then "?" else "") + (if isPure then "->" else "=>")

  private def toTextFunction(tp: AppliedType, refs: Text = Str("")): Text =
    val AppliedType(tycon, args) = (tp: @unchecked)
    val tsym = tycon.typeSymbol
    val isGiven = tsym.name.isContextFunction
    val capturesRoot = refs == rootSetText
    val isPure =
      Feature.pureFunsEnabled && !tsym.name.isImpureFunction && !capturesRoot
    changePrec(GlobalPrec) {
      val argStr: Text =
        if args.length == 2
            && !defn.isTupleNType(args.head)
            && !isGiven
        then
          atPrec(InfixPrec) { argText(args.head) }
        else
          "("
          ~ argsText(args.init)
          ~ ")"
      argStr
      ~ " " ~ arrow(isGiven, isPure)
      ~ (refs provided !capturesRoot)
      ~ " " ~ argText(args.last)
    }

  protected def toTextMethodAsFunction(info: Type, isPure: Boolean, refs: Text = Str("")): Text = info match
    case info: MethodType =>
      val capturesRoot = refs == rootSetText
      changePrec(GlobalPrec) {
        "("
        ~ paramsText(info)
        ~ ") "
        ~ arrow(info.isImplicitMethod, isPure && !capturesRoot)
        ~ (refs provided !capturesRoot)
        ~ " "
        ~ toTextMethodAsFunction(info.resultType, isPure)
      }
    case info: PolyType =>
      changePrec(GlobalPrec) {
        "["
        ~ paramsText(info)
        ~ "] => "
        ~ toTextMethodAsFunction(info.resultType, isPure)
      }
    case _ =>
      toText(info)

  override def toText(tp: Type): Text = controlled {
    def toTextTuple(args: List[Type]): Text =
      "(" ~ argsText(args) ~ ")"

    def toTextNamedTuple(elems: List[(TermName, Type)]): Text =
      val elemsText = atPrec(GlobalPrec):
        Text(elems.map((name, tp) => toText(name) ~ " : " ~ argText(tp)), ", ")
      "(" ~ elemsText ~ ")"

    def isInfixType(tp: Type): Boolean = tp match
      case AppliedType(tycon, args) =>
        args.length == 2
        && {
          val sym = tycon.typeSymbol
          sym.is(Infix)
          || sym.getAnnotation(defn.ShowAsInfixAnnot)
              .exists(_.argumentConstant(0).forall(_.booleanValue))
          || !Character.isUnicodeIdentifierStart(tycon.typeSymbol.name.toString.head)
        }
      case _ => false

    def tyconName(tp: Type): Name = tp.typeSymbol.name
    def checkAssocMismatch(tp: Type, isRightAssoc: Boolean) = tp match {
      case AppliedType(tycon, _) => isInfixType(tp) && tyconName(tycon).endsWith(":") != isRightAssoc
      case AndType(_, _) => isRightAssoc
      case OrType(_, _) => isRightAssoc
      case _ => false
    }

    def toTextInfixType(opName: Name, l: Type, r: Type)(op: => Text): Text = {
      val isRightAssoc = opName.endsWith(":")
      val opPrec = parsing.precedence(opName)

      changePrec(opPrec) {
        val leftPrec = if (isRightAssoc || checkAssocMismatch(l, isRightAssoc)) opPrec + 1 else opPrec
        val rightPrec = if (!isRightAssoc || checkAssocMismatch(r, isRightAssoc)) opPrec + 1 else opPrec

        atPrec(leftPrec) { argText(l) }  ~ " " ~ op ~ " " ~ atPrec(rightPrec) { argText(r) }
      }
    }

    def appliedText(tp: Type): Text = tp match
      case tp @ AppliedType(tycon, args) =>
        val namedElems =
          try tp.namedTupleElementTypesUpTo(200, normalize = false)
          catch case ex: TypeError => Nil
        if namedElems.nonEmpty then
          toTextNamedTuple(namedElems)
        else tp.tupleElementTypesUpTo(200, normalize = false) match
          //case Some(types @ (defn.NamedTupleElem(_, _) :: _)) if !printDebug =>
          //  toTextTuple(types)
          case Some(types) if types.size >= 2 && !printDebug =>
            toTextTuple(types)
          case _ =>
            val tsym = tycon.typeSymbol
            if tycon.isRepeatedParam then toTextLocal(args.head) ~ "*"
            else if defn.isFunctionSymbol(tsym) then toTextFunction(tp)
            else if isInfixType(tp) then
              val l :: r :: Nil = args: @unchecked
              val opName = tyconName(tycon)
              toTextInfixType(tyconName(tycon), l, r) { simpleNameString(tycon.typeSymbol) }
            else Str("")
      case _ =>
        Str("")

    homogenize(tp) match {
      case tp: AppliedType =>
        val refined = appliedText(tp)
        if refined.isEmpty then super.toText(tp) else refined
      // Since RefinedPrinter, unlike PlainPrinter, can output right-associative type-operators, we must override handling
      // of AndType and OrType to account for associativity
      case AndType(tp1, tp2) =>
        toTextInfixType(tpnme.raw.AMP, tp1, tp2) { toText(tpnme.raw.AMP) }
      case tp @ OrType(tp1, tp2) =>
        toTextInfixType(tpnme.raw.BAR, tp1, tp2) {
          if tp.isSoft && printDebug then toText(tpnme.ZOR) else toText(tpnme.raw.BAR)
        }
      case tp @ EtaExpansion(tycon)
      if !printDebug && appliedText(tp.asInstanceOf[HKLambda].resType).isEmpty =>
        // don't eta contract if the application would be printed specially
        toText(tycon)
      case tp: RefinedType if defn.isFunctionType(tp) && !printDebug =>
        toTextMethodAsFunction(tp.refinedInfo,
          isPure = Feature.pureFunsEnabled && !tp.typeSymbol.name.isImpureFunction,
          refs = tp.parent match
            case CapturingType(_, cs) => toTextCaptureSet(cs)
            case _ => "")
      case tp: TypeRef =>
        if (tp.symbol.isAnonymousClass && !showUniqueIds)
          toText(tp.info)
        else if (tp.symbol.is(Param))
          tp.prefix match {
            case pre: ThisType if pre.cls == tp.symbol.owner =>
              nameString(tp.symbol)
            case _ => super.toText(tp)
          }
        else super.toText(tp)
      case ErasedValueType(tycon, underlying) =>
        "ErasedValueType(" ~ toText(tycon) ~ ", " ~ toText(underlying) ~ ")"
      case tp: ClassInfo =>
        if tp.cls.derivesFrom(defn.PolyFunctionClass) then
          tp.member(nme.apply).info match
            case info: PolyType => toTextMethodAsFunction(info, isPure = false)
            case _ => toTextParents(tp.parents) ~~ "{...}"
        else toTextParents(tp.parents) ~~ "{...}"
      case JavaArrayType(elemtp) =>
        toText(elemtp) ~ "[]"
      case tp: LazyRef if !printDebug =>
        try toText(tp.ref)
        catch case ex: Throwable => "..."
      case AnySelectionProto =>
        "a type that can be selected or applied"
      case tp: SelectionProto =>
        "?{ " ~ toText(tp.name) ~
           (Str(" ") provided !tp.name.toSimpleName.last.isLetterOrDigit) ~
           ": " ~ toText(tp.memberProto) ~ " }"
      case tp: ViewProto =>
        toText(tp.argType) ~ " ?=>? " ~ toText(tp.resultType)
      case tp @ FunProto(args, resultType) =>
        "[applied to ("
        ~ keywordText("using ").provided(tp.applyKind == ApplyKind.Using)
        ~ argsTreeText(args)
        ~ ") returning "
        ~ toText(resultType)
        ~ "]"
      case IgnoredProto(ignored) =>
        "?" ~ (("(ignored: " ~ toText(ignored) ~ ")") provided printDebug)
      case tp @ PolyProto(targs, resType) =>
        "[applied to [" ~ toTextGlobal(targs, ", ") ~ "] returning " ~ toText(resType)
      case ReachCapability(_) | MaybeCapability(_) =>
        toTextCaptureRef(tp)
      case _ =>
        super.toText(tp)
    }
  }

  protected def exprToText(tp: ExprType): Text =
    "=> " ~ toText(tp.resType)

  protected def argsTreeText(args: List[untpd.Tree]): Text = args match
    case dummyTreeOfType(tp) :: Nil if !tp.isRef(defn.NullClass) && !homogenizedView => toText(Constant(null)) ~ ": " ~ toText(tp)
    case _                                                                           => toTextGlobal(args, ", ")

  protected def blockToText[T <: Untyped](block: Block[T]): Text =
    blockText(block.stats :+ block.expr)

  protected def blockText[T <: Untyped](trees: List[Tree[T]]): Text =
    inContextBracket {
      ("{" ~ toText(trees, "\n") ~ "}").close
    }

  protected def typeApplyText[T <: Untyped](tree: TypeApply[T]): Text = {
    val funText = toTextLocal(tree.fun)
    tree.fun match {
      case Select(New(tpt), nme.CONSTRUCTOR) if tpt.typeOpt.dealias.isInstanceOf[AppliedType] =>
        funText  // type was already printed by toText(new)
      case _ =>
        funText ~ "[" ~ toTextGlobal(tree.args, ", ") ~ "]"
    }
  }

  protected def toTextCore[T <: Untyped](tree: Tree[T]): Text = {
    import untpd.*

    def isLocalThis(tree: Tree) = tree.typeOpt match {
      case tp: ThisType => tp.cls == ctx.owner.enclosingClass
      case _ => false
    }

    def optDotPrefix(tree: This) = optText(tree.qual)(_ ~ ".") provided !isLocalThis(tree)

    def caseBlockText(tree: Tree): Text = tree match {
      case Block(stats, expr) => toText(stats :+ expr, "\n")
      case expr => toText(expr)
    }

    // Dotty deviation: called with an untpd.Tree, so cannot be a untpd.Tree[T] (seems to be a Scala2 problem to allow this)
    // More deviations marked below as // DD
    def enumText(tree: untpd.Tree) = tree match { // DD
      case _: untpd.GenFrom | _: untpd.GenAlias => toText(tree)
      case _ => keywordStr("if ") ~ toText(tree)
    }

    def forText(enums: List[untpd.Tree], expr: untpd.Tree, sep: String): Text = // DD
      changePrec(GlobalPrec) { keywordStr("for ") ~ Text(enums map enumText, "; ") ~ sep ~ toText(expr) }

    def cxBoundToText(bound: untpd.Tree): Text = bound match { // DD
      case ContextBoundTypeTree(tpt, _, _) => " : " ~ toText(tpt)
      case untpd.Function(_, tpt) => " <% " ~ toText(tpt)
    }

    def varianceText(mods: untpd.Modifiers) =
      if (mods.is(Covariant)) "+"
      else if (mods.is(Contravariant)) "-"
      else ""

    def argText(arg: Tree): Text = arg match {
      case arg: TypeBoundsTree => "?" ~ toTextGlobal(arg)
      case arg: TypeTree =>
        arg.typeOpt match {
          case tp: TypeBounds => "?" ~ toTextGlobal(arg)
          case _ => toTextGlobal(arg)
        }
      case _ => toTextGlobal(arg)
    }

    def dropBlock(tree: Tree): Tree = tree match {
      case Block(Nil, expr) => expr
      case _ => tree
    }

    def importText(expr: Tree, selectors: List[untpd.ImportSelector]) =

      def selectorText(sel: untpd.ImportSelector): Text =
        val id: Text =
          if sel.isGiven then keywordText("given")
          else sel.imported.name match
            case nme.WILDCARD => "*"
            case nme.raw.STAR => "`*`"
            case name => toText(name)
        val rename: Text =
          if sel.renamed.isEmpty then "" else Str(" as ") ~ toText(sel.renamed)
        val bound: Text =
          if sel.bound.isEmpty then ""
          else if sel.isGiven then Str(" ") ~ toText(sel.bound)
          else Str(" : ") ~ toText(sel.bound)
        id ~ rename ~ bound

      val selectorsText: Text = selectors match
        case (sel @ untpd.ImportSelector(name, EmptyTree, EmptyTree)) :: Nil =>
          selectorText(sel)
        case _ =>
          "{" ~ Text(selectors.map(selectorText), ", ") ~ "}"

      toTextLocal(expr) ~ "." ~ selectorsText

    tree match {
      case id: Trees.SearchFailureIdent[?] =>
        tree.typeOpt match {
          case reason: Implicits.SearchFailureType =>
            Str(id.explanation)
            ~ ("summon[" ~ toText(reason.clarify(reason.expectedType)) ~ "]").close
          case _ =>
            Str(id.explanation)
        }
      case id @ Ident(name) =>
        val txt = tree.typeOpt match {
          case tp: NamedType if name != nme.WILDCARD =>
            toTextPrefixOf(tp) ~ withPos(selectionString(tp), tree.sourcePos)
          case _ =>
            toText(name)
        }
        if (isBackquoted(tree) && !homogenizedView) "`" ~ toText(name) ~ "`"
        else if (name.isTypeName) typeText(txt)
        else txt
      case tree @ Select(qual, name) =>
        if (qual.isType) toTextLocal(qual) ~ "#" ~ typeText(toText(name))
        else toTextLocal(qual) ~ ("." ~ nameIdText(tree) provided (name != nme.CONSTRUCTOR || printDebug))
      case tree: This =>
        optDotPrefix(tree) ~ keywordStr("this") ~ idText(tree)
      case Super(qual: This, mix) =>
        optDotPrefix(qual) ~ keywordStr("super") ~ optText(mix)("[" ~ _ ~ "]")
      case app @ Apply(fun, args) =>
        if (fun.hasType && fun.symbol == defn.throwMethod)
          changePrec (GlobalPrec) {
            keywordStr("throw ") ~ toText(args.head)
          }
        else
          toTextLocal(fun)
          ~ "("
          ~ Str("using ").provided(app.applyKind == ApplyKind.Using && !homogenizedView)
          ~ argsTreeText(args)
          ~ ")"
      case tree: TypeApply =>
        typeApplyText(tree)
      case Literal(c) =>
        tree.typeOpt match {
          case ConstantType(tc) => withPos(toText(tc), tree.sourcePos)
          case _ => withPos(toText(c), tree.sourcePos)
        }
      case New(tpt) =>
        keywordStr("new ") ~ {
          tpt match {
            case tpt: Template => toTextTemplate(tpt, ofNew = true)
            case _ => toTextLocal(tpt)
          }
        }
      case Typed(expr, tpt) =>
        changePrec(InfixPrec) {
          if isWildcardStarArg(tree) then
            expr match
              case Ident(nme.WILDCARD_STAR) =>
                // `_*` is used as a wildcard name to indicate a vararg splice pattern;
                // avoid the double `*` in this case.
                toText(expr)
              case _ =>
                toText(expr) ~ "*"
          else
            val exprText = toText(expr)
            val line = exprText.lastLine
            val colon = if !line.isEmpty && isOperatorPart(line.last) then " :" else ":"
            exprText ~ colon ~ toText(tpt)
        }
      case NamedArg(name, arg) =>
        toText(name) ~ (if name.isTermName && arg.isType then " : " else " = ") ~ toText(arg)
      case Assign(lhs, rhs) =>
        changePrec(GlobalPrec) { toTextLocal(lhs) ~ " = " ~ toText(rhs) }
      case closureDef(meth) if !printDebug =>
        withEnclosingDef(meth):
          meth.paramss.map(paramsText).foldRight(toText(meth.rhs))(_ ~ " => " ~ _)
      case block: Block =>
        blockToText(block)
      case If(cond, thenp, elsep) =>
        val isInline = tree.isInstanceOf[Trees.InlineIf[?]]
        changePrec(GlobalPrec) {
          keywordStr(if (isInline) "inline if " else "if ") ~
          toText(cond) ~ (keywordText(" then") provided !cond.isInstanceOf[Parens]) ~~
          toText(thenp) ~ optText(elsep)(keywordStr(" else ") ~ _)
        }
      case Closure(env, ref, target) =>
        "closure(" ~ (toTextGlobal(env, ", ") ~ " | " provided env.nonEmpty) ~
        toTextGlobal(ref) ~ (":" ~ toText(target) provided !target.isEmpty) ~ ")"
      case Match(sel, cases) =>
        val isInline = tree.isInstanceOf[Trees.InlineMatch[?]]
        if (sel.isEmpty && !isInline) blockText(cases)
        else changePrec(GlobalPrec) {
          val selTxt: Text =
            if (isInline)
              if (sel.isEmpty) keywordStr("implicit")
              else keywordStr("inline ") ~ toText(sel)
            else toText(sel)
          selTxt ~ keywordStr(" match ") ~ blockText(cases)
        }
      case CaseDef(pat, guard, body) =>
        keywordStr("case ") ~ inPattern(toText(pat)) ~ optText(guard)(keywordStr(" if ") ~ _) ~ " => " ~ caseBlockText(body)
      case Labeled(bind, expr) =>
        changePrec(GlobalPrec) { toText(bind.name) ~ keywordStr("[") ~ toText(bind.symbol.info) ~ keywordStr("]: ") ~ toText(expr) }
      case Return(expr, from) =>
        val sym = from.symbol
        if (sym.is(Label))
          changePrec(GlobalPrec) { keywordStr("return[") ~ toText(sym.name) ~ keywordStr("]") ~ optText(expr)(" " ~ _) }
        else
          changePrec(GlobalPrec) { keywordStr("return") ~ optText(expr)(" " ~ _) }
      case WhileDo(cond, body) =>
        changePrec(GlobalPrec) { keywordStr("while ") ~ toText(cond) ~ keywordStr(" do ") ~ toText(body) }
      case Try(expr, cases, finalizer) =>
        changePrec(GlobalPrec) {
          keywordStr("try ") ~ toText(expr) ~ optText(cases)(keywordStr(" catch ") ~ _) ~ optText(finalizer)(keywordStr(" finally ") ~ _)
        }
      case Throw(expr) =>
        changePrec(GlobalPrec) {
          keywordStr("throw ") ~ toText(expr)
        }
      case SeqLiteral(elems, elemtpt) =>
        "[" ~ toTextGlobal(elems, ",") ~ " : " ~ toText(elemtpt) ~ "]"
      case tree @ Inlined(call, bindings, body) =>
        val bodyText = if bindings.isEmpty then toText(body) else blockText(bindings :+ body)
        if homogenizedView || !ctx.settings.XprintInline.value then bodyText
        else if tree.inlinedFromOuterScope then stringText("{{") ~ stringText("/* inlined from outside */") ~ bodyText ~ stringText("}}")
        else keywordText("{{") ~ keywordText("/* inlined from ") ~ toText(call) ~ keywordText(" */") ~ bodyText ~ keywordText("}}")
      case tpt: untpd.DerivedTypeTree =>
        ""
      case tree: TypeTree =>
        typeText(toText(tree.typeOpt))
        ~ Str("(inf)").provided(tree.isInferred && printDebug)
      case SingletonTypeTree(ref) =>
        toTextLocal(ref) ~ "." ~ keywordStr("type")
      case RefinedTypeTree(tpt, refines) =>
        if defn.isFunctionSymbol(tpt.symbol) && tree.hasType && !printDebug
        then changePrec(GlobalPrec) { toText(tree.typeOpt) }
        else toTextLocal(tpt) ~ blockText(refines)
      case AppliedTypeTree(tpt, args) =>
        if (tpt.symbol == defn.orType && args.length == 2)
          changePrec(OrTypePrec) { toText(args(0)) ~ " | " ~ atPrec(OrTypePrec + 1) { toText(args(1)) } }
        else if (tpt.symbol == defn.andType && args.length == 2)
          changePrec(AndTypePrec) { toText(args(0)) ~ " & " ~ atPrec(AndTypePrec + 1) { toText(args(1)) } }
        else if defn.isFunctionSymbol(tpt.symbol)
            && tpt.isInstanceOf[TypeTree] && tree.hasType && !printDebug
        then
          changePrec(GlobalPrec) { toText(tree.typeOpt) }
        else if tpt.symbol == defn.NamedTupleTypeRef.symbol
            && !printDebug && tree.typeOpt.exists
        then
          toText(tree.typeOpt)
        else args match
          case arg :: _ if arg.isTerm =>
            toTextLocal(tpt) ~ "(" ~ Text(args.map(argText), ", ") ~ ")"
          case _ =>
            toTextLocal(tpt) ~ "[" ~ Text(args.map(argText), ", ") ~ "]"
      case LambdaTypeTree(tparams, body) =>
        changePrec(GlobalPrec) {
          paramsText(tparams) ~ " =>> " ~ toText(body)
        }
      case TermLambdaTypeTree(params, body) =>
        changePrec(GlobalPrec) {
          paramsText(params) ~ " =>> " ~ toText(body)
        }
      case MatchTypeTree(bound, sel, cases) =>
        changePrec(GlobalPrec) {
          toText(sel) ~ keywordStr(" match ") ~ blockText(cases) ~
          (" <: " ~ toText(bound) provided !bound.isEmpty)
        }
      case ImpureByNameTypeTree(tpt) =>
        "=> " ~ toTextLocal(tpt)
      case ByNameTypeTree(tpt) =>
        (if Feature.pureFunsEnabled then "-> " else "=> ") ~ toTextLocal(tpt)
      case TypeBoundsTree(lo, hi, alias) =>
        if (lo eq hi) && alias.isEmpty then optText(lo)(" = " ~ _)
        else optText(lo)(" >: " ~ _) ~ optText(hi)(" <: " ~ _) ~ optText(alias)(" = " ~ _)
      case bind @ Bind(name, body) =>
        toTextOwner(bind) ~ keywordText("given ").provided(tree.symbol.isOneOf(GivenOrImplicit) && !homogenizedView) ~ // Used for scala.quoted.Type in quote patterns (not pickled)
        changePrec(InfixPrec) { nameIdText(bind) ~ " @ " ~ toText(body) }
      case Alternative(trees) =>
        changePrec(OrPrec) { toText(trees, " | ") }
      case UnApply(fun, implicits, patterns) =>
        val extractor = fun match {
          case Select(extractor, name) if name.isUnapplyName => extractor
          case _ => fun
        }
        toTextLocal(extractor) ~
        "(" ~ toTextGlobal(patterns, ", ") ~ ")" ~
        ("(" ~ toTextGlobal(implicits, ", ") ~ ")" provided implicits.nonEmpty)
      case tree @ ValDef(_, _, _) =>
        valDefToText(tree)
      case tree @ DefDef(_, _, _, _) =>
        defDefToText(tree)
      case tree @ TypeDef(name, rhs) =>
        def typeDefText(tparamsText: => Text, rhsText: => Text) =
          dclTextOr(tree) {
            modText(tree.mods, tree.symbol, keywordStr("type"), isType = true) ~~
            (varianceText(tree.mods) ~ typeText(nameIdText(tree))) ~
            withEnclosingDef(tree) { tparamsText ~ rhsText }
          }
        def recur(rhs: Tree, tparamsTxt: => Text, printMemberArgs: Boolean): Text = rhs match {
          case impl: Template =>
            templateText(tree, impl)
          case rhs: TypeBoundsTree =>
            typeDefText(tparamsTxt, toText(rhs))
          case LambdaTypeTree(tparams, body) if printMemberArgs =>
            recur(body, paramsText(tparams), false)
          case rhs: TypeTree if isBounds(rhs.typeOpt) =>
            typeDefText(tparamsTxt, toText(rhs))
          case rhs =>
            typeDefText(tparamsTxt, optText(rhs)(" = " ~ _))
        }
        recur(rhs, "", true)
      case tree @ Import(expr, selectors) =>
        myCtx = myCtx.importContext(tree, tree.symbol)
        keywordText("import ") ~ importText(expr, selectors)
      case Export(expr, selectors) =>
        keywordText("export ") ~ importText(expr, selectors)
      case ExtMethods(paramss, mdefs) =>
        addParamssText(keywordText("extension "), paramss)
        ~ " " ~ (if mdefs.length == 1 then toText(mdefs.head) else blockText(mdefs))
      case packageDef: PackageDef =>
        packageDefText(packageDef)
      case tree: Template =>
        toTextTemplate(tree)
      case Annotated(arg, annot) =>
        def captureSet =
          annot.asInstanceOf[tpd.Tree].toCaptureSet
        def toTextAnnot =
          toTextLocal(arg) ~~ annotText(annot.symbol.enclosingClass, annot)
        def toTextRetainsAnnot =
          try changePrec(GlobalPrec)(toTextLocal(arg) ~ "^" ~ toTextCaptureSet(captureSet))
          catch case ex: IllegalCaptureRef => toTextAnnot
        if annot.symbol.maybeOwner.isRetains
            && Feature.ccEnabled && !printDebug
            && Phases.checkCapturesPhase.exists // might be missing on -Ytest-pickler
        then toTextRetainsAnnot
        else if annot.symbol.enclosingClass == defn.IntoAnnot && !printDebug then
          atPrec(GlobalPrec):
            Str("into ") ~ toText(arg)
        else toTextAnnot
      case EmptyTree =>
        ""
      case TypedSplice(t) =>
        if (printDebug) "[" ~ toText(t) ~ "]#TS#"
        else toText(t)
      case tree @ ModuleDef(name, impl) =>
        withEnclosingDef(tree) {
          modText(tree.mods, NoSymbol, keywordStr("object"), isType = false) ~~
          nameIdText(tree) ~ toTextTemplate(impl)
        }
      case SymbolLit(str) =>
        "'" + str
      case InterpolatedString(id, segments) =>
        def strText(str: Literal) = withPos(escapedString(str.const.stringValue), tree.sourcePos)
        def segmentText(segment: Tree) = segment match {
          case Thicket(List(str: Literal, expr)) => strText(str) ~ "{" ~ toTextGlobal(expr) ~ "}"
          case str: Literal => strText(str)
        }
        toText(id) ~ "\"" ~ Text(segments map segmentText, "") ~ "\""
      case fn @ Function(args, body) =>
        var implicitSeen: Boolean = false
        var isGiven: Boolean = false
        val erasedParams = fn match {
          case fn: FunctionWithMods => fn.erasedParams
          case _ => fn.args.map(_ => false)
        }
        def argToText(arg: Tree, isErased: Boolean) = arg match {
          case arg @ ValDef(name, tpt, _) =>
            val implicitText =
              if ((arg.mods.is(Given))) { isGiven = true; "" }
              else if ((arg.mods.is(Implicit)) && !implicitSeen) { implicitSeen = true; keywordStr("implicit ") }
              else ""
            val erasedText = if isErased then keywordStr("erased ") else ""
            implicitText ~ erasedText ~ toText(name) ~ optAscription(tpt)
          case _ =>
            toText(arg)
        }
        val argsText = args match {
          case (arg @ ValDef(_, tpt, _)) :: Nil if tpt.isEmpty => argToText(arg, erasedParams(0))
          case _ =>
            "("
            ~ Text(args.zip(erasedParams).map(argToText), ", ")
            ~ ")"
        }
        val isPure =
          Feature.pureFunsEnabled
          && tree.match
            case tree: FunctionWithMods => !tree.mods.is(Impure)
            case _ => true
        argsText ~ " " ~ arrow(isGiven, isPure) ~ " " ~ toText(body)
      case PolyFunction(targs, body) =>
        val targsText = "[" ~ Text(targs.map((arg: Tree) => toText(arg)), ", ") ~ "]"
        changePrec(GlobalPrec) {
          targsText ~ " => " ~ toText(body)
        }
      case InfixOp(l, op, r) =>
        val opPrec = parsing.precedence(op.name)
        changePrec(opPrec) { toText(l) ~ " " ~ toText(op) ~ " " ~ toText(r) }
      case PostfixOp(l, op) =>
        if op.name == nme.CC_REACH then
          changePrec(DotPrec) { toText(l) ~ "*" }
        else
          changePrec(InfixPrec) { toText(l) ~ " " ~ toText(op) }
      case PrefixOp(op, r) =>
        changePrec(DotPrec) { toText(op) ~ " " ~ toText(r) }
      case Parens(t) =>
        "(" ~ toTextGlobal(t) ~ ")"
      case Tuple(ts) =>
        "(" ~ toTextGlobal(ts, ", ") ~ ")"
      case ForYield(enums, expr) =>
        forText(enums, expr, keywordStr(" yield "))
      case ForDo(enums, expr) =>
        forText(enums, expr, keywordStr(" do "))
      case GenFrom(pat, expr, checkMode) =>
        (Str("case ") provided checkMode == untpd.GenCheckMode.FilterAlways) ~
        toText(pat) ~ " <- " ~ toText(expr)
      case GenAlias(pat, expr) =>
        toText(pat) ~ " = " ~ toText(expr)
      case ContextBounds(bounds, cxBounds) =>
        if sourceVersion.isAtLeast(`3.6`) then
          def boundsText(bounds: Tree) = bounds match
            case ContextBoundTypeTree(tpt, _, ownName) =>
              toText(tpt) ~ (" as " ~ toText(ownName) `provided` !ownName.isEmpty)
            case bounds => toText(bounds)
          cxBounds match
            case bound :: Nil => ": " ~ boundsText(bound)
            case _ => ": {" ~ Text(cxBounds.map(boundsText), ", ") ~ "}"
        else
          cxBounds.foldLeft(toText(bounds)) {(t, cxb) =>
            t ~ cxBoundToText(cxb)
          }
      case PatDef(mods, pats, tpt, rhs) =>
        modText(mods, NoSymbol, keywordStr("val"), isType = false) ~~
        toText(pats, ", ") ~ optAscription(tpt) ~ optText(rhs)(" = " ~ _)
      case ParsedTry(expr, handler, finalizer) =>
        changePrec(GlobalPrec) {
          keywordStr("try ") ~ toText(expr) ~ " " ~ keywordStr("catch") ~ " {" ~ toText(handler) ~ "}" ~ optText(finalizer)(keywordStr(" finally ") ~ _)
        }
      case Number(digits, kind) =>
        digits
      case Thicket(trees) =>
        "Thicket {" ~~ toTextGlobal(trees, "\n") ~~ "}"
      case MacroTree(call) =>
        keywordStr("macro ") ~ toTextGlobal(call)
      case tree @ Quote(body, tags) =>
        val tagsText = (keywordStr("<") ~ toTextGlobal(tags, ", ") ~ keywordStr(">")).provided(tree.tags.nonEmpty)
        val exprTypeText = (keywordStr("[") ~ toTextGlobal(tpd.bodyType(tree.asInstanceOf[tpd.Quote])) ~ keywordStr("]")).provided(printDebug && tree.typeOpt.exists)
        val open = if (body.isTerm) keywordStr("{") else keywordStr("[")
        val close = if (body.isTerm) keywordStr("}") else keywordStr("]")
        keywordStr("'") ~ tagsText ~ exprTypeText ~ open ~ toTextGlobal(body) ~ close
      case Splice(expr) =>
        val spliceTypeText = (keywordStr("[") ~ toTextGlobal(tree.typeOpt) ~ keywordStr("]")).provided(printDebug && tree.typeOpt.exists)
        keywordStr("$") ~ spliceTypeText ~ keywordStr("{") ~ toTextGlobal(expr) ~ keywordStr("}")
      case tree @ QuotePattern(bindings, body, quotes) =>
        val quotesText = (keywordStr("<") ~ toText(quotes) ~ keywordStr(">")).provided(printDebug)
        val bindingsText = bindings.map(binding => {
          keywordStr("type ") ~ toText(binding.symbol.name) ~ toText(binding.symbol.info) ~ "; "
        }).reduceLeft(_ ~~ _).provided(bindings.nonEmpty)
        val open = if (body.isTerm) keywordStr("{") else keywordStr("[")
        val close = if (body.isTerm) keywordStr("}") else keywordStr("]")
        keywordStr("'") ~ quotesText ~ open ~ bindingsText ~ toTextGlobal(body) ~ close
      case SplicePattern(pattern, typeargs, args) =>
        val spliceTypeText = (keywordStr("[") ~ toTextGlobal(tree.typeOpt) ~ keywordStr("]")).provided(printDebug && tree.typeOpt.exists)
        keywordStr("$") ~ spliceTypeText ~ {
          if typeargs.isEmpty && args.isEmpty then keywordStr("{") ~ inPattern(toText(pattern)) ~ keywordStr("}")
          else if typeargs.isEmpty then toText(pattern) ~ "(" ~ toTextGlobal(args, ", ") ~ ")"
          else toText(pattern) ~ "[" ~  toTextGlobal(typeargs, ", ")~ "]" ~ "(" ~ toTextGlobal(args, ", ") ~ ")"
        }
      case Hole(isTerm, idx, args, content) =>
        val (prefix, postfix) = if isTerm then ("{{{", "}}}") else ("[[[", "]]]")
        val argsText = toTextGlobal(args, ", ")
        val contentText = toTextGlobal(content)
        val tpeText = toTextGlobal(tree.typeOpt)
        prefix ~~ idx.toString ~~ "|" ~~ tpeText ~~ "|" ~~ argsText ~~ "|" ~~ contentText ~~ postfix
      case CapturesAndResult(refs, parent) =>
        changePrec(GlobalPrec)("^{" ~ Text(refs.map(toText), ", ") ~ "}" ~ toText(parent))
      case ContextBoundTypeTree(tycon, pname, ownName) =>
        toText(pname) ~ " : " ~ toText(tycon) ~ (" as " ~ toText(ownName) `provided` !ownName.isEmpty)
      case _ =>
        tree.fallbackToText(this)
    }
  }

  override protected def toTextCapturing(tp: Type, refsText: Text, boxText: Text): Text = tp match
    case tp: AppliedType if defn.isFunctionSymbol(tp.typeSymbol) && !printDebug =>
      boxText ~ toTextFunction(tp, refsText)
    case tp: RefinedType if defn.isFunctionType(tp) && !printDebug =>
      boxText ~ toTextMethodAsFunction(tp.refinedInfo, isPure = !tp.typeSymbol.name.isImpureFunction, refsText)
    case _ =>
      super.toTextCapturing(tp, refsText, boxText)

  override def toText[T <: Untyped](tree: Tree[T]): Text = controlled {
    import untpd.*

    var txt = toTextCore(tree)

    def suppressTypes =
      tree.isType || tree.isDef || // don't print types of types or defs
      homogenizedView && ctx.mode.is(Mode.Pattern)
        // When comparing pickled info, disregard types of patterns.
        // The reason is that GADT matching can rewrite types of pattern trees
        // without changing the trees themselves. (see Typer.typedCase.indexPatterns.transform).
        // But then pickling and unpickling the original trees will yield trees
        // with the original types before they are rewritten, which causes a discrepancy.

    def suppressPositions = tree match {
      case _: WithoutTypeOrPos[?] | _: TypeTree => true // TypeTrees never have an interesting position
      case _ => false
    }

    if (ctx.settings.XprintTypes.value && tree.hasType) {
      // add type to term nodes; replace type nodes with their types unless -Yprint-pos is also set.
      val tp1 = tree.typeOpt match {
        case tp: TermRef if tree.isInstanceOf[RefTree] && !tp.denot.isOverloaded => tp.underlying
        case tp => tp
      }
      val tp2 = {
        val tp = tp1.tryNormalize
        if (tp != NoType) tp else tp1
      }
      val tp3 =
        if (homogenizedView && tree.isInstanceOf[If | Match | Annotated | Block | CaseDef]) {
          // Types of non-leaf trees are not pickled but reconstructed when
          // unpickled using the TypeAssigner. Sometimes, this requires choosing
          // arbitrarily between two =:= types (e.g., when typing an `if`, where
          // one branch is typed with a type alias and the other with a
          // dealiased version of the same type) and we cannot guarantee that
          // the same choice was made by the original Typer (e.g., because the
          // original choice involved type variables). So we need to get rid of
          // any alias in these types to make -Ytest-pickler work (the list of
          // types in the isInstanceOf check above is conservative and might
          // need to be expanded).
          val dealiasMap = new TypeMap {
            def apply(tp: Type) = mapOver(tp.dealias)
          }
          dealiasMap(tp2)
        }
        else tp2

      if (!suppressTypes)
        txt = ("<" ~ txt ~ ":" ~ toText(tp3) ~ ">").close
      else if (tree.isType && !homogenizedView)
        txt = toText(tp3)
    }
    if (!suppressPositions) {
      if (printPos) {
        val posStr =
          if (homogenizedView || debugPos)
            if (tree.isInstanceOf[MemberDef]) Str(s"${tree.source}${tree.span}")
            else Str(s"${tree.source}${tree.span.toSynthetic}")
          else
            "<" ~ toText(tree.sourcePos) ~ ">"
        val clsStr = ""//if (tree.isType) tree.getClass.toString else ""
        txt = (txt ~ "@" ~ posStr ~ clsStr).close
      }
      if (ctx.settings.YprintPosSyms.value && tree.isDef)
        txt = (txt ~
          s"@@(${tree.symbol.name}=" ~ tree.symbol.sourcePos.toString ~ ")").close
    }
    if (ctx.settings.YshowTreeIds.value)
      txt = (txt ~ "#" ~ tree.uniqueId.toString).close
    tree match {
      case Block(_, _) | Template(_, _, _, _) => txt
      case _ => txt.close
    }
  }

  /** Print modifiers from symbols if tree has type, overriding the behavior in Trees. */
  extension (mdef: untpd.DefTree) def mods: untpd.Modifiers =
    if mdef.hasType then Modifiers(mdef.symbol) else mdef.rawMods

  private def Modifiers(sym: Symbol): Modifiers = untpd.Modifiers(
    sym.flagsUNSAFE & (if (sym.isType) ModifierFlags | VarianceFlags else ModifierFlags),
    if (sym.privateWithin.exists) sym.privateWithin.asType.name else tpnme.EMPTY,
    sym.annotationsUNSAFE.filterNot(ann => dropAnnotForModText(ann.symbol)).map(_.tree))

  protected def dropAnnotForModText(sym: Symbol): Boolean = sym == defn.BodyAnnot

  protected def optAscription[T <: Untyped](tpt: Tree[T]): Text = optText(tpt)(": " ~ _)

  private def idText(tree: untpd.Tree): Text =
    (if showUniqueIds && tree.hasType && tree.symbol.exists then s"#${tree.symbol.id}" else "") ~
    (if showNestingLevel then tree.typeOpt match
      case tp: NamedType if !tp.symbol.isStatic => s"%${tp.symbol.nestingLevel}"
      case tp: TypeVar => s"%${tp.nestingLevel}"
      case tp: TypeParamRef => ctx.typerState.constraint.typeVarOfParam(tp) match
        case tvar: TypeVar => s"%${tvar.nestingLevel}"
        case _ => ""
      case _ => ""
     else "")

  private def useSymbol(tree: untpd.Tree) =
    tree.hasType && tree.symbol.exists && ctx.settings.YprintSyms.value

  protected def nameIdText[T <: Untyped](tree: NameTree[T]): Text =
    if (tree.hasType && tree.symbol.exists) {
      val str = nameString(tree.symbol)
      tree match {
        case tree: RefTree => withPos(str, tree.sourcePos)
        case tree: MemberDef => withPos(str, tree.sourcePos.withSpan(tree.nameSpan))
        case _ => str
      }
    }
    else toText(tree.name) ~ idText(tree)

  private def toTextOwner(tree: Tree[?]) =
    "[owner = " ~ tree.symbol.maybeOwner.show ~ "]" provided ctx.settings.YprintDebugOwners.value

  protected def dclTextOr[T <: Untyped](tree: Tree[T])(treeText: => Text): Text =
    toTextOwner(tree) ~ {
      if (useSymbol(tree)) annotsText(tree.symbol) ~~ dclText(tree.symbol)
      else treeText
    }

  def paramsText[T <: Untyped](params: ParamClause[T]): Text = (params: @unchecked) match
    case Nil =>
      "()"
    case untpd.ValDefs(vparams @ (vparam :: _)) =>
      "(" ~ keywordText("using ").provided(vparam.mods.is(Given))
          ~ toText(vparams, ", ") ~ ")"
    case untpd.TypeDefs(tparams) =>
      "[" ~ toText(tparams, ", ") ~ "]"

  def addParamssText[T <: Untyped](leading: Text, paramss: List[ParamClause[T]]): Text =
    paramss.foldLeft(leading)((txt, params) => txt ~ paramsText(params))

  protected def valDefToText[T <: Untyped](tree: ValDef[T]): Text = {
    dclTextOr(tree) {
      modText(tree.mods, tree.symbol, keywordStr(if (tree.mods.is(Mutable)) "var" else "val"), isType = false) ~~
        valDefText(nameIdText(tree)) ~ optAscription(tree.tpt) ~
        withEnclosingDef(tree) { rhsValDef(tree) }
    }
  }

  protected def defDefToText[T <: Untyped](tree: DefDef[T]): Text = {
    import untpd.*
    dclTextOr(tree) {
      val defKeyword = modText(tree.mods, tree.symbol, keywordStr("def"), isType = false)
      val isExtension = tree.hasType && tree.symbol.is(ExtensionMethod)
      withEnclosingDef(tree) {
        val coreSig =
          if isExtension then
            val paramss =
              if tree.name.isRightAssocOperatorName then
                // If you change the names of the clauses below, also change them in right-associative-extension-methods.md
                // we have the following encoding of tree.paramss:
                //   (leftTyParams ++ leadingUsing
                //      ++ rightTyParams ++ rightParam
                //      ++ leftParam ++ trailingUsing ++ rest)
                //   e.g.
                //     extension [A](using B)(c: C)(using D)
                //       def %:[E](f: F)(g: G)(using H): Res = ???
                //   will have the following values:
                //   - leftTyParams = List(`[A]`)
                //   - leadingUsing = List(`(using B)`)
                //   - rightTyParams = List(`[E]`)
                //   - rightParam = List(`(f: F)`)
                //   - leftParam = List(`(c: C)`)
                //   - trailingUsing = List(`(using D)`)
                //   - rest = List(`(g: G)`, `(using H)`)
                // we need to swap (rightTyParams ++ rightParam) with (leftParam ++ trailingUsing)
                val (leftTyParams, rest1) = tree.paramss match
                  case fst :: tail if isTypeParamClause(fst) => (List(fst), tail)
                  case other => (List(), other)
                val (leadingUsing, rest2) = rest1.span(isUsingClause)
                val (rightTyParams, rest3) = rest2.span(isTypeParamClause)
                val (rightParam, rest4) = rest3.splitAt(1)
                val (leftParam, rest5) = rest4.splitAt(1)
                val (trailingUsing, rest6) = rest5.span(isUsingClause)
                if leftParam.nonEmpty then
                  leftTyParams ::: leadingUsing ::: leftParam ::: trailingUsing ::: rightTyParams ::: rightParam ::: rest6
                else
                  tree.paramss // it wasn't a binary operator, after all.
              else
                tree.paramss
            val trailingParamss = paramss
              .dropWhile(isUsingOrTypeParamClause)
              .drop(1)
              .dropWhile(isUsingClause)
            val leadingParamss = paramss.take(paramss.length - trailingParamss.length)
            addParamssText(
              addParamssText(keywordStr("extension "), leadingParamss)
              ~~ (defKeyword ~~ valDefText(nameIdText(tree))).close,
             trailingParamss)
          else
            addParamssText(defKeyword ~~ valDefText(nameIdText(tree)), tree.paramss)

        coreSig
        ~ optAscription(tree.tpt)
        ~ rhsDefDef(tree)
      }
    }
  }

  /** Inspect the rhs of a ValDef, overridden in OutlinePrinter */
  protected def rhsValDef[T <: Untyped](tree: ValDef[T]): Text =
    optText(tree.rhs)(" = " ~ _)

  /** Inspect the rhs of a DefDef, overridden in OutlinePrinter */
  protected def rhsDefDef[T <: Untyped](tree: DefDef[T]): Text =
    optText(tree.rhs)(" = " ~ keywordText("macro ").provided(tree.symbol.isScala2Macro) ~ _)

  protected def toTextTemplate(impl: Template, ofNew: Boolean = false): Text = {
    val Template(constr @ DefDef(_, paramss, _, _), _, self, _) = impl
    val tparamsTxt = withEnclosingDef(constr) {
      paramsText(constr.leadingTypeParams) provided constr.leadingTypeParams.nonEmpty
    }
    val primaryConstrs = if (constr.rhs.isEmpty) Nil else constr :: Nil
    val prefix: Text =
      if (constr.trailingParamss.isEmpty || primaryConstrs.nonEmpty) tparamsTxt
      else {
        var modsText = modText(constr.mods, constr.symbol, "", isType = false)
        if (!modsText.isEmpty) modsText = " " ~ modsText
        if (constr.mods.hasAnnotations && !constr.mods.hasFlags) modsText = modsText ~~ " this"
        val ctorParamss =
          // for fake `(x$1: Unit): Foo` constructor, don't print the param (span is not reconstructed correctly)
          if constr.symbol.isAllOf(JavaParsers.fakeFlags) then Nil else constr.trailingParamss
        withEnclosingDef(constr) { addParamssText(tparamsTxt ~~ modsText, ctorParamss) }
      }
    val parentsText = Text(impl.parents.map(constrText), if (ofNew) keywordStr(" with ") else ", ")
    val derivedText = Text(impl.derived.map(toText(_)), ", ")
    val selfText = {
      val selfName = if (self.name == nme.WILDCARD) keywordStr("this") else self.name.toString
      (selfName ~ optText(self.tpt)(": " ~ _) ~ " =>").close
    }.provided(!self.isEmpty)
    val body = if (ctx.settings.YtestPickler.value) {
      // Pickling/unpickling reorders the body members, so we need to homogenize
      val (params, rest) = impl.body partition {
        case stat: TypeDef => stat.symbol.is(Param)
        case stat: ValOrDefDef =>
          val sym = stat.symbol
          sym.is(ParamAccessor) && !sym.isSetter
          || sym.isAllOf(JavaParsers.fakeFlags | Param)
        case _ => false
      }
      val params0 =
        if constr.symbol.isAllOf(JavaParsers.fakeFlags) then
          // filter out fake param accessors
          params.filterNot(_.symbol.isAllOf(JavaParsers.fakeFlags | Param))
        else
          params
      params0 ::: rest
    }
    else impl.body

    val bodyText = inContextBracket(
      " {" ~~ selfText ~ toTextGlobal(primaryConstrs ::: body, "\n") ~ "}")

    prefix ~
    keywordText(" extends").provided(!ofNew && impl.parents.nonEmpty) ~~ parentsText ~
    keywordText(" derives").provided(impl.derived.nonEmpty) ~~ derivedText ~
    bodyText
  }

  protected def templateText(tree: TypeDef, impl: Template): Text = {
    val kw = if tree.mods.is(Trait) then "trait" else "class"
    val decl = modText(tree.mods, tree.symbol, keywordStr(kw), isType = true)
    ( decl ~~ typeText(nameIdText(tree)) ~ withEnclosingDef(tree) { toTextTemplate(impl) }
    // ~ (if (tree.hasType && printDebug) i"[decls = ${tree.symbol.info.decls}]" else "") // uncomment to enable
    )
  }

  protected def toTextPackageId[T <: Untyped](pid: Tree[T]): Text =
    if (homogenizedView && pid.hasType) toTextLocal(pid.typeOpt)
    else toTextLocal(pid)

  protected def packageDefText(tree: PackageDef): Text = {
    val statsText = tree.stats match {
      case (pdef: PackageDef) :: Nil => toText(pdef)
      case _ => inContextBracket(toTextGlobal(tree.stats, "\n"))
    }
    val bodyText =
      if (currentPrecedence == TopLevelPrec) "\n" ~ statsText else " {" ~ statsText ~ "}"
    keywordStr("package ") ~ toTextPackageId(tree.pid) ~ bodyText
  }

  /** Textual representation of an instance creation expression without the leading `new` */
  protected def constrText(tree: untpd.Tree): Text = toTextLocal(tree).stripPrefix(keywordStr("new ")) // DD

  protected def annotText(sym: Symbol, tree: untpd.Tree): Text =
    def recur(t: untpd.Tree): Text = t match
      case Apply(fn, Nil) => recur(fn)
      case Apply(fn, args) =>
        val explicitArgs = args.filterNot(_.symbol.name.is(DefaultGetterName))
        recur(fn) ~ "(" ~ toTextGlobal(explicitArgs, ", ") ~ ")"
      case TypeApply(fn, args) => recur(fn) ~ "[" ~ toTextGlobal(args, ", ") ~ "]"
      case Select(qual, nme.CONSTRUCTOR) => recur(qual)
      case id @ Ident(tpnme.BOUNDTYPE_ANNOT) => "@" ~ toText(id.symbol.name)
      case New(tpt) => recur(tpt)
      case _ =>
        val annotSym = sym.orElse(tree.symbol.enclosingClass)
        s"@${if annotSym.exists then annotSym.name.toString else t.show}"
    recur(tree)

  protected def modText(mods: untpd.Modifiers, sym: Symbol, kw: String, isType: Boolean): Text = { // DD
    val suppressKw = if (enclDefIsClass) mods.isAllOf(LocalParam) else mods.is(Param)
    var flagMask =
      if (ctx.settings.YdebugFlags.value) AnyFlags
      else if (suppressKw) PrintableFlags(isType) &~ Private
      else PrintableFlags(isType)
    if (homogenizedView && mods.flags.isTypeFlags) flagMask &~= GivenOrImplicit // drop implicit/given from classes
    val rawFlags = if (sym.exists) sym.flagsUNSAFE else mods.flags
    if (rawFlags.is(Param)) flagMask = flagMask &~ Given
    val flags = rawFlags & flagMask
    var flagsText = toTextFlags(sym, flags)
    val annotTexts =
      if sym.exists then
        sym.annotationsUNSAFE.filterNot(ann => dropAnnotForModText(ann.symbol)).map(toText)
      else
        mods.annotations.filterNot(tree => dropAnnotForModText(tree.symbol)).map(annotText(NoSymbol, _))
    Text(annotTexts, " ") ~~ flagsText ~~ (Str(kw) provided !suppressKw)
  }

  override def annotText(annot: Annotation): Text = annotText(annot.symbol, annot.tree)

  def optText(name: Name)(encl: Text => Text): Text =
    if (name.isEmpty) "" else encl(toText(name))

  def optText[T <: Untyped](tree: Tree[T])(encl: Text => Text): Text =
    if (tree.isEmpty) "" else encl(toText(tree))

  def optText[T <: Untyped](tree: List[Tree[T]])(encl: Text => Text): Text =
    if (tree.exists(!_.isEmpty)) encl(blockText(tree)) else ""

  override protected def treatAsTypeParam(sym: Symbol): Boolean = sym.is(TypeParam)

  override protected def treatAsTypeArg(sym: Symbol): Boolean =
    sym.isType && (sym.isAllOf(ProtectedLocal)) &&
      (sym.allOverriddenSymbols exists (_.is(TypeParam)))

  override def toText(sym: Symbol): Text = {
    if (sym.isImport)
      sym.infoOrCompleter match {
        case info: Namer#Completer => return info.original.show
        case info: ImportType => return s"import ${info.expr.show}"
        case _ =>
      }
    def name =
      if (printDebug)
        nameString(sym)
      else if sym.is(Package) then
        fullNameString(sym)
      else if (sym.is(ModuleClass) && sym.isPackageObject && sym.name.stripModuleClassSuffix == tpnme.PACKAGE)
        nameString(sym.owner.name)
      else if (sym.is(ModuleClass) && sym.isTopLevelDefinitionsObject)
        nameString(sym.owner.name)
      else if (sym.is(ModuleClass))
        nameString(sym.name.stripModuleClassSuffix) + idString(sym)
      else if (hasMeaninglessName(sym))
        simpleNameString(sym.owner) + idString(sym)
      else
        nameString(sym)

    if sym.is(ModuleClass) && sym.isTopLevelDefinitionsObject then
      "the top-level definitions in package " + nameString(sym.owner.name)
    else (keywordText(kindString(sym)) ~~ {
      if (sym.isAnonymousClass)
        toTextParents(sym.info.parents) ~~ "{...}"
      else
        typeText(name)
    }).close
  }

  /** String representation of symbol's kind. */
  override def kindString(sym: Symbol): String = {
    val flags = sym.flagsUNSAFE
    if (flags.is(Package)) "package"
    else if (sym.isPackageObject) "package object"
    else if (flags.is(Module)) "object"
    else if (sym.isClassConstructor) "constructor"
    else super.kindString(sym)
  }

  /** String representation of symbol's definition keyword */
  override protected def keyString(sym: Symbol): String = {
    val flags = sym.flagsUNSAFE
    if (sym.isType && sym.owner.isTerm) ""
    else if (sym.isPackageObject) "package object"
    else if (flags.is(Module) && flags.is(Case)) "case object"
    else if (sym.isClass && flags.is(Case)) "case class"
    else super.keyString(sym)
  }

  override def toTextFlags(sym: Symbol): Text =
    if (ctx.settings.YdebugFlags.value)
      super.toTextFlags(sym)
    else {
      var flags = sym.flagsUNSAFE
      if (flags.is(TypeParam)) flags = flags &~ Protected
      toTextFlags(sym, flags & PrintableFlags(sym.isType))
    }

  override def toText(denot: Denotation): Text = denot match {
    case denot: MultiDenotation => Text(denot.alternatives.map(dclText), "  ")
    case NoDenotation => "NoDenotation"
    case _ =>
      if (denot.symbol.exists) toText(denot.symbol)
      else "some " ~ toText(denot.info)
  }

  override def plain: PlainPrinter = new PlainPrinter(_ctx)

  private def withPos(txt: Text, pos: SourcePosition): Text =
    if (!printLines || !pos.exists) txt
    else txt match {
      case Str(s, _) => Str(s, LineRange(pos.line, pos.endLine))
      case _ => txt
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy