DocSpace-buildtools/common/ASC.Textile/States/ListFormatterState.cs

154 lines
4.3 KiB
C#
Raw Normal View History

2019-08-19 12:35:39 +00:00
#region License Statement
// Copyright (c) L.A.B.Soft. All rights reserved.
//
// The use and distribution terms for this software are covered by the
// Common Public License 1.0 (http://opensource.org/licenses/cpl.php)
// which can be found in the file CPL.TXT at the root of this distribution.
// By using this software in any fashion, you are agreeing to be bound by
// the terms of this license.
//
// You must not remove this notice, or any other, from this software.
#endregion
2022-03-17 15:13:38 +00:00
namespace Textile.States;
/// <summary>
/// Base formatting state for all lists.
/// </summary>
abstract public class ListFormatterState : FormatterState
2019-08-19 12:35:39 +00:00
{
2022-03-17 16:57:02 +00:00
internal const string _patternBegin = @"^\s*(?<tag>";
internal const string _patternEnd = @")" + Globals.BlockModifiersPattern + @"(?:\s+)? (?<content>.*)$";
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
private bool _firstItem = true;
private bool _firstItemLine = true;
private string _tag;
private string _attsInfo;
private string _alignInfo;
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
protected int NestingDepth
{
get { return _tag.Length; }
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
protected ListFormatterState(TextileFormatter formatter)
: base(formatter)
{
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
public override string Consume(string input, Match m)
{
_tag = m.Groups["tag"].Value;
_alignInfo = m.Groups["align"].Value;
_attsInfo = m.Groups["atts"].Value;
input = m.Groups["content"].Value;
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
this.Formatter.ChangeState(this);
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
return input;
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
public sealed override void Enter()
{
_firstItem = true;
_firstItemLine = true;
WriteIndent();
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
public sealed override void Exit()
{
Formatter.Output.WriteLine("</li>");
WriteOutdent();
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
public sealed override void FormatLine(string input)
{
if (_firstItemLine)
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
if (!_firstItem)
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
Formatter.Output.WriteLine("</li>");
2019-08-19 12:35:39 +00:00
}
2022-03-17 15:13:38 +00:00
Formatter.Output.Write("<li " + FormattedStylesAndAlignment("li") + ">");
_firstItemLine = false;
}
else
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
Formatter.Output.WriteLine("<br />");
2019-08-19 12:35:39 +00:00
}
2022-03-17 15:13:38 +00:00
Formatter.Output.Write(input);
_firstItem = false;
}
public sealed override bool ShouldNestState(FormatterState other)
{
var listState = (ListFormatterState)other;
return listState.NestingDepth > NestingDepth;
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
public sealed override bool ShouldExit(string input)
{
// If we have an empty line, we can exit.
if (string.IsNullOrEmpty(input))
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
return true;
}
2019-08-19 12:35:39 +00:00
2022-03-17 15:13:38 +00:00
// We exit this list if the next
// list item is of the same type but less
// deep as us, or of the other type of
// list and as deep or less.
if (NestingDepth > 1)
{
if (IsMatchForMe(input, 1, NestingDepth - 1))
2022-03-17 15:01:39 +00:00
{
2019-08-19 12:35:39 +00:00
return true;
2022-03-17 15:01:39 +00:00
}
2022-03-17 15:13:38 +00:00
}
if (IsMatchForOthers(input, 1, NestingDepth))
{
return true;
2019-08-19 12:35:39 +00:00
}
2022-03-17 15:13:38 +00:00
// As it seems we're going to continue taking
// care of this line, we take the opportunity
// to check whether it's the same list item as
// previously (no "**" or "##" tags), or if it's
// a new list item.
if (IsMatchForMe(input, NestingDepth, NestingDepth))
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
_firstItemLine = true;
}
2022-03-17 15:01:39 +00:00
2022-03-17 15:13:38 +00:00
return false;
}
2022-03-17 15:01:39 +00:00
2022-03-17 15:13:38 +00:00
public sealed override bool ShouldParseForNewFormatterState(string input)
{
// We don't let anyone but ourselves mess with our stuff.
if (IsMatchForMe(input, 1, 100))
{
return true;
2019-08-19 12:35:39 +00:00
}
2022-03-17 15:13:38 +00:00
if (IsMatchForOthers(input, 1, 100))
2019-08-19 12:35:39 +00:00
{
2022-03-17 15:13:38 +00:00
return true;
2019-08-19 12:35:39 +00:00
}
2022-03-17 15:13:38 +00:00
return false;
}
protected abstract void WriteIndent();
protected abstract void WriteOutdent();
protected abstract bool IsMatchForMe(string input, int minNestingDepth, int maxNestingDepth);
protected abstract bool IsMatchForOthers(string input, int minNestingDepth, int maxNestingDepth);
protected string FormattedStylesAndAlignment(string element)
{
return Blocks.BlockAttributesParser.ParseBlockAttributes(_alignInfo + _attsInfo, element);
2019-08-19 12:35:39 +00:00
}
}