c# - 在新的 Windows 10 CalendarView 上显示/设计额外信息

标签 c# wpf xaml win-universal-app windows-10

我最近开始开发 Windows 10 应用程序,该应用程序需要在 CalendarView 上显示事件和额外信息。随着新API的发布,还引入了新的CalendarView组件,所以我决定尝试一下。这是一个不错的小部件,但定制却很困难。

我已经可以使用 ControlTemplate 显示自定义信息,但使用 VisualState 绑定(bind)事件和样式却相当困难。

这是我使用的包裹在样式中的 ControlTemplate。

<Style x:Key="dayItemStyle" TargetType="CalendarViewDayItem">
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="CalendarViewDayItem">
                    <Grid x:Name="DayItemEventListRoot">
                        <ListView ItemsSource="{TemplateBinding DataContext}" Padding="20,0,0,0" x:Name="EventInfoList" 
                                  IsItemClickEnabled="True" cm:Message.Attach="[Event ItemClick] = [ListTapped]">
                            <ListView.ItemTemplate>
                                <DataTemplate>
                                    <StackPanel x:Name="EventInfoPanel" Orientation="Horizontal">
                                        <TextBlock x:Name="EventTime" Text="{Binding Date, Converter={StaticResource StringFormatter}, ConverterParameter=\{0:HH:mm\} }"
                                                  Foreground="{Binding Foreground, RelativeSource={RelativeSource Self}}" >
                                        </TextBlock>
                                        <TextBlock x:Name="EventDesc" Text="{Binding Name}" Padding="5,0,0,0" Foreground="Black" >

                                        </TextBlock>
                                        <VisualStateManager.VisualStateGroups>
                                            <VisualStateGroup>
                                                <VisualState x:Name="Normal" />
                                                <VisualState x:Name="Today" >
                                                    <VisualState.Setters>
                                                        <Setter Target="EventTime.Foreground" Value="White" />
                                                        <Setter Target="EventDesc.Foreground" Value="White" />
                                                        <Setter Target="EventTime.Text" Value="DASFASDDF" />
                                                    </VisualState.Setters>
                                                </VisualState>
                                            </VisualStateGroup>
                                        </VisualStateManager.VisualStateGroups>
                                    </StackPanel>
                                </DataTemplate>
                            </ListView.ItemTemplate>
                        </ListView>

                    </Grid>
                </ControlTemplate>
            </Setter.Value>
        </Setter>
    </Style>

然后直接在 CalenderView 组件上设置样式。

<CalendarView Name="FlowCalendar" HorizontalAlignment="Stretch" VerticalAlignment="Stretch" VerticalDayItemAlignment="Top" HorizontalDayItemAlignment="Left"
                  Grid.Column="0" Grid.Row="0" Grid.ColumnSpan="5" Grid.RowSpan="4" CalendarViewDayItemChanging="FlowCalendar_CalendarViewDayItemChanging"
                  CalendarViewDayItemStyle="{StaticResource dayItemStyle}">
</CalendarView>

额外信息和状态由隐藏代码中的 CalenderViewDayItemChanging 事件控制。

private void FlowCalendar_CalendarViewDayItemChanging(CalendarView sender, CalendarViewDayItemChangingEventArgs args)
{            
    if(args.Item.Date.Date.Equals(DateTime.Now.Date))
    {
        VisualStateManager.GoToState(args.Item, "Today", false);
        // Testing, gives NullPointerException
        //TextBlock bla = (TextBlock) args.Item.FindName("EventTime");
        //bla.Text = "SDADASFASDF";
    }
    if (args.Phase == 0)
    {
        var eventsByDate = ViewModel.Upcoming.FirstOrDefault(eg => eg.Key.Date == args.Item.Date.Date);
        if (eventsByDate != null)
        {
            args.Item.DataContext = eventsByDate.ToList();
        }
    }
}

设置视觉状态没有任何作用(我已经检查过它是否被调用),将 VisualState(Group) 移到 ControlTemplate 之外只会给我一个错误,指出无法找到目标。

我希望通过视觉状态来控制事件 ListView 样式,目前这是自定义的,因为我不确定 CalendarViewDayItem 有哪些内置状态。我对视觉状态很陌生,因此非常感谢任何指示。

谢谢。

最佳答案

您的代码将无法工作,因为 VisualStateManager.GoToStateControl 作为第一个参数;但是,您的 VisualStateManager 是在类型为 PanelStackPanel 内定义的。

您将需要实现自己的 VSM,它需要一个 Panel。看看this post的答案关于如何做到这一点。

但是,这仍然无法解决您的问题。由于您需要以某种方式找到 StackPanel(注意,因为它位于 ListView 内,所以可能有多个),然后调用 ExtendedVisualStateManager.GoToState >.

我建议您将此模板包装在 UserControl 中(通过这样做,您可能 不需要甚至不需要扩展VSM 因为您可以使用 GoToStateAction 代替)并具有依赖属性 (IsToday) 来控制状态。然后,您可以使用 ElementName 绑定(bind)将 CalendarViewDayItem 级别的属性向下传递到 IsToday 以便进行状态更改。

更新

关于需要使用ExtendedVisualManager来更改视觉状态,我实际上是错误的。由于它已经位于 UserControl 内,因此您实际上可以直接调用 VisualStateManager.GoToState(this, "statename", flag);

但是,您定义依赖属性的方式是错误的。

将后面的代码替换为以下代码,它应该可以工作。

public bool IsToday
{
    get { return (bool)GetValue(IsTodayProperty); }
    set { SetValue(IsTodayProperty, value); }
}

public static readonly DependencyProperty IsTodayProperty =
    DependencyProperty.Register("IsToday", typeof(bool), typeof(EventListTemplate), new PropertyMetadata(false, OnIsTodayChanged));

static void OnIsTodayChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
    var template = (EventListTemplate)d;

    if ((bool)e.NewValue)
    {
        var stateset = VisualStateManager.GoToState(template, "DayItemToday", false);

        Debug.WriteLine("did it:", stateset);
    }
}

关于c# - 在新的 Windows 10 CalendarView 上显示/设计额外信息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32136188/

相关文章:

c# - 托管 WinRT 组件 + 虚拟方法

c# - 当 WPF ComboBox 更改时触发命令

wpf - 如何在 XAML 中将颜色转换为画笔?

c# - 将 DataTable 作为参数发送到存储过程

C# 并行扩展 Task.Factory.StartNew 在错误的对象上调用方法

c# - 如果 StreamReader 或 StreamWriter 没有关闭会发生什么?

c# - 在 WPF 客户端应用程序中设置用户控件的数据上下文

WPF - 部署问题 - 错误模块名称 : KERNELBASE. dll

c# - 使用 WPF/C# 中的绑定(bind)获取更改的数据

wpf - XAML 文本框在焦点上更改边框颜色