You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
224 lines
7.0 KiB
224 lines
7.0 KiB
// Copyright (c) 2011 AlphaSierraPapa for the SharpDevelop Team
|
|
//
|
|
// 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.
|
|
|
|
#nullable enable
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Media;
|
|
|
|
using ICSharpCode.Decompiler.Metadata;
|
|
using ICSharpCode.Decompiler.TypeSystem;
|
|
using ICSharpCode.ILSpyX;
|
|
|
|
using TomsToolbox.Essentials;
|
|
|
|
namespace ICSharpCode.ILSpy
|
|
{
|
|
/// <summary>
|
|
/// ExtensionMethods used in ILSpy.
|
|
/// </summary>
|
|
public static class ExtensionMethods
|
|
{
|
|
public static string ToSuffixString(this System.Reflection.Metadata.EntityHandle handle, bool showMetadataTokens, bool useBase10)
|
|
{
|
|
if (!showMetadataTokens)
|
|
return string.Empty;
|
|
|
|
int token = System.Reflection.Metadata.Ecma335.MetadataTokens.GetToken(handle);
|
|
if (useBase10)
|
|
return " @" + token;
|
|
return " @" + token.ToString("x8");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Takes at most <paramref name="length" /> first characters from string, and appends '...' if string is longer.
|
|
/// String can be null.
|
|
/// </summary>
|
|
public static string TakeStartEllipsis(this string s, int length)
|
|
{
|
|
if (string.IsNullOrEmpty(s) || length >= s.Length)
|
|
return s;
|
|
return s.Substring(0, length) + "...";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Equivalent to <code>collection.Select(func).ToArray()</code>, but more efficient as it makes
|
|
/// use of the input collection's known size.
|
|
/// </summary>
|
|
public static U[] SelectArray<T, U>(this ICollection<T> collection, Func<T, U> func)
|
|
{
|
|
U[] result = new U[collection.Count];
|
|
int index = 0;
|
|
foreach (var element in collection)
|
|
{
|
|
result[index++] = func(element);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
public static ICompilation? GetTypeSystemWithCurrentOptionsOrNull(this MetadataFile file, SettingsService settingsService, LanguageVersion languageVersion)
|
|
{
|
|
var decompilerSettings = settingsService.DecompilerSettings.Clone();
|
|
if (!Enum.TryParse(languageVersion?.Version, out Decompiler.CSharp.LanguageVersion csharpLanguageVersion))
|
|
csharpLanguageVersion = Decompiler.CSharp.LanguageVersion.Latest;
|
|
decompilerSettings.SetLanguageVersion(csharpLanguageVersion);
|
|
return file
|
|
.GetLoadedAssembly()
|
|
.GetTypeSystemOrNull(DecompilerTypeSystem.GetOptions(decompilerSettings));
|
|
}
|
|
|
|
#region DPI independence
|
|
public static Rect TransformToDevice(this Rect rect, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformToDevice;
|
|
return Rect.Transform(rect, matrix);
|
|
}
|
|
|
|
public static Rect TransformFromDevice(this Rect rect, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformFromDevice;
|
|
return Rect.Transform(rect, matrix);
|
|
}
|
|
|
|
public static Size TransformToDevice(this Size size, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformToDevice;
|
|
return new Size(size.Width * matrix.M11, size.Height * matrix.M22);
|
|
}
|
|
|
|
public static Size TransformFromDevice(this Size size, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformFromDevice;
|
|
return new Size(size.Width * matrix.M11, size.Height * matrix.M22);
|
|
}
|
|
|
|
public static Point TransformToDevice(this Point point, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformToDevice;
|
|
return matrix.Transform(point);
|
|
}
|
|
|
|
public static Point TransformFromDevice(this Point point, Visual visual)
|
|
{
|
|
Matrix matrix = PresentationSource.FromVisual(visual).CompositionTarget.TransformFromDevice;
|
|
return matrix.Transform(point);
|
|
}
|
|
#endregion
|
|
|
|
public static T? FindVisualChild<T>(this DependencyObject? depObj) where T : DependencyObject
|
|
{
|
|
if (depObj != null)
|
|
{
|
|
for (int i = 0; i < VisualTreeHelper.GetChildrenCount(depObj); i++)
|
|
{
|
|
DependencyObject child = VisualTreeHelper.GetChild(depObj, i);
|
|
if (child is T dependencyObject)
|
|
{
|
|
return dependencyObject;
|
|
}
|
|
|
|
T? childItem = FindVisualChild<T>(child);
|
|
if (childItem != null)
|
|
return childItem;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public static T? GetParent<T>(this DependencyObject? depObj) where T : DependencyObject
|
|
{
|
|
if (depObj == null)
|
|
return null;
|
|
while (!(depObj is T))
|
|
{
|
|
var parent = VisualTreeHelper.GetParent(depObj);
|
|
if (parent == null)
|
|
return null;
|
|
depObj = parent;
|
|
}
|
|
return (T)depObj;
|
|
}
|
|
|
|
public static void SelectItem(this DataGrid view, object item)
|
|
{
|
|
var container = (DataGridRow)view.ItemContainerGenerator.ContainerFromItem(item);
|
|
if (container != null)
|
|
container.IsSelected = true;
|
|
view.Focus();
|
|
}
|
|
|
|
public static double ToGray(this Color? color)
|
|
{
|
|
return color?.R * 0.3 + color?.G * 0.6 + color?.B * 0.1 ?? 0.0;
|
|
}
|
|
|
|
internal static string? FormatExceptions(this IList<App.ExceptionData> exceptions)
|
|
{
|
|
if (exceptions.Count == 0)
|
|
return null;
|
|
|
|
string delimiter = $"-------------------------------------------------{Environment.NewLine}";
|
|
|
|
return string.Join(delimiter, exceptions.Select(FormatException));
|
|
}
|
|
|
|
private static string FormatException(App.ExceptionData item)
|
|
{
|
|
var output = new StringBuilder();
|
|
|
|
if (!item.PluginName.IsNullOrEmpty())
|
|
output.AppendLine("Error(s) loading plugin: " + item.PluginName);
|
|
|
|
if (item.Exception is System.Reflection.ReflectionTypeLoadException exception)
|
|
{
|
|
foreach (var ex in exception.LoaderExceptions.ExceptNullItems())
|
|
{
|
|
output.AppendLine(ex.ToString());
|
|
output.AppendLine();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
output.AppendLine(item.Exception.ToString());
|
|
}
|
|
|
|
return output.ToString();
|
|
}
|
|
|
|
public static IDisposable PreserveFocus(this IInputElement? inputElement, bool preserve = true)
|
|
{
|
|
return new RestoreFocusHelper(inputElement, preserve);
|
|
}
|
|
|
|
private sealed class RestoreFocusHelper(IInputElement? inputElement, bool preserve) : IDisposable
|
|
{
|
|
public void Dispose()
|
|
{
|
|
if (preserve)
|
|
{
|
|
inputElement?.Focus();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|