// Copyright (c) Microsoft Corporation. All rights reserved. See License.txt in the project root for license information. using System.Web.Razor.Editor; using System.Web.Razor.Parser; using System.Web.Razor.Parser.SyntaxTree; using System.Web.Razor.Resources; using System.Web.Razor.Test.Framework; using System.Web.Razor.Tokenizer.Symbols; using Xunit; namespace System.Web.Razor.Test.Parser.CSharp { public class CSharpToMarkupSwitchTest : CsHtmlCodeParserTestBase { [Fact] public void SingleAngleBracketDoesNotCauseSwitchIfOuterBlockIsTerminated() { ParseBlockTest("{ List< }", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code(" List< ").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None))); } [Fact] public void ParseBlockGivesSpacesToCodeOnAtTagTemplateTransitionInDesignTimeMode() { ParseBlockTest(@"Foo( @
Foo
)", new ExpressionBlock( Factory.Code(@"Foo( ") .AsImplicitExpression(CSharpCodeParser.DefaultKeywords) .Accepts(AcceptedCharacters.Any), new TemplateBlock( new MarkupBlock( Factory.MarkupTransition(), Factory.Markup("Foo
").Accepts(AcceptedCharacters.None) ) ), Factory.Code(@" )") .AsImplicitExpression(CSharpCodeParser.DefaultKeywords) .Accepts(AcceptedCharacters.NonWhiteSpace) ), designTimeParser: true); } [Fact] public void ParseBlockGivesSpacesToCodeOnAtColonTemplateTransitionInDesignTimeMode() { ParseBlockTest(@"Foo( @:Foo
)", new ExpressionBlock( Factory.Code("Foo( \r\n").AsImplicitExpression(CSharpCodeParser.DefaultKeywords), new TemplateBlock( new MarkupBlock( Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Foo
\r\n") .With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ) ), Factory.Code(@")") .AsImplicitExpression(CSharpCodeParser.DefaultKeywords) .Accepts(AcceptedCharacters.NonWhiteSpace) ), designTimeParser: true); } [Fact] public void ParseBlockGivesSpacesToCodeOnTagTransitionInDesignTimeMode() { ParseBlockTest(@"{Foo
}", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code("\r\n ").AsStatement(), new MarkupBlock( Factory.Markup("Foo
").Accepts(AcceptedCharacters.None) ), Factory.Code(" \r\n").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None) ), designTimeParser: true); } [Fact] public void ParseBlockGivesSpacesToCodeOnInvalidAtTagTransitionInDesignTimeMode() { ParseBlockTest(@"{ @Foo
}", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code("\r\n ").AsStatement(), new MarkupBlock( Factory.MarkupTransition(), Factory.Markup("Foo
").Accepts(AcceptedCharacters.None) ), Factory.Code(" \r\n").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None) ), true, new RazorError(RazorResources.ParseError_AtInCode_Must_Be_Followed_By_Colon_Paren_Or_Identifier_Start, 7, 1, 4)); } [Fact] public void ParseBlockGivesSpacesToCodeOnAtColonTransitionInDesignTimeMode() { ParseBlockTest(@"{ @:Foo
}", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code("\r\n ").AsStatement(), new MarkupBlock( Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Foo
\r\n") .With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ), Factory.EmptyCSharp().AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None) ), designTimeParser: true); } [Fact] public void ParseBlockShouldSupportSingleLineMarkupContainingStatementBlock() { ParseBlockTest(@"Repeat(10, @: @{} )", new ExpressionBlock( Factory.Code("Repeat(10,\r\n ") .AsImplicitExpression(CSharpCodeParser.DefaultKeywords), new TemplateBlock( new MarkupBlock( Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup(" ") .With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString)), new StatementBlock( Factory.CodeTransition(), Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.EmptyCSharp().AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None) ), Factory.Markup("\r\n") .With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ) ), Factory.Code(")") .AsImplicitExpression(CSharpCodeParser.DefaultKeywords) .Accepts(AcceptedCharacters.NonWhiteSpace) )); } [Fact] public void ParseBlockShouldSupportMarkupWithoutPreceedingWhitespace() { ParseBlockTest(@"foreach(var file in files){ @:BazFoo @bar
@:Hello! var biz = boz; }", new StatementBlock( Factory.Code("if(foo) {\r\n var foo = \"After this statement there are 10 spaces\"; \r\n").AsStatement(), new MarkupBlock( Factory.Markup("\r\n Foo\r\n"), new ExpressionBlock( Factory.Code(" ").AsStatement(), Factory.CodeTransition(), Factory.Code(@"bar").AsImplicitExpression(CSharpCodeParser.DefaultKeywords).Accepts(AcceptedCharacters.NonWhiteSpace) ), Factory.Markup("\r\n
\r\n").Accepts(AcceptedCharacters.None) ), new MarkupBlock( Factory.Markup(@" "), Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Hello!\r\n").With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ), Factory.Code(" var biz = boz;\r\n}").AsStatement())); } [Fact] public void ParseBlockAllowsMarkupInIfBodyWithBraces() { ParseBlockTest("if(foo) {Bar
} else if(bar) {Baz
} else {Boz
}", new StatementBlock( Factory.Code("if(foo) {").AsStatement(), new MarkupBlock( Factory.Markup("Bar
").Accepts(AcceptedCharacters.None) ), Factory.Code("} else if(bar) {").AsStatement(), new MarkupBlock( Factory.Markup("Baz
").Accepts(AcceptedCharacters.None) ), Factory.Code("} else {").AsStatement(), new MarkupBlock( Factory.Markup("Boz
").Accepts(AcceptedCharacters.None) ), Factory.Code("}").AsStatement().Accepts(AcceptedCharacters.None) )); } [Fact] public void ParseBlockAllowsMarkupInIfBodyWithBracesWithinCodeBlock() { ParseBlockTest("{ if(foo) {Bar
} else if(bar) {Baz
} else {Boz
} }", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code(" if(foo) {").AsStatement(), new MarkupBlock( Factory.Markup("Bar
").Accepts(AcceptedCharacters.None) ), Factory.Code("} else if(bar) {").AsStatement(), new MarkupBlock( Factory.Markup("Baz
").Accepts(AcceptedCharacters.None) ), Factory.Code("} else {").AsStatement(), new MarkupBlock( Factory.Markup("Boz
").Accepts(AcceptedCharacters.None) ), Factory.Code("} ").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None) )); } [Fact] public void ParseBlockSupportsMarkupInCaseAndDefaultBranchesOfSwitch() { // Arrange ParseBlockTest(@"switch(foo) { case 0:Foo
break; case 1:Bar
return; case 2: {Baz
Boz
} default:Biz
}", new StatementBlock( Factory.Code("switch(foo) {\r\n case 0:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Foo
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" break;\r\n case 1:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Bar
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" return;\r\n case 2:\r\n {\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Baz
\r\n").Accepts(AcceptedCharacters.None) ), new MarkupBlock( Factory.Markup("Boz
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" }\r\n default:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Biz
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code("}").AsStatement().Accepts(AcceptedCharacters.None))); } [Fact] public void ParseBlockSupportsMarkupInCaseAndDefaultBranchesOfSwitchInCodeBlock() { // Arrange ParseBlockTest(@"{ switch(foo) { case 0:Foo
break; case 1:Bar
return; case 2: {Baz
Boz
} default:Biz
} }", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code(" switch(foo) {\r\n case 0:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Foo
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" break;\r\n case 1:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Bar
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" return;\r\n case 2:\r\n {\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Baz
\r\n").Accepts(AcceptedCharacters.None) ), new MarkupBlock( Factory.Markup("Boz
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" }\r\n default:\r\n").AsStatement(), new MarkupBlock( Factory.Markup("Biz
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code("} ").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None))); } [Fact] public void ParseBlockParsesMarkupStatementOnOpenAngleBracket() { ParseBlockTest("for(int i = 0; i < 10; i++) {Foo
}", new StatementBlock( Factory.Code("for(int i = 0; i < 10; i++) {").AsStatement(), new MarkupBlock( Factory.Markup("Foo
").Accepts(AcceptedCharacters.None) ), Factory.Code("}").AsStatement().Accepts(AcceptedCharacters.None) )); } [Fact] public void ParseBlockParsesMarkupStatementOnOpenAngleBracketInCodeBlock() { ParseBlockTest("{ for(int i = 0; i < 10; i++) {Foo
} }", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code(" for(int i = 0; i < 10; i++) {").AsStatement(), new MarkupBlock( Factory.Markup("Foo
").Accepts(AcceptedCharacters.None) ), Factory.Code("} ").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None))); } [Fact] public void ParseBlockParsesMarkupStatementOnSwitchCharacterFollowedByColon() { // Arrange ParseBlockTest(@"if(foo) { @:Bar } zoop", new StatementBlock( Factory.Code("if(foo) {").AsStatement(), new MarkupBlock( Factory.Markup(" "), Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Bar\r\n").With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ), Factory.Code("}").AsStatement())); } [Fact] public void ParseBlockParsesMarkupStatementOnSwitchCharacterFollowedByColonInCodeBlock() { // Arrange ParseBlockTest(@"{ if(foo) { @:Bar } } zoop", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code(" if(foo) {").AsStatement(), new MarkupBlock( Factory.Markup(" "), Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Bar\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code("} ").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None))); } [Fact] public void ParseBlockCorrectlyReturnsFromMarkupBlockWithPseudoTag() { ParseBlockTest(@"if (i > 0) {A real tag!
} }", new StatementBlock( Factory.MetaCode("{").Accepts(AcceptedCharacters.None), Factory.Code("\r\n if(true) {\r\n").AsStatement(), new MarkupBlock( Factory.Markup(" "), Factory.MarkupTransition(), Factory.MetaMarkup(":", HtmlSymbolType.Colon), Factory.Markup("Single Line Markup\r\n").With(new SingleLineMarkupEditHandler(CSharpLanguageCharacteristics.Instance.TokenizeString, AcceptedCharacters.None)) ), Factory.Code(" }\r\n foreach (var p in Enumerable.Range(1, 10)) {\r\n").AsStatement(), new MarkupBlock( Factory.Markup(@" "), Factory.MarkupTransition("A real tag!
\r\n").Accepts(AcceptedCharacters.None) ), Factory.Code(" }\r\n").AsStatement(), Factory.MetaCode("}").Accepts(AcceptedCharacters.None))); } } }