最近我遇到了一个业务需求,我必须在 Xamarin 表单的条目中显示删除线文本。现在Label中有删除线的Text修饰属性,但是Entry没有这样的属性。现在我们有一个这样的条目,
但是我们需要在sddad上显示一个删除线。有一些资源说使用效果或 Unicode 的删除线字符集。但它并不能满足我们的需求。有谁知道如何实现这一点,以便我们可以通过属性绑定将文本更改为删除线?
任何帮助将不胜感激。
每当您想要 Xamarin.Forms 中未通过 Forms API 公开的自定义 UI 行为时,您都需要进入平台级别。
删除线文本可以使用以下任何一种来实现:CustomRenderer
, Effect
, Behaviour
.
下面是一个平台效果示例,它将删除整个文本:
在您的跨平台项目中创建一个新的RoutingEffect
:
using Xamarin.Forms;
namespace StrikethroughEntry.Effects
{
public class StrikethroughEntryEffect : RoutingEffect
{
public StrikethroughEntryEffect()
: base("StrikethroughEntry.Effects.StrikethroughEntryEffect")
{
}
}
}
在你的 iOS 项目中的某个地方创建一个PlatformEffect
using Foundation;
using StrikethroughEntry.Effects;
using StrikethroughEntry.iOS;
using UIKit;
using Xamarin.Forms;
using Xamarin.Forms.Platform.iOS;
[assembly: ResolutionGroupName("StrikethroughEntry.Effects")]
[assembly: ExportEffect(typeof(iOSStrikethroughEntryEffect), nameof(StrikethroughEntryEffect))]
namespace StrikethroughEntry.iOS
{
public class iOSStrikethroughEntryEffect : PlatformEffect
{
private static NSNumber _strikethroughStyle => new NSNumber((int)NSUnderlineStyle.Single);
private string _originalText;
protected override void OnAttached()
{
var textField = Control as UITextField;
if (textField is null)
return;
_originalText = textField.Text;
var attributedText = new NSMutableAttributedString(textField.Text);
attributedText.AddAttribute(UIStringAttributeKey.StrikethroughStyle, _strikethroughStyle, new NSRange(0, textField.Text.Length));
textField.AttributedText = attributedText;
}
protected override void OnDetached()
{
var textField = Control as UITextField;
if (textField is null)
return;
textField.AttributedText = null;
textField.Text = _originalText;
}
}
}
此示例展示了如何在不再需要效果后对其进行清理。属性字符串在 iOS 上的工作方式很容易将您的删除线行为扩展到条目的目标部分。也许您只想删除特定的单词等。
在您的 Android 项目中的某处创建一个PlatformEffect
:
using Android.Graphics;
using Android.Widget;
using StrikethroughEntry.Droid;
using StrikethroughEntry.Effects;
using Xamarin.Forms;
using Xamarin.Forms.Platform.Android;
[assembly: ResolutionGroupName("StrikethroughEntry.Effects")]
[assembly: ExportEffect(typeof(AndroidStrikethroughEntryEffect), nameof(StrikethroughEntryEffect))]
namespace StrikethroughEntry.Droid
{
public class AndroidStrikethroughEntryEffect : PlatformEffect
{
private PaintFlags _originalFlags;
protected override void OnAttached()
{
var editText = Control as EditText;
if (editText is null)
return;
_originalFlags = editText.PaintFlags;
editText.PaintFlags = PaintFlags.StrikeThruText;
}
protected override void OnDetached()
{
var editText = Control as EditText;
if (editText is null)
return;
editText.PaintFlags = _originalFlags;
}
}
}
如果您想删除Entry
.
此示例将通过 xaml:
xmlns:effects="clr-namespace:StrikethroughEntry.Effects"
Entry
:<Entry Text="This entry has strikethrough">
<Entry.Effects>
<effects:StrikethroughEntryEffect />
</Entry.Effects>
</Entry>
现在你有一个可爱Entry
的带删除线的文字!