Xamarin iOS 使用 AssociatedObject 为 UIAlertController 设置 AutomationId 时应用崩溃 - objc_getAssociatedObject
Xamarin iOS App getting crashed while using AssociatedObject to set AutomationId for UIAlertController - objc_getAssociatedObject
我们正在尝试使用 Xamarin.iOS 中的扩展设置 UIAlertController 的 AutomationId。但是在 GetAccessibilityIdentifier 方法下,应用程序在 var valuePtr = objc_getAssociatedObject(alertAction.Handle, DescriptiveName.Handle);
处不一致地崩溃。也不知道为什么会崩溃。
下面是代码示例:
public static class UIAlertControllerExtension
{
public static void ApplyAccessibilityIdentifiers(this UIAlertController alertController)
{
foreach (var action in alertController.Actions)
{
var lable = action.ValueForKey((NSString)"__representer");
var view = lable as UIView;
view.AccessibilityIdentifier = action.GetAccessibilityIdentifier();
}
}
}
public static class UIAlertActionExtension
{
private static readonly NSString DescriptiveName = new NSString("AccesabilityIdentifier" + nameof(UIAlertActionExtension));
#region "Extension properties"
enum AssociationPolicy
{
Assign = 0,
RetainNonAtomic = 1,
CopyNonAtomic = 3,
Retain = 01401,
Copy = 01403,
}
[DllImport("/usr/lib/libobjc.dylib")]
static extern void objc_setAssociatedObject(
IntPtr pointer, IntPtr key,
IntPtr value, AssociationPolicy policy);
[DllImport("/usr/lib/libobjc.dylib")]
static extern IntPtr objc_getAssociatedObject(
IntPtr pointer, IntPtr key);
#endregion
public static void SetAccessibilityIdentifier(this UIAlertAction alertAction, NSString accessabilityIdentifier)
{
objc_setAssociatedObject (alertAction.Handle, DescriptiveName.Handle, accessabilityIdentifier.Handle, AssociationPolicy.Retain);
}
public static NSString GetAccessibilityIdentifier(this UIAlertAction alertAction)
{
try
{
var valuePtr = objc_getAssociatedObject(alertAction.Handle, DescriptiveName.Handle); // App is crashed at this line.
if (valuePtr != IntPtr.Zero)
{
var result = ObjCRuntime.Runtime.GetNSObject(valuePtr);
if (result != null)
return (NSString)result;
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
//var value = ObjCRuntime.Runtime.GetNSObject(valuePtr) as NSString;
return new NSString("");
}
}
使用上述扩展方法创建警报控制器并设置 AutomationId,如下所示。
private void ShowAlert()
{
var alertController = UIAlertController.Create("Title", "Message", UIAlertControllerStyle.Alert);
var action = UIAlertAction.Create("Ok", UIAlertActionStyle.Default, delegate
{
// Do Somthing
});
action.SetAccessibilityIdentifier((NSString) accessibilityIdentifier);
alertController.AddAction(action);
this.PresentViewController(alertController, true, () => { alertController.ApplyAccessibilityIdentifiers(); });
}
我们从视图控制器的按钮点击调用这个方法,它有几次工作正常。但是当第 3 次或第 4 次显示警报时,应用程序会崩溃。
请帮我解决这个问题。
提前致谢。
AssociationPolicy.Retain
可能会在意外的时间发布此 属性,因此您的应用程序会崩溃。您可以将其更改为 AssociationPolicy.RetainNonAtomic
。 NonAtomic 将使您的应用 运行 更流畅、更快。
如果您只是想存储一个字符串并稍后使用它。为什么不尝试 NSUserDefaults 或静态实例?
我们正在尝试使用 Xamarin.iOS 中的扩展设置 UIAlertController 的 AutomationId。但是在 GetAccessibilityIdentifier 方法下,应用程序在 var valuePtr = objc_getAssociatedObject(alertAction.Handle, DescriptiveName.Handle);
处不一致地崩溃。也不知道为什么会崩溃。
下面是代码示例:
public static class UIAlertControllerExtension
{
public static void ApplyAccessibilityIdentifiers(this UIAlertController alertController)
{
foreach (var action in alertController.Actions)
{
var lable = action.ValueForKey((NSString)"__representer");
var view = lable as UIView;
view.AccessibilityIdentifier = action.GetAccessibilityIdentifier();
}
}
}
public static class UIAlertActionExtension
{
private static readonly NSString DescriptiveName = new NSString("AccesabilityIdentifier" + nameof(UIAlertActionExtension));
#region "Extension properties"
enum AssociationPolicy
{
Assign = 0,
RetainNonAtomic = 1,
CopyNonAtomic = 3,
Retain = 01401,
Copy = 01403,
}
[DllImport("/usr/lib/libobjc.dylib")]
static extern void objc_setAssociatedObject(
IntPtr pointer, IntPtr key,
IntPtr value, AssociationPolicy policy);
[DllImport("/usr/lib/libobjc.dylib")]
static extern IntPtr objc_getAssociatedObject(
IntPtr pointer, IntPtr key);
#endregion
public static void SetAccessibilityIdentifier(this UIAlertAction alertAction, NSString accessabilityIdentifier)
{
objc_setAssociatedObject (alertAction.Handle, DescriptiveName.Handle, accessabilityIdentifier.Handle, AssociationPolicy.Retain);
}
public static NSString GetAccessibilityIdentifier(this UIAlertAction alertAction)
{
try
{
var valuePtr = objc_getAssociatedObject(alertAction.Handle, DescriptiveName.Handle); // App is crashed at this line.
if (valuePtr != IntPtr.Zero)
{
var result = ObjCRuntime.Runtime.GetNSObject(valuePtr);
if (result != null)
return (NSString)result;
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
//var value = ObjCRuntime.Runtime.GetNSObject(valuePtr) as NSString;
return new NSString("");
}
}
使用上述扩展方法创建警报控制器并设置 AutomationId,如下所示。
private void ShowAlert()
{
var alertController = UIAlertController.Create("Title", "Message", UIAlertControllerStyle.Alert);
var action = UIAlertAction.Create("Ok", UIAlertActionStyle.Default, delegate
{
// Do Somthing
});
action.SetAccessibilityIdentifier((NSString) accessibilityIdentifier);
alertController.AddAction(action);
this.PresentViewController(alertController, true, () => { alertController.ApplyAccessibilityIdentifiers(); });
}
我们从视图控制器的按钮点击调用这个方法,它有几次工作正常。但是当第 3 次或第 4 次显示警报时,应用程序会崩溃。
请帮我解决这个问题。
提前致谢。
AssociationPolicy.Retain
可能会在意外的时间发布此 属性,因此您的应用程序会崩溃。您可以将其更改为 AssociationPolicy.RetainNonAtomic
。 NonAtomic 将使您的应用 运行 更流畅、更快。
如果您只是想存储一个字符串并稍后使用它。为什么不尝试 NSUserDefaults 或静态实例?