Fix anchor parsing issue in FormattedText

When parsing an anchor element, allow relative uris.
Added unit tests for FormattedText and LinkLabel.
This commit is contained in:
Lluis Sanchez 2013-11-14 10:20:43 +01:00
Родитель ef54d065c7
Коммит 0b819f24b1
4 изменённых файлов: 312 добавлений и 4 удалений

Просмотреть файл

@ -107,6 +107,8 @@
<Compile Include="Tests\WindowTests.cs" /> <Compile Include="Tests\WindowTests.cs" />
<Compile Include="Tests\XwtTest.cs" /> <Compile Include="Tests\XwtTest.cs" />
<Compile Include="Tests\InternalChildrenTests.cs" /> <Compile Include="Tests\InternalChildrenTests.cs" />
<Compile Include="Tests\LinkLabelTests.cs" />
<Compile Include="Tests\FormattedTextTests.cs" />
</ItemGroup> </ItemGroup>
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
<ItemGroup> <ItemGroup>

Просмотреть файл

@ -0,0 +1,246 @@
//
// FormattedTextTests.cs
//
// Author:
// Lluis Sanchez <lluis@xamarin.com>
//
// Copyright (c) 2013 Xamarin Inc
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System;
using NUnit.Framework;
using Xwt.Drawing;
namespace Xwt
{
public class FormattedTextTests
{
[Test]
public void ParseEmptySpan ()
{
var s = "0<span>12</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (0, ft.Attributes.Count);
}
[Test]
public void ParseIncorrectlyNestedSpan ()
{
var s = "0<b><u>12</b></u>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (1, ft.Attributes.Count);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [0]);
}
[Test]
public void UnclosedElements ()
{
var s = "0<b><u>12</u>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (1, ft.Attributes.Count);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [0]);
}
[Test]
public void ParseBrokenMarkup ()
{
var s = "0<b<u>12</u>>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (1, ft.Attributes.Count);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [0]);
}
[Test]
public void ParseFontWeight ()
{
var s = "0<b>12</b><span weight='ultrabold'>34</span><span font-weight='Light'>56</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (3, ft.Attributes.Count);
Assert.IsAssignableFrom<FontWeightTextAttribute> (ft.Attributes [0]);
var at = (FontWeightTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (FontWeight.Bold, at.Weight);
Assert.IsAssignableFrom<FontWeightTextAttribute> (ft.Attributes [1]);
at = (FontWeightTextAttribute)ft.Attributes [1];
Assert.AreEqual (3, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (FontWeight.Ultrabold, at.Weight);
Assert.IsAssignableFrom<FontWeightTextAttribute> (ft.Attributes [2]);
at = (FontWeightTextAttribute)ft.Attributes [2];
Assert.AreEqual (5, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (FontWeight.Light, at.Weight);
}
[Test]
public void ParseItalic ()
{
var s = "0<i>12</i>3<span font-style='oBlique'>45</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (2, ft.Attributes.Count);
Assert.IsAssignableFrom<FontStyleTextAttribute> (ft.Attributes [0]);
var at = (FontStyleTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (FontStyle.Italic, at.Style);
Assert.IsAssignableFrom<FontStyleTextAttribute> (ft.Attributes [1]);
at = (FontStyleTextAttribute)ft.Attributes [1];
Assert.AreEqual (4, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (FontStyle.Oblique, at.Style);
}
[Test]
public void ParseStrikethrough ()
{
var s = "0<s>12</s>3<span strikethrough='true'>45</span>6<span strikethrough='f'>7</span><span strikethrough=''>8</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (4, ft.Attributes.Count);
Assert.IsAssignableFrom<StrikethroughTextAttribute> (ft.Attributes [0]);
var at = (StrikethroughTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.IsTrue (at.Strikethrough);
Assert.IsAssignableFrom<StrikethroughTextAttribute> (ft.Attributes [1]);
at = (StrikethroughTextAttribute)ft.Attributes [1];
Assert.AreEqual (4, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.IsTrue (at.Strikethrough);
Assert.IsAssignableFrom<StrikethroughTextAttribute> (ft.Attributes [2]);
at = (StrikethroughTextAttribute)ft.Attributes [2];
Assert.AreEqual (7, at.StartIndex);
Assert.AreEqual (1, at.Count);
Assert.IsFalse (at.Strikethrough);
Assert.IsAssignableFrom<StrikethroughTextAttribute> (ft.Attributes [3]);
at = (StrikethroughTextAttribute)ft.Attributes [3];
Assert.AreEqual (8, at.StartIndex);
Assert.AreEqual (1, at.Count);
Assert.IsFalse (at.Strikethrough);
}
[Test]
public void ParseUnderline ()
{
var s = "0<u>12</u>3<span underline='true'>45</span>6<span underline='f'>7</span><span underline=''>8</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (4, ft.Attributes.Count);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [0]);
var at = (UnderlineTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.IsTrue (at.Underline);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [1]);
at = (UnderlineTextAttribute)ft.Attributes [1];
Assert.AreEqual (4, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.IsTrue (at.Underline);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [2]);
at = (UnderlineTextAttribute)ft.Attributes [2];
Assert.AreEqual (7, at.StartIndex);
Assert.AreEqual (1, at.Count);
Assert.IsFalse (at.Underline);
Assert.IsAssignableFrom<UnderlineTextAttribute> (ft.Attributes [3]);
at = (UnderlineTextAttribute)ft.Attributes [3];
Assert.AreEqual (8, at.StartIndex);
Assert.AreEqual (1, at.Count);
Assert.IsFalse (at.Underline);
}
[Test]
public void ParseAnchor ()
{
var s = "<a href='http://foo.bar'>01</a><a href='#anchor'>23</a><a href='relative'>45</a><a href='custom:uri'>67</a><a href=''>89</a>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (5, ft.Attributes.Count);
Assert.IsAssignableFrom<LinkTextAttribute> (ft.Attributes [0]);
var at = (LinkTextAttribute)ft.Attributes [0];
Assert.AreEqual (0, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (new Uri ("http://foo.bar", UriKind.RelativeOrAbsolute), at.Target);
Assert.IsAssignableFrom<LinkTextAttribute> (ft.Attributes [1]);
at = (LinkTextAttribute)ft.Attributes [1];
Assert.AreEqual (2, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (new Uri ("#anchor", UriKind.RelativeOrAbsolute), at.Target);
Assert.IsAssignableFrom<LinkTextAttribute> (ft.Attributes [2]);
at = (LinkTextAttribute)ft.Attributes [2];
Assert.AreEqual (4, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (new Uri ("relative", UriKind.RelativeOrAbsolute), at.Target);
Assert.IsAssignableFrom<LinkTextAttribute> (ft.Attributes [3]);
at = (LinkTextAttribute)ft.Attributes [3];
Assert.AreEqual (6, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (new Uri ("custom:uri", UriKind.RelativeOrAbsolute), at.Target);
Assert.IsAssignableFrom<LinkTextAttribute> (ft.Attributes [4]);
at = (LinkTextAttribute)ft.Attributes [4];
Assert.AreEqual (8, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (new Uri ("", UriKind.RelativeOrAbsolute), at.Target);
}
[Test]
public void ParseColor ()
{
var s = "0<span color='#ff0000'>12</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (1, ft.Attributes.Count);
Assert.IsAssignableFrom<ColorTextAttribute> (ft.Attributes [0]);
var at = (ColorTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (Colors.Red, at.Color);
}
[Test]
public void ParseBackgroundColor ()
{
var s = "0<span background-color='#0000ff'>12</span>";
var ft = FormattedText.FromMarkup (s);
Assert.AreEqual (1, ft.Attributes.Count);
Assert.IsAssignableFrom<BackgroundTextAttribute> (ft.Attributes [0]);
var at = (BackgroundTextAttribute)ft.Attributes [0];
Assert.AreEqual (1, at.StartIndex);
Assert.AreEqual (2, at.Count);
Assert.AreEqual (Colors.Blue, at.Color);
}
}
}

Просмотреть файл

@ -0,0 +1,39 @@
//
// LinkLabelTests.cs
//
// Author:
// Lluis Sanchez <lluis@xamarin.com>
//
// Copyright (c) 2013 Xamarin Inc
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System;
using NUnit.Framework;
namespace Xwt
{
public class LinkLabelTests: WidgetTests
{
public override Widget CreateWidget ()
{
return new LinkLabel ("Hello World");
}
}
}

Просмотреть файл

@ -109,7 +109,7 @@ namespace Xwt
Uri href = null; Uri href = null;
ReadXmlAttributes (markup, ref k, (name, val) => { ReadXmlAttributes (markup, ref k, (name, val) => {
if (name == "href") { if (name == "href") {
href = new Uri (val); href = new Uri (val, UriKind.RelativeOrAbsolute);
return true; return true;
} }
return false; return false;
@ -218,6 +218,7 @@ namespace Xwt
{ {
switch (name) { switch (name) {
case "font": case "font":
case "font-desc":
case "font_desc": case "font_desc":
return new FontTextAttribute () { Font = Font.FromName (val) }; return new FontTextAttribute () { Font = Font.FromName (val) };
@ -229,12 +230,20 @@ namespace Xwt
return new FontSizeTextAttribute () { Size = s }; return new FontSizeTextAttribute () { Size = s };
*/ */
case "font_weight": case "font_weight":
case "font-weight":
case "weight": case "weight":
FontWeight w; FontWeight w;
if (!Enum.TryParse<FontWeight> (val, true, out w)) if (!Enum.TryParse<FontWeight> (val, true, out w))
return null; return null;
return new FontWeightTextAttribute () { Weight = w }; return new FontWeightTextAttribute () { Weight = w };
case "font_style":
case "font-style":
FontStyle s;
if (!Enum.TryParse<FontStyle> (val, true, out s))
return null;
return new FontStyleTextAttribute () { Style = s };
case "foreground": case "foreground":
case "fgcolor": case "fgcolor":
case "color": case "color":
@ -244,6 +253,7 @@ namespace Xwt
return new ColorTextAttribute () { Color = c }; return new ColorTextAttribute () { Color = c };
case "background": case "background":
case "background-color":
case "bgcolor": case "bgcolor":
Color bc; Color bc;
if (!Color.TryParse (val, out bc)) if (!Color.TryParse (val, out bc))
@ -251,14 +261,25 @@ namespace Xwt
return new BackgroundTextAttribute () { Color = bc }; return new BackgroundTextAttribute () { Color = bc };
case "underline": case "underline":
return new UnderlineTextAttribute (); return new UnderlineTextAttribute () {
Underline = ParseBool (val, false)
};
case "strikethrough": case "strikethrough":
return new StrikethroughTextAttribute (); return new StrikethroughTextAttribute () {
Strikethrough = ParseBool (val, false)
};
} }
return null; return null;
} }
bool ParseBool (string s, bool defaultValue)
{
if (s.Length == 0)
return defaultValue;
return string.Equals (s, "true", StringComparison.OrdinalIgnoreCase);
}
bool ReadId (string markup, ref int i, out string tag) bool ReadId (string markup, ref int i, out string tag)
{ {
tag = null; tag = null;