Dlaczego otrzymuję komunikat „Określona obsada jest nieprawidłowa”, gdy używam widoku niestandardowego z powiązaniem, ale nie, gdy ustawiam ją ręcznie?
Buduję widok niestandardowy przy użyciu ObservableCollection jako BindableProperty (moim celem było zaktualizowanie formatu widoku w zależności od jego zawartości, w tym przypadku listy kolorów). Myślę, że właściwość została poprawnie skonfigurowana w widoku niestandardowym. Jeśli ustawię właściwość możliwą do powiązania w C #, to działa. Jeśli spróbuję użyć {Binding varname} wewnątrz XAML, zgłasza błąd „Specified Cast is not Valid”. Próbuję dowiedzieć się, gdzie jest mój problem.
Poniżej w public MainPage () możesz zobaczyć zakomentowaną linię, w której ręcznie ustawiłem właściwość. Jeśli odkomentuję to i wyciągnę „Color = {Binding Test}” z MainPage.xaml, wszystko będzie działać. Jeśli uruchomię to tak, jak jest, otrzymam wyjątek.
Oto cały mój kod:
MainPage.xaml.cs
public partial class MainPage : ContentPage
{
public ObservableCollection<Color> Test { get; set; }
public MainPage()
{
Test = new ObservableCollection<Color>();
Test.Add(Color.Blue);
Test.Add(Color.Green);
Test.Add(Color.Red);
Test.Add(Color.Purple);
this.BindingContext = this;
InitializeComponent();
//myView.Colors = Test;
}
private void OnClicked(object sender, EventArgs e)
{
myView.Colors.Add(Color.Yellow);
}
}
MainPage.xaml
<?xml version="1.0" encoding="utf-8" ?>
<ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:local="clr-namespace:ColorBoxes"
x:Class="ColorBoxes.MainPage">
<StackLayout HorizontalOptions="CenterAndExpand">
<local:ColorBoxView x:Name="myView" Colors="{Binding Test}" WidthRequest="400" HeightRequest="600"/>
<Button Text="Add Color" Clicked="OnClicked"/>
</StackLayout>
</ContentPage>
ColorBoxView.xaml.cs
public partial class ColorBoxView : ContentView
{
public ColorBoxView()
{
InitializeComponent();
}
public static BindableProperty ColorList = BindableProperty.Create(
propertyName: "Colors",
returnType: typeof(ObservableCollection<Color>),
declaringType: typeof(ColorBoxView),
defaultValue: new ObservableCollection<Color>(),
defaultBindingMode: BindingMode.OneWay,
propertyChanged: HandleColorListPropertyChanged
);
public ObservableCollection<Color> Colors
{
get { return (ObservableCollection<Color>) base.GetValue(ColorList); }
set
{
if (value != this.Colors)
{
base.SetValue(ColorList, value);
Colors.CollectionChanged += Items_CollectionChanged;
}
}
}
void Items_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
{
buildView(Colors);
}
private void buildView(ObservableCollection<Color> newColors)
{
/* This code is never reached */
}
private static void HandleColorListPropertyChanged(BindableObject bindable, object oldValue, object newValue)
{
ObservableCollection<Color> newColors = (ObservableCollection<Color>) newValue;
ColorBoxView thisView = (ColorBoxView) bindable;
thisView.buildView(newColors);
}
ColorBoxView.xaml
<?xml version="1.0" encoding="utf-8"?>
<ContentView xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="ColorBoxes.ColorBoxView" >
<Grid x:Name="BoxGrid" HorizontalOptions="FillAndExpand" VerticalOptions="FillAndExpand"
RowSpacing="0" ColumnSpacing="0" Padding="0">
</Grid>
</ContentView>
Ślad stosu:
System.InvalidCastException: Specified cast is not valid.
at ColorBoxes.MainPage.InitializeComponent () [0x00023] in /Users/docmani/RiderProjects/ColorBoxes/ColorBoxes/ColorBoxes/obj/Debug/netstandard2.0/MainPage.xaml.g.cs:26
at ColorBoxes.MainPage..ctor () [0x00060] in /Users/docmani/RiderProjects/ColorBoxes/ColorBoxes/ColorBoxes/MainPage.xaml.cs:25
at ColorBoxes.App..ctor () [0x0000f] in /Users/docmani/RiderProjects/ColorBoxes/ColorBoxes/ColorBoxes/App.xaml.cs:15
at ColorBoxes.iOS.AppDelegate.FinishedLaunching (UIKit.UIApplication app, Foundation.NSDictionary options) [0x00007] in /Users/docmani/RiderProjects/ColorBoxes/ColorBoxes/ColorBoxes.iOS/AppDelegate.cs:25
at at (wrapper managed-to-native) UIKit.UIApplication.UIApplicationMain(int,string[],intptr,intptr)
at UIKit.UIApplication.Main (System.String[] args, System.IntPtr principal, System.IntPtr delegate) [0x00005] in /Library/Frameworks/Xamarin.iOS.framework/Versions/13.18.2.1/src/Xamarin.iOS/UIKit/UIApplication.cs:86
at UIKit.UIApplication.Main (System.String[] args, System.String principalClassName, System.String delegateClassName) [0x0000e] in /Library/Frameworks/Xamarin.iOS.framework/Versions/13.18.2.1/src/Xamarin.iOS/UIKit/UIApplication.cs:65
at ColorBoxes.iOS.Application.Main (System.String[] args) [0x00001] in /Users/docmani/RiderProjects/ColorBoxes/ColorBoxes/ColorBoxes.iOS/Main.cs:16
Dodatkowe informacje - próbowałem uaktualnić Xamarin.Forms w NuGet, ponieważ minęło trochę czasu. Teraz nawet się nie skompiluje z powiązaniem. To daje mi:
MainPage.xaml(9, 46): [XFC0009] No property, BindableProperty, or event found for "Colors", or mismatching type between value and property.
Ale znowu, jeśli ręcznie ustawię wartość w C #, nadal działa. Zdecydowanie brakuje mi czegoś w moim XAML, ale nie wiem, co to jest.
Odpowiedzi
Możesz zmienić BindableProperty
następujące ColorListProperty
rzeczy, a następnie wykonać test w pliku XAML.
public partial class ColorBoxView : ContentView
{
public ColorBoxView ()
{
InitializeComponent ();
}
public static readonly BindableProperty ColorListProperty = BindableProperty.Create(
nameof(ColorList),
typeof(ObservableCollection<object>),
typeof(ColorBoxView),
new ObservableCollection<object>(),
BindingMode.OneWay,
propertyChanged: ContractListPropertyChangedDelegate);
public ObservableCollection<object> ColorList
{
get =>
(ObservableCollection<object>)GetValue(ColorListProperty);
set => SetValue(ColorListProperty, value);
}
private static void ContractListPropertyChangedDelegate(BindableObject bindable, object oldValue, object newValue)
{
// throw new NotImplementedException();
// ColorBoxView colorBoxView= bindable as ColorBoxView;
}
<local:ColorBoxView x:Name="myView" ColorList="{Binding Test}" WidthRequest="400" HeightRequest="600"/>
Jeśli chcesz CollectionChanged
zdarzenia mają być wykonane, po kliknięciu przycisku można dodać ColorList.CollectionChanged += ColorList_CollectionChanged
w ColorBoxView
„s konstruktora.
public ColorBoxView ()
{
InitializeComponent ();
ColorList.CollectionChanged += ColorList_CollectionChanged;
}
private void ColorList_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
{
// throw new NotImplementedException();
}
Twoja set
metoda zostanie wykonana, gdy ustawisz nowy ObservableCollection
, Jeśli tylko dodać element do istniejącego ObservableCollection
, wykorzystania CollectionChanged
w ColorBoxView
być może lepiej.
Po wielu poszukiwaniach znalazłem kilka przykładów, w których typ BindingProperty został ustawiony na „IEnumerable”. Po ponownym wykonaniu kodu wymyśliłem, co działa:
public partial class ColorBoxView : ContentView
{
public ColorBoxView()
{
InitializeComponent();
}
public static readonly BindableProperty ColorsProperty =
BindableProperty.Create (nameof (Colors), typeof (IEnumerable), typeof (ColorBoxView), null,
BindingMode.Default, null, HandleColorListPropertyChanged);
public IEnumerable Colors
{
get { return (IEnumerable) GetValue (ColorsProperty); }
set
{
SetValue (ColorsProperty, value);
var oc = (ObservableCollection<Color>) (this.Colors);
oc.CollectionChanged += Items_CollectionChanged;
}
}
private void Items_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
{
buildView( (ObservableCollection<Color>)Colors);
}
private void buildView(ObservableCollection<Color> newColors)
{
/* Update code */
}
private static void HandleColorListPropertyChanged(BindableObject bindable, object oldValue, object newValue)
{
ObservableCollection<Color> newColors = (ObservableCollection<Color>) newValue;
ColorBoxView thisView = (ColorBoxView) bindable;
thisView.Colors = newColors;
thisView.buildView(newColors);
}
}
Byłbym bardzo zainteresowany, aby poznać oficjalny powód, dla którego to działa. Komentarze są mile widziane.