| /****************************************************************************  | 
| **  | 
| ** Copyright (C) 2019 The Qt Company Ltd.  | 
| ** Contact: https://www.qt.io/licensing/  | 
| **  | 
| ** This file is part of the Qt VS Tools.  | 
| **  | 
| ** $QT_BEGIN_LICENSE:GPL-EXCEPT$  | 
| ** Commercial License Usage  | 
| ** Licensees holding valid commercial Qt licenses may use this file in  | 
| ** accordance with the commercial license agreement provided with the  | 
| ** Software or, alternatively, in accordance with the terms contained in  | 
| ** a written agreement between you and The Qt Company. For licensing terms  | 
| ** and conditions see https://www.qt.io/terms-conditions. For further  | 
| ** information use the contact form at https://www.qt.io/contact-us.  | 
| **  | 
| ** GNU General Public License Usage  | 
| ** Alternatively, this file may be used under the terms of the GNU  | 
| ** General Public License version 3 as published by the Free Software  | 
| ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT  | 
| ** included in the packaging of this file. Please review the following  | 
| ** information to ensure the GNU General Public License requirements will  | 
| ** be met: https://www.gnu.org/licenses/gpl-3.0.html.  | 
| **  | 
| ** $QT_END_LICENSE$  | 
| **  | 
| ****************************************************************************/  | 
|   | 
| using System;  | 
| using System.Collections.Generic;  | 
| using System.Collections.Concurrent;  | 
| using System.Text;  | 
|   | 
| namespace QtVsTools.SyntaxAnalysis  | 
| {  | 
|     using static RegExprAssert;  | 
|   | 
|     ////////////////////////////////////////////////////////////////////////////////////////////////  | 
|     ///  | 
|     /// RegExprAssert ( -> RegExpr)  | 
|     ///  | 
|     ////////////////////////////////////////////////////////////////////////////////////////////////  | 
|     /// <summary>  | 
|     /// Asserts a pattern on the input string without consuming chars.  | 
|     /// </summary>  | 
|     ///  | 
|     public partial class RegExprAssert : RegExpr  | 
|     {  | 
|         public enum AssertLook { Ahead, Behind }  | 
|   | 
|         public AssertLook Context { get; set; }  | 
|         public bool Negative { get; set; }  | 
|         public RegExpr Expr { get; set; }  | 
|   | 
|         protected override IEnumerable<RegExpr> OnRender(RegExpr defaultTokenWs, RegExpr parent,  | 
|             StringBuilder pattern, ref RenderMode mode, Stack<Token> tokenStack)  | 
|         {  | 
|             base.OnRender(defaultTokenWs, parent, pattern, ref mode, tokenStack);  | 
|   | 
|             if (mode.HasFlag(RenderMode.Assert))  | 
|                 throw new NestedAssertException();  | 
|   | 
|             switch (Context) {  | 
|             case AssertLook.Ahead:  | 
|                 if (Negative)  | 
|                     pattern.Append("(?!");  | 
|                 else  | 
|                     pattern.Append("(?=");  | 
|                 break;  | 
|             case AssertLook.Behind:  | 
|                 if (Negative)  | 
|                     pattern.Append("(?<!");  | 
|                 else  | 
|                     pattern.Append("(?<=");  | 
|                 break;  | 
|             }  | 
|   | 
|             mode |= RenderMode.Assert;  | 
|             return Items(Expr);  | 
|         }  | 
|   | 
|         protected override void OnRenderEnd(RegExpr defaultTokenWs, RegExpr parent,  | 
|             StringBuilder pattern, ref RenderMode mode, Stack<Token> tokenStack)  | 
|         {  | 
|             base.OnRenderEnd(defaultTokenWs, parent, pattern, ref mode, tokenStack);  | 
|             pattern.Append(")");  | 
|             mode &= ~RenderMode.Assert;  | 
|         }  | 
|     }  | 
|   | 
|     public abstract partial class RegExpr  | 
|     {  | 
|         RegExprAssert AsAssert()  | 
|         {  | 
|             if (this is RegExprAssert)  | 
|                 return this as RegExprAssert;  | 
|   | 
|             return new RegExprAssert  | 
|             {  | 
|                 Context = AssertLook.Ahead,  | 
|                 Negative = false,  | 
|                 Expr = this  | 
|             };  | 
|         }  | 
|   | 
|         public static RegExprAssert AssertLookAhead(RegExpr expr)  | 
|         {  | 
|             var assert = expr.AsAssert();  | 
|             return new RegExprAssert  | 
|             {  | 
|                 Context = AssertLook.Ahead,  | 
|                 Negative = assert.Negative,  | 
|                 Expr = assert.Expr  | 
|             };  | 
|         }  | 
|   | 
|         public static RegExprAssert AssertLookBehind(RegExpr expr)  | 
|         {  | 
|             var assert = expr.AsAssert();  | 
|             return new RegExprAssert  | 
|             {  | 
|                 Context = AssertLook.Behind,  | 
|                 Negative = assert.Negative,  | 
|                 Expr = assert.Expr  | 
|             };  | 
|         }  | 
|   | 
|         public static RegExprAssert AssertNegated(RegExpr expr)  | 
|         {  | 
|             var assert = expr.AsAssert();  | 
|             return new RegExprAssert  | 
|             {  | 
|                 Context = assert.Context,  | 
|                 Negative = !assert.Negative,  | 
|                 Expr = assert.Expr  | 
|             };  | 
|         }  | 
|   | 
|         public delegate RegExprAssert AssertTemplate(RegExpr expr);  | 
|   | 
|         public class AssertExprBuilder  | 
|         {  | 
|             AssertTemplate Template { get; set; }  | 
|   | 
|             public AssertExprBuilder(AssertTemplate template)  | 
|             {  | 
|                 Template = template;  | 
|             }  | 
|   | 
|             public class Expr  | 
|             {  | 
|                 public RegExprAssert Assert { get; set; }  | 
|                 public Expr(RegExprAssert assert) { Assert = assert; }  | 
|   | 
|                 public static implicit operator RegExpr(Expr e)  | 
|                 {  | 
|                     return e.Assert;  | 
|                 }  | 
|   | 
|                 public static RegExpr operator &(RegExpr rx1, Expr rx2)  | 
|                 {  | 
|                     return Concat(rx1, rx2);  | 
|                 }  | 
|   | 
|                 public static RegExpr operator |(RegExpr rx1, Expr rx2)  | 
|                 {  | 
|                     return Choice(rx1, rx2);  | 
|                 }  | 
|             }  | 
|   | 
|             public class NegateableExpr : Expr  | 
|             {  | 
|                 public NegateableExpr(RegExprAssert assert) : base(assert) { }  | 
|   | 
|                 public static Expr operator !(NegateableExpr x)  | 
|                 {  | 
|                     return new Expr(AssertNegated(x.Assert));  | 
|                 }  | 
|             }  | 
|   | 
|             public NegateableExpr this[RegExpr expr]  | 
|             {  | 
|                 get { return new NegateableExpr(Template(expr)); }  | 
|             }  | 
|         }  | 
|   | 
|         public class NestedAssertException : RegExprException  | 
|         {  | 
|             public NestedAssertException(string message = null) : base(message) { }  | 
|         }  | 
|     }  | 
| }  |