PowerToys/src/modules/previewpane/SvgPreviewHandler/Utilities/SvgPreviewHandlerHelper.cs
Stefan Markovic 88517bfdf7
[FileExplorerPreview] Move everything from WebBrowser to WebView2 (#17588)
* Move MarkdownPreviewHandler from WebBrowser to WebView2

* Disable context menu
Open links in default browser

* Update expect.txt

* Move SvgPreviewHandler from WebBrowser to WebView2

* Migrate SvgThumbnailProvider from WebBrowser to WebView2

* Migrate CustomControlTest to WebView2
Remove WebBrowser related stuff

* Update tests

* Revert GetThumbnail return value
Disable javascript dialogs in WebView2 for Svg thumbnail and preview

* expect.txt

* Increase timeout for Markdown tests

* Add sleeps

* Add zero check
2022-04-14 16:27:22 +01:00

168 lines
6.6 KiB
C#

// Copyright (c) Microsoft Corporation
// The Microsoft Corporation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Xml.Linq;
namespace Microsoft.PowerToys.PreviewHandler.Svg.Utilities
{
/// <summary>
/// Helper utilities for Svg Preview Handler.
/// </summary>
public static class SvgPreviewHandlerHelper
{
/// <summary>
/// Dictionary of elements in lower case that are blocked from Svg for preview pane.
/// Reference for list of Svg Elements: https://developer.mozilla.org/en-US/docs/Web/SVG/Element.
/// </summary>
private static Dictionary<string, bool> blockedElementsName = new Dictionary<string, bool>
{
{ "script", true },
{ "image", true },
{ "feimage", true },
};
/// <summary>
/// Check if any of the blocked elements present in Svg.
/// </summary>
/// <param name="svgData">Input Svg.</param>
/// <returns>Returns true in case any of the blocked element is present otherwise false.</returns>
public static bool CheckBlockedElements(string svgData)
{
bool foundBlockedElement = false;
if (string.IsNullOrWhiteSpace(svgData))
{
return foundBlockedElement;
}
// Check if any of the blocked element is present. If failed to parse or iterate over Svg return default false.
// No need to throw because all the external content and script are blocked on the Web Browser Control itself.
try
{
var doc = XDocument.Parse(svgData);
var elements = doc.Descendants().ToList();
foreach (XElement element in elements)
{
// Using Invariant since we are doing an exact match for HTML tags and we want it to behave the same in every culture
#pragma warning disable CA1308 // Normalize strings to uppercase
var elementName = element?.Name?.LocalName?.ToLowerInvariant();
#pragma warning restore CA1308 // Normalize strings to uppercase
if (elementName != null && blockedElementsName.ContainsKey(elementName))
{
foundBlockedElement = true;
// No need to iterate further since we are displaying info bar with condition of atleast one occurrence of blocked element is present.
break;
}
}
}
#pragma warning disable CA1031 // Do not catch general exception types
catch (Exception)
#pragma warning restore CA1031 // Do not catch general exception types
{
}
return foundBlockedElement;
}
/// <summary>
/// Add proper
/// </summary>
/// <param name="stringSvgData">Input Svg</param>
/// <returns>Returns modified svgData with added style</returns>
public static string AddStyleSVG(string stringSvgData)
{
XElement svgData = XElement.Parse(stringSvgData);
var attributes = svgData.Attributes();
string width = string.Empty;
string height = string.Empty;
string widthR = string.Empty;
string heightR = string.Empty;
string oldStyle = string.Empty;
// Get width and height of element and remove it afterwards because it will be added inside style attribute
for (int i = 0; i < attributes.Count(); i++)
{
if (attributes.ElementAt(i).Name == "height")
{
height = attributes.ElementAt(i).Value;
attributes.ElementAt(i).Remove();
i--;
}
else if (attributes.ElementAt(i).Name == "width")
{
width = attributes.ElementAt(i).Value;
attributes.ElementAt(i).Remove();
i--;
}
else if (attributes.ElementAt(i).Name == "style")
{
oldStyle = attributes.ElementAt(i).Value;
attributes.ElementAt(i).Remove();
i--;
}
}
svgData.ReplaceAttributes(attributes);
height = CheckUnit(height);
width = CheckUnit(width);
heightR = RemoveUnit(height);
widthR = RemoveUnit(width);
string centering = "position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%);";
// max-width and max-height not supported. Extra CSS is needed for it to work.
string scaling = $"max-width: {width} ; max-height: {height} ;";
scaling += $" _height:expression(this.scrollHeight > {heightR} ? \" {height}\" : \"auto\"); _width:expression(this.scrollWidth > {widthR} ? \"{width}\" : \"auto\");";
svgData.Add(new XAttribute("style", scaling + centering + oldStyle));
return svgData.ToString();
}
/// <summary>
/// If there is a CSS unit at the end return the same string, else return the string with a px unit at the end
/// </summary>
/// <param name="length">CSS length</param>
/// <returns>Returns modified length</returns>
private static string CheckUnit(string length)
{
string[] cssUnits = { "cm", "mm", "in", "px", "pt", "pc", "em", "ex", "ch", "rem", "vw", "vh", "vmin", "vmax", "%" };
foreach (var unit in cssUnits)
{
if (length.EndsWith(unit, System.StringComparison.CurrentCultureIgnoreCase))
{
return length;
}
}
return length + "px";
}
/// <summary>
/// Remove a CSS unit from the end of the string
/// </summary>
/// <param name="length">CSS length</param>
/// <returns>Returns modified length</returns>
private static string RemoveUnit(string length)
{
string[] cssUnits = { "cm", "mm", "in", "px", "pt", "pc", "em", "ex", "ch", "rem", "vw", "vh", "vmin", "vmax", "%" };
foreach (var unit in cssUnits)
{
if (length.EndsWith(unit, System.StringComparison.CurrentCultureIgnoreCase))
{
length = length.Remove(length.Length - unit.Length);
return length;
}
}
return length;
}
}
}