MVVMCross Binding rozbija aplikację na Androida
Mam aplikację na Androida opartą na Xamarin i MvvmCross. W tej aplikacji jest widok z ExpandableListView, który stworzyłem samodzielnie. Teraz ta lista wyświetla kilka elementów, które są powiązane z ich DataContext za pomocą MvvmCross. Jednakże, ponieważ widoki poszczególnych ListItemView różnią się tak bardzo, część tych ListItemViews jest generowana programowo w ExpandedListViewAdapter. Działa to tak:
public override View GetChildView(int groupPosition, int childPosition, bool isLastChild, View convertView, ViewGroup parent)
{
object child = GetRawChild(groupPosition, childPosition);
if (child == null)
{
MvxBindingTrace.Trace(MvxTraceLevel.Error, "GetView called for group that seems to have no itemssource: it is null");
return null;
}
var view = (MvxListItemView)GetBindableView(convertView, child, ChildItemTemplateId);
var placeholder = view.FindViewById<BindableFrameLayout>(Resource.Id.placeholder);
var questionVm = (QuestionViewModel)child;
if(questionVm.ViewType == "TextBox")
{
placeholder.RemoveAllViews();
var text = new BindableEditText(context);
text.InputType = InputType;
text.SetRawInputType(InputType);
placeholder.RemoveAllViews();
placeholder.AddView(text);
var answer = questionVm.Children.First();
text.DataContext = answer;
var binding = text.CreateInlineBindingTarget<AnswerViewModel>();
text.Bind(binding, et => et.Text, vm => vm.Model.Value, (string)null, null, null,
MvxBindingMode.TwoWay);
}
else if(questionVm.ViewType == "Spinner")
{
placeholder.RemoveAllViews();
MvxSpinner spinner = new MvxSpinner(context, null);
spinner.ItemsSource = questionVm.Children;
spinner.ItemSelected += (sender, args) =>
{
for (int i = 0; i < questionVm.Children.Count; i++)
{
var answer = (IAnswerViewModel)questionVm.Children[i];
if (i == spinner.SelectedItemPosition)
answer.IsSelected = true;
else
answer.IsSelected = false;
}
};
spinner.Bind(bindings, ctrl => ctrl.ItemsSource, vm => vm.Children, (string)null, null, null, MvxBindingMode.OneWay);
var chosenAnswer = questionVm.Children.Cast<IAnswerViewModel>().FirstOrDefault(@a => @a.IsSelected == true);
if (chosenAnswer != null)
spinner.SetSelection(questionVm.Children.Cast<IAnswerViewModel>().ToList().IndexOf(chosenAnswer));
placeholder.AddView(spinner);
}
... i "BindableEditText" wygląda następująco: using System; przy użyciu Android.Content; przy użyciu Android.Runtime; korzystanie z Android.Views; korzystanie z Android.Widget; przy użyciu Android.Util; przy użyciu Cirrious.MvvmCross.Binding.Droid.BindingContext; przy użyciu Cirrious.MvvmCross.Binding.BindingContext;
namespace iCL.Filler.Droid.Controls
{
public class BindableEditText : EditText, IMvxBindingContextOwner
{
private readonly IMvxAndroidBindingContext _bindingContext;
public BindableEditText(Context context)
: base(context)
{
_bindingContext = new MvxAndroidBindingContext(context, null);
}
public BindableEditText(Context context, IAttributeSet attributes)
: base(context, attributes)
{
_bindingContext = new MvxAndroidBindingContext(context, null);
}
public BindableEditText(Context context, IAttributeSet attributes, int defStyle)
: base(context, attributes, defStyle)
{
_bindingContext = new MvxAndroidBindingContext(context, null);
}
public BindableEditText(IntPtr javaReference, JniHandleOwnership transfer)
: base(javaReference, transfer)
{
}
protected IMvxAndroidBindingContext AndroidBindingContext
{
get { return _bindingContext; }
}
public IMvxBindingContext BindingContext
{
get { return _bindingContext; }
set { throw new NotImplementedException("BindingContext is readonly in the radio button"); }
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
this.BindingContext.ClearAllBindings();
}
base.Dispose(disposing);
}
public override void SetText(Java.Lang.ICharSequence text, TextView.BufferType type)
{
try
{
base.SetText(text, type);
}
catch (Exception ex)
{
}
}
protected View Content { get; set; }
public object DataContext
{
get { return _bindingContext.DataContext; }
set { _bindingContext.DataContext = value; }
}
}
}
Więc moim problemem jest to, że raz na jakiś czas, gdy przewijam i klikam na mój widok listy, otrzymuję błąd wykonania, taki jak poniżej, a moja aplikacja „zawiesza się”, co oznacza, że wraca do poprzedniego ekranu.
10-29 14:04:37.140 D/dalvikvm( 5989): GC_EXPLICIT freed 751K, 11% free 11369K/12679K, paused 0ms+1ms
10-29 14:04:39.970 D/dalvikvm( 5989): GC_FOR_ALLOC freed 715K, 16% free 10692K/12679K, paused 5ms
10-29 14:04:41.831 E/mono-rt ( 5989): Stacktrace:
10-29 14:04:41.831 E/mono-rt ( 5989):
10-29 14:04:41.831 E/mono-rt ( 5989): at <unknown> <0xffffffff>
10-29 14:04:41.831 E/mono-rt ( 5989): at (wrapper managed-to-native) object.wrapper_native_0xb71f1820 (intptr,intptr,intptr,intptr,Android.Runtime.JValue[]) <IL 0x00124, 0xffffffff>
10-29 14:04:41.831 E/mono-rt ( 5989): at Android.Runtime.JNIEnv.CallNonvirtualVoidMethod (intptr,intptr,intptr,Android.Runtime.JValue[]) [0x00000] in /Users/builder/data/lanes/monodroid-mlion-monodroid-4.8.2-branch/a25a31d0/source/monodroid/src/Mono.Android/src/Runtime/JNIEnv.g.cs:612
10-29 14:04:41.831 E/mono-rt ( 5989): at Android.Widget.CompoundButton.set_Checked (bool) [0x00070] in /Users/builder/data/lanes/monodroid-mlion-monodroid-4.8.2-branch/a25a31d0/source/monodroid/src/Mono.Android/platforms/android-14/src/generated/Android.Widget.CompoundButton.cs:255
10-29 14:04:41.831 E/mono-rt ( 5989): at (wrapper runtime-invoke) <Module>.runtime_invoke_void__this___byte (object,intptr,intptr,intptr) <IL 0x00054, 0xffffffff>
10-29 14:04:41.831 E/mono-rt ( 5989): at <unknown> <0xffffffff>
10-29 14:04:41.831 E/mono-rt ( 5989): at (wrapper managed-to-native) System.Reflection.MonoMethod.InternalInvoke (System.Reflection.MonoMethod,object,object[],System.Exception&) <IL 0x00030, 0xffffffff>
10-29 14:04:41.831 E/mono-rt ( 5989): at System.Reflection.MonoMethod.Invoke (object,System.Reflection.BindingFlags,System.Reflection.Binder,object[],System.Globalization.CultureInfo) <IL 0x0004a, 0x0016f>
10-29 14:04:41.831 E/mono-rt ( 5989): at System.Reflection.MethodBase.Invoke (object,object[]) <IL 0x00006, 0x00048>
10-29 14:04:41.831 E/mono-rt ( 5989): at Cirrious.MvvmCross.Binding.Bindings.Target.MvxPropertyInfoTargetBinding.SetValueImpl (object,object) <IL 0x0001a, 0x000a7>
10-29 14:04:41.831 E/mono-rt ( 5989): at Cirrious.MvvmCross.Binding.Bindings.Target.MvxConvertingTargetBinding.SetValue (object) <IL 0x0008c, 0x002b1>
The program 'Mono' has exited with code 0 (0x0).
Nie wiem, na czym polega problem .... Czy to możliwe, że jakiś obiekt Java jest już sfinalizowany, a moje powiązania próbują go nazwać?