使用AssociatedObject为UIAlertController设置AutomationId时,Xamarin iOS应用程序崩溃-objc_getAssociatedObject

Ram*_*rai 5 xamarin.ios xamarin.uitest

我们正在尝试使用Xamarin.iOS中的扩展名为UIAlertController设置AutomationId。但是该应用var valuePtr = objc_getAssociatedObject(alertAction.Handle, DescriptiveName.Handle);在GetAccessibilityIdentifier方法下的崩溃情况始终不一致。也不知道为什么它崩溃了。

下面是代码示例:

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("");
        }

    }
Run Code Online (Sandbox Code Playgroud)

创建警报控制器,并使用以下扩展方法设置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(); });
}
Run Code Online (Sandbox Code Playgroud)

我们通过从视图控制器中单击按钮来调用此方法,并且该方法可以正常运行几次。但是当第3次或第4次显示警报时,应用程序崩溃了。

请帮助我解决此问题。

提前致谢。

Lan*_*SFT 2

AssociationPolicy.Retain也许会在意外的时间发布此属性,因此您的应用程序崩溃了。您可以将其更改为AssociationPolicy.RetainNonAtomic. NonAtomic 将使您的应用程序运行更流畅、更快。

如果您只想存储一个字符串并稍后使用它。为什么不尝试NSUserDefaults或静态实例?