Xamarin.Forms绑定到绑定属性或其他内容

时间:2018-08-25 10:17:30

标签: xaml xamarin xamarin.forms

我有一个页面

<?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:controls="clr-namespace:PassSystem.Controls;assembly=PassSystem"
             x:Class="PassSystem.Views.CreatePassPage"
             Title="Оформление пропуска">
    <ScrollView HorizontalOptions="FillAndExpand" VerticalOptions="FillAndExpand">
        <StackLayout>
            <FlexLayout Direction="Column" BackgroundColor="White">
                <controls:ActionOption Title="Название" Value="1" LeftMargin="18">
                </controls:ActionOption>
            </FlexLayout>
        </StackLayout>
    </ScrollView>
</ContentPage>

我有contentview

<?xml version="1.0" encoding="UTF-8"?>
<ContentView xmlns="http://xamarin.com/schemas/2014/forms"
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             xmlns:controls="clr-namespace:PassSystem.Controls;assembly=PassSystem"
             xmlns:effects="clr-namespace:PassSystem.Effects;assembly=PassSystem"
             xmlns:local="clr-namespace:PassSystem"
             x:Class="PassSystem.Controls.ActionOption">
    <FlexLayout x:Name="MainLayout" Direction="Row" JustifyContent="Center" AlignContent="Center" AlignItems="Center" HeightRequest="52"
                effects:ClickableEffect.ClickCommand="!!"
                effects:ClickableEffect.CancelCommand="{Binding UpCommand}"
                effects:ClickableEffect.UpCommand="{Binding UpCommand}"
                effects:ClickableEffect.DownCommand="{Binding DownCommand}">

        <Label Text="{Binding Title}"
               TextColor="{StaticResource PrimaryTextColor}"
               FontSize="15"
               VerticalTextAlignment="Center"
               VerticalOptions="FillAndExpand"
               FlexLayout.Grow="1"
               Margin="{Binding TitleMargin, Mode=TwoWay}">
        </Label>

        <StackLayout Orientation="Horizontal" HeightRequest="16" VerticalOptions="CenterAndExpand" Margin="0,0,8,0">
            <Label Text="{Binding Value}" FontSize="12" TextColor="{StaticResource SecondaryTextColor}" VerticalTextAlignment="Center"/>
            <controls:IconView Source="forward" HeightRequest="14" WidthRequest="14" ForegroundColor="{StaticResource AdditionalTextColor}"/>
        </StackLayout>
        <FlexLayout.Effects>
            <effects:ClickableEffect />
        </FlexLayout.Effects>
    </FlexLayout>
</ContentView>

.cs文件

public partial class ActionOption : ContentView
    {
        public ActionOption()
        {
            InitializeComponent();
            BindingContext = this;
        }

        private string _title;
        public string Title
        {
            get => _title;
            set
            {
                if(value == _title) return;
                _title = value;
                OnPropertyChanged();
            }
        }

        private string _value;
        public string Value
        {
            get => _value;
            set
            {
                if (value == _value) return;
                _value = value;
                OnPropertyChanged();
            }
        }

        public Thickness TitleMargin => new Thickness(_leftMargin, 0, 6, 0);

        private double _leftMargin;
        public double LeftMargin
        {
            get => _leftMargin;
            set
            {
                if (Math.Abs(value - _leftMargin) < 0.01d) return;
                _leftMargin = value;
                OnPropertyChanged();
                OnPropertyChanged(nameof(TitleMargin));
            }
        }

        private ICommand _clicked;
        public ICommand Clicked
        {
            get => _clicked;
            set
            {
                if (value == _clicked) return;
                _clicked = value;
                OnPropertyChanged();
            }
        }

        public ICommand UpCommand => new Command(() => MainLayout.BackgroundColor = Color.FromHex("#fff"));
        public ICommand DownCommand => new Command(() => MainLayout.BackgroundColor = (Color)((App)Application.Current).Resources["HighlightingColor"]);
    }

我需要从页面绑定ClickableEffect.ClickCommand。即

<controls:ActionOption Title="TitleHere" Value="1" LeftMargin="18" Clicked="{Binding ClickedCommand}">
</controls:ActionOption>

在控制之中

`effects:ClickableEffect.ClickCommand="{Binding ClickCommand (FromPage)}"`

其他信息。 ClickableEffect

{
        public ClickableEffect() : base("PassSystem.ClickableEffect")
        {
        }

        #region Click

        public static readonly BindableProperty ClickCommandProperty = BindableProperty
            .CreateAttached("ClickCommand", typeof(ICommand), typeof(ClickableEffect), (object)null);

        public static ICommand GetClickCommand(BindableObject view)
        {
            return (ICommand)view.GetValue(ClickCommandProperty);
        }

        public static void SetClickCommand(BindableObject view, ICommand value)
        {
            view.SetValue(ClickCommandProperty, value);
        }


        public static readonly BindableProperty ClickCommandParameterProperty = BindableProperty
            .CreateAttached("ClickCommandParameter", typeof(object), typeof(ClickableEffect), (object)null);

        public static object GetClickCommandParameter(BindableObject view)
        {
            return view.GetValue(ClickCommandParameterProperty);
        }

        public static void SetClickCommandParameter(BindableObject view, object value)
        {
            view.SetValue(ClickCommandParameterProperty, value);
        }

        #endregion
}

在Android上实现可点击效果:

public class ClickableListener : Java.Lang.Object, View.IOnTouchListener, View.IOnClickListener, View.IOnLongClickListener
    {
        private Element Element { get; }
        private View View { get; }

        public ClickableListener(Element element, View view)
        {
            Element = element;
            View = view;
        }

        ...

        public void OnClick(View v)
        {
            Tap();
        }

        private void Tap()
        {
            var command = ClickableEffect.GetClickCommand(Element);
            var parameter = ClickableEffect.GetClickCommandParameter(Element);
            command?.Execute(parameter);
        }
    }

[assembly: ResolutionGroupName("PassSystem")]
[assembly: ExportEffect(typeof(AndroidClickableEffect), "ClickableEffect")]
namespace PassSystem.Droid.Native.Effects
{
    public class AndroidClickableEffect : PlatformEffect
    {
        private bool _attached;

        protected override void OnAttached()
        {
            //because an effect can be detached immediately after attached (happens in listview), only attach the handler one time.
            if(!_attached)
            {
                var control = Control ?? Container;

                var listener = new ClickableListener(Element, control);
                control.SetOnClickListener(listener);
                ...
                _attached = true;
            }
        }
}

如何实施?谢谢。 附言我很难解释,希望您能理解。 :)

1 个答案:

答案 0 :(得分:0)

为了获得所需的行为:

  1. ClickCommand必须是ClickableEffect的附加属性。可以在官方文档Passing Effect Parameters as Attached Properties中找到更多信息。
  2. 由于效果和页面之间有一个ContentView,因此必须确保正确设置BindindContext中的ContentView。恐怕您将不得不在ClickCommand级别定义一个ContentView可绑定属性,并将其绑定到效果的命令。因此:Page.BindingContext.ClickCommand => ContentView.ClickCommand => ClickableEffect.ClickCommand(其中=>具有约束力)。

基本上提出了一个问题-为什么需要效果?