@@ -78,7 +78,7 @@ trait QuotesAndSplices {
7878 def typedSplice (tree : untpd.Splice , pt : Type )(using Context ): Tree = {
7979 record(" typedSplice" )
8080 checkSpliceOutsideQuote(tree)
81- assert(! ctx.mode.is( Mode . QuotedPattern ) )
81+ assert(! ctx.mode.isQuotedPattern )
8282 tree.expr match {
8383 case untpd.Quote (innerExpr, Nil ) if innerExpr.isTerm =>
8484 report.warning(" Canceled quote directly inside a splice. ${ '{ XYZ } } is equivalent to XYZ." , tree.srcPos)
@@ -110,8 +110,6 @@ trait QuotesAndSplices {
110110 def typedSplicePattern (tree : untpd.SplicePattern , pt : Type )(using Context ): Tree = {
111111 record(" typedSplicePattern" )
112112 if isFullyDefined(pt, ForceDegree .flipBottom) then
113- def patternOuterContext (ctx : Context ): Context =
114- if (ctx.mode.is(Mode .QuotedPattern )) patternOuterContext(ctx.outer) else ctx
115113 val typedArgs = withMode(Mode .InQuotePatternHoasArgs ) {
116114 tree.args.map {
117115 case arg : untpd.Ident =>
@@ -125,8 +123,7 @@ trait QuotesAndSplices {
125123 report.error(" References to `var`s cannot be used in higher-order pattern" , arg.srcPos)
126124 val argTypes = typedArgs.map(_.tpe.widenTermRefExpr)
127125 val patType = if tree.args.isEmpty then pt else defn.FunctionOf (argTypes, pt)
128- val pat = typedPattern(tree.body, defn.QuotedExprClass .typeRef.appliedTo(patType))(
129- using spliceContext.retractMode(Mode .QuotedPattern ).addMode(Mode .Pattern ).withOwner(patternOuterContext(ctx).owner))
126+ val pat = typedPattern(tree.body, defn.QuotedExprClass .typeRef.appliedTo(patType))(using quotePatternSpliceContext)
130127 val baseType = pat.tpe.baseType(defn.QuotedExprClass )
131128 val argType = if baseType.exists then baseType.argTypesHi.head else defn.NothingType
132129 untpd.cpy.SplicePattern (tree)(pat, typedArgs).withType(pt)
@@ -145,7 +142,7 @@ trait QuotesAndSplices {
145142 * The prototype must be fully defined to be able to infer the type of `R`.
146143 */
147144 def typedAppliedSplice (tree : untpd.Apply , pt : Type )(using Context ): Tree = {
148- assert(ctx.mode.is( Mode . QuotedPattern ) )
145+ assert(ctx.mode.isQuotedPattern )
149146 val untpd .Apply (splice : untpd.SplicePattern , args) = tree : @ unchecked
150147 def isInBraces : Boolean = splice.span.end != splice.body.span.end
151148 if isInBraces then // ${x}(...) match an application
@@ -166,26 +163,37 @@ trait QuotesAndSplices {
166163 val typeSymInfo = pt match
167164 case pt : TypeBounds => pt
168165 case _ => TypeBounds .empty
166+
167+ def warnOnInferredBounds (typeSym : Symbol ) =
168+ if ! (typeSymInfo =:= TypeBounds .empty) && ! (typeSym.info <:< typeSymInfo) then
169+ val (openQuote, closeQuote) = if ctx.mode.is(Mode .QuotedExprPattern ) then (" '{" , " }" ) else (" '[" , " ]" )
170+ if ctx.mode.is(Mode .QuotedTypePattern ) then // TODO remove this branch once SIP-53 is non-experimental
171+ report.warning(
172+ em """ Ignored bound $typeSymInfo
173+ |
174+ |This kind of pattern will be supported with SIP-53.
175+ |SIP-53: https://docs.scala-lang.org/sips/quote-pattern-type-variable-syntax.html
176+ | """ , tree.srcPos)
177+ else
178+ report.warning(em " Ignored bound $typeSymInfo\n\n Consider defining bounds explicitly: \n $openQuote $typeSym${typeSym.info & typeSymInfo}; ... $closeQuote" , tree.srcPos)
179+
169180 getQuotedPatternTypeVariable(tree.name.asTypeName) match
170181 case Some (typeSym) =>
171182 checkExperimentalFeature(
172183 " support for multiple references to the same type (without backticks) in quoted type patterns (SIP-53)" ,
173184 tree.srcPos,
174185 " \n\n SIP-53: https://docs.scala-lang.org/sips/quote-pattern-type-variable-syntax.html" )
175- if ! (typeSymInfo =:= TypeBounds .empty) && ! (typeSym.info <:< typeSymInfo) then
176- report.warning(em " Ignored bound $typeSymInfo\n\n Consider defining bounds explicitly `'{ $typeSym${typeSym.info & typeSymInfo}; ... }` " , tree.srcPos)
186+ warnOnInferredBounds(typeSym)
177187 ref(typeSym)
178188 case None =>
179- def spliceOwner (ctx : Context ): Symbol =
180- if (ctx.mode.is(Mode .QuotedPattern )) spliceOwner(ctx.outer) else ctx.owner
189+ val spliceContext = quotePatternSpliceContext
181190 val name = tree.name.toTypeName
182191 val nameOfSyntheticGiven = PatMatGivenVarName .fresh(tree.name.toTermName)
183192 val expr = untpd.cpy.Ident (tree)(nameOfSyntheticGiven)
184- val typeSym = newSymbol(spliceOwner(ctx) , name, EmptyFlags , typeSymInfo, NoSymbol , tree.span)
193+ val typeSym = newSymbol(spliceContext.owner , name, EmptyFlags , typeSymInfo, NoSymbol , tree.span)
185194 typeSym.addAnnotation(Annotation (New (ref(defn.QuotedRuntimePatterns_patternTypeAnnot .typeRef)).withSpan(tree.span)))
186195 addQuotedPatternTypeVariable(typeSym)
187- val pat = typedPattern(expr, defn.QuotedTypeClass .typeRef.appliedTo(typeSym.typeRef))(
188- using spliceContext.retractMode(Mode .QuotedPattern ).withOwner(spliceOwner(ctx)))
196+ val pat = typedPattern(expr, defn.QuotedTypeClass .typeRef.appliedTo(typeSym.typeRef))(using spliceContext)
189197 pat.select(tpnme.Underlying )
190198
191199 private def checkSpliceOutsideQuote (tree : untpd.Tree )(using Context ): Unit =
@@ -454,7 +462,7 @@ trait QuotesAndSplices {
454462 " \n\n SIP-53: https://docs.scala-lang.org/sips/quote-pattern-type-variable-syntax.html" )
455463
456464 val (typeTypeVariables, patternCtx) =
457- val quoteCtx = quotePatternContext()
465+ val quoteCtx = quotePatternContext(quoted.isType )
458466 if untpdTypeVariables.isEmpty then (Nil , quoteCtx)
459467 else typedBlockStats(untpdTypeVariables)(using quoteCtx)
460468
@@ -543,13 +551,26 @@ object QuotesAndSplices {
543551 def getQuotedPatternTypeVariable (name : TypeName )(using Context ): Option [Symbol ] =
544552 ctx.property(TypeVariableKey ).get.get(name)
545553
546- /** Get the symbol for the quoted pattern type variable if it exists */
554+ /** Get the symbol for the quoted pattern type variable if it exists */
547555 def addQuotedPatternTypeVariable (sym : Symbol )(using Context ): Unit =
548556 ctx.property(TypeVariableKey ).get.update(sym.name.asTypeName, sym)
549557
550- /** Context used to type the contents of a quoted */
551- def quotePatternContext ()(using Context ): Context =
558+ /** Context used to type the contents of a quote pattern */
559+ def quotePatternContext (isTypePattern : Boolean )(using Context ): Context =
552560 quoteContext.fresh.setNewScope
553- .addMode(Mode .QuotedPattern ).retractMode(Mode .Pattern )
561+ .addMode(if isTypePattern then Mode .QuotedTypePattern else Mode .QuotedExprPattern )
562+ .retractMode(Mode .Pattern ) // TODO do we need Mode.QuotedPattern?
554563 .setProperty(TypeVariableKey , collection.mutable.Map .empty)
564+
565+ /** Context used to type the contents of a quote pattern splice */
566+ def quotePatternSpliceContext (using Context ): Context =
567+ spliceContext
568+ .retractMode(Mode .QuotedPatternBits )
569+ .addMode(Mode .Pattern )
570+ .withOwner(quoteOwner(ctx))
571+
572+ /** Owner of the quoted pattern */
573+ private def quoteOwner (ctx : Context ): Symbol =
574+ if ctx.mode.isQuotedPattern then quoteOwner(ctx.outer) else ctx.owner
575+
555576}
0 commit comments