WPF 控件 (十五、时间选择器)

一、TimePicker

WPF 没有自带的时间选择器

1.1 xaml

由三个Combox(时、分、秒)和两个TextBlock(::)组成

<UserControl x:Class="lyrics.ui.Controls.TimePickerCustom.TimePicker"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:lyrics.ui.Controls.TimePickerCustom"
             mc:Ignorable="d" 
             d:DesignHeight="50" d:DesignWidth="250" Background="White">
    <Grid>
        <StackPanel Orientation="Horizontal">
            <ComboBox Name="hoursCb" MinWidth="80" ItemsSource="{Binding hours}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50" SelectionChanged="hoursChanged"/>
            <TextBlock Text=":" VerticalAlignment="Center"/>
            <ComboBox Name="minutesCb" MinWidth="80" ItemsSource="{Binding minutes}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50" SelectionChanged="minutesChanged"/>
            <TextBlock Text=":" VerticalAlignment="Center"/>
            <ComboBox Name="secondsCb" MinWidth="80" ItemsSource="{Binding seconds}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50"  SelectionChanged="secondsChanged"/>
        </StackPanel>
    </Grid>
</UserControl>

1.2 后台

提供时间改变的事件

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;

namespace lyrics.ui.Controls.TimePickerCustom
{
    /// <summary>
    /// TimePicker.xaml 的交互逻辑
    /// </summary>
    public partial class TimePicker : UserControl
    {
        public List<int> hours { get; set; } = new List<int>();
        public List<int> minutes { get; set; } = new List<int>();
        public List<int> seconds { get; set; } = new List<int>();

        public int hour { get; set; } = 0;

        public int minute { get; set; } = 0;

        public int second { get; set; } = 0;

        public TimeSpan time { get; set; }

        public static readonly RoutedEvent TimeChangedEvent = EventManager.RegisterRoutedEvent("TimeChanged", RoutingStrategy.Direct, typeof(RoutedEventHandler), typeof(TimePicker));

        public event RoutedEventHandler TimeChanged
        {
            add
            {
                AddHandler(TimeChangedEvent, value);
            }
            remove
            {
                RemoveHandler(TimeChangedEvent, value);
            }
        }
        public TimePicker()
        {
            for (int i = 0; i < 24; i++) { 
                hours.Add(i);
            }
            for (int i = 0; i < 60; i++)
            {
                minutes.Add(i);
                seconds.Add(i);
            }
            
            InitializeComponent();
            this.DataContext = this;
        }


        private void hoursChanged(object sender, SelectionChangedEventArgs e)
        {
            hour = hoursCb.SelectedIndex;
            time = new TimeSpan(hour,minute,second);
            NotifyEvent();
        }

        private void minutesChanged(object sender, SelectionChangedEventArgs e)
        {
            minute = minutesCb.SelectedIndex;
            time = new TimeSpan(hour, minute, second);
            NotifyEvent();
        }

        private void secondsChanged(object sender, SelectionChangedEventArgs e)
        {
            second = secondsCb.SelectedIndex;
            time = new TimeSpan(hour, minute, second);
            NotifyEvent();
        }

        private void NotifyEvent() {
            if (!IsLoaded) {
                return;
            }
            RoutedEventArgs routedEvent = new RoutedEventArgs();
            routedEvent.RoutedEvent = TimeChangedEvent;
            RaiseEvent(routedEvent);
        }
    }
}

二、使用

2.1 Demo

xaml只需要这一行

 <tp:TimePicker x:Name="tp" Height="50" TimeChanged="tp_TimeChanged"/>

后台事件

private void tp_TimeChanged(object sender, RoutedEventArgs e)
{
    MessageBox.Show("已经设置时间为:"+tp.time.ToString());
}

2.2 效果

控件刷新被弹窗阻,需要点击弹窗才会刷新
在这里插入图片描述
在这里插入图片描述

三、修改样式

3.1 Combox样式修改

 <Style x:Key="ComboBoxReadonlyToggleButton2" TargetType="ToggleButton">
        <Setter Property="IsTabStop" Value="false" />
        <Setter Property="ClickMode" Value="Press" />
        <Setter Property="MinHeight" Value="22"></Setter>
        <Setter Property="MinWidth" Value="80"></Setter>
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="ToggleButton">
                    <Grid>
                        <Border x:Name="Background" BorderBrush="#787878"
                                BorderThickness="1"  Background="Black">
                        </Border>
                        <!--倒三角的颜色#0099ff-->
                    </Grid>

                    <ControlTemplate.Triggers>
                        <Trigger Property="IsMouseOver" Value="True">
                            <!--移入时按钮颜色 图案颜色 背景颜色 锯齿颜色-->
                            <Setter TargetName="Background" Property="Opacity" Value="0.4"></Setter>
                        </Trigger>
                    </ControlTemplate.Triggers>
                </ControlTemplate>
            </Setter.Value>
        </Setter>
    </Style>
    <Style x:Key="ComboBoxStyle3" TargetType="ComboBox">
        <Setter Property="Foreground" Value="#FFFFFFFF" />
        <Setter Property="Width" Value="50"></Setter>
        <Setter Property="Height" Value="30"></Setter>
        <Setter Property="VerticalAlignment" Value="Center"></Setter>
        <Setter Property="HorizontalContentAlignment" Value="Left"></Setter>
        <Setter Property="BorderBrush" Value="#FFFFFFFF" />
        <Setter Property="BorderThickness" Value="1" />
        <Setter Property="VerticalContentAlignment" Value="Center" />
        <Setter Property="ScrollViewer.HorizontalScrollBarVisibility" Value="Auto" />
        <Setter Property="ScrollViewer.VerticalScrollBarVisibility" Value="Auto" />
        <Setter Property="Padding" Value="4,3" />
        <Setter Property="FontSize" Value="18" />
        <Setter Property="ItemContainerStyle">
            <Setter.Value>
                <!--这个是下拉框的属性设置-->
                <Style TargetType="ComboBoxItem">
                    <Setter Property="Template">
                        <Setter.Value>
                            <ControlTemplate TargetType="ComboBoxItem">
                                <Border Name="Back"  Background="#525252"  BorderThickness="0,0,0,0" BorderBrush="#525252" >
                                    <ContentPresenter ContentSource="{Binding Source}"  VerticalAlignment="Center" HorizontalAlignment="Left" Margin="10,0,0,0" ></ContentPresenter>
                                </Border>
                                <ControlTemplate.Triggers>
                                    <Trigger Property="IsMouseOver" Value="True">
                                        <Setter TargetName="Back" Property="Background" Value="#484848"></Setter>
                                    </Trigger>
                                    <!--下拉框背景色-->
                                    <Trigger Property="IsHighlighted" Value="True">
                                        <Setter TargetName="Back" Property="Background" Value="Gray"></Setter>
                                    </Trigger>
                                </ControlTemplate.Triggers>
                            </ControlTemplate>
                        </Setter.Value>
                    </Setter>
                </Style>
            </Setter.Value>
        </Setter>
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="ComboBox">
                    <Grid x:Name="MainGrid">
                        <Popup x:Name="PART_Popup" Margin="1" Grid.Row="1" 
                                IsOpen="{Binding Path=IsDropDownOpen, RelativeSource={RelativeSource TemplatedParent}}">
                            <Grid x:Name="Shdw" MaxHeight="{TemplateBinding MaxDropDownHeight}"
                                    MinWidth="{Binding Path=ActualWidth, ElementName=MainGrid}" >
                                <!--下拉框的属性设置-->
                                <!--Background -->
                                <Border x:Name="Bordertop"   Width="60" CornerRadius="0" BorderThickness="0" Background="#525252">
                                    <Border.Effect>
                                        <DropShadowEffect Color="Black"  BlurRadius="2" ShadowDepth="0" Opacity="1"/>
                                    </Border.Effect>
                                    <ScrollViewer Width="150">
                                        <ItemsPresenter/>
                                    </ScrollViewer>
                                </Border>
                            </Grid>
                        </Popup>
                        <ToggleButton Style="{StaticResource ComboBoxReadonlyToggleButton2}" Opacity="0"
                                BorderThickness="3" BorderBrush="#03ffea"   Background="{TemplateBinding Background}"
                                IsChecked="{Binding Path=IsDropDownOpen, Mode=TwoWay, RelativeSource={RelativeSource TemplatedParent}}" />
                        <!--combox字体的间距-->
                        <TextBlock Margin="10 0 0 0"  
                                   FontSize="20"
                                HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}"
                                VerticalAlignment="{TemplateBinding VerticalContentAlignment}"
                                IsHitTestVisible="false"
                                   Foreground="White"
                                Text="{TemplateBinding Text}"
                               />
                    </Grid>

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

3.2 修改Xaml

<UserControl x:Class="lyrics.ui.Controls.TimePickerCustom.TimePicker"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:lyrics.ui.Controls.TimePickerCustom"
             mc:Ignorable="d" 
             d:DesignHeight="50" d:DesignWidth="150" Background="Transparent">
    <Grid>
        <Border Background="Black" CornerRadius="25">
            <StackPanel Orientation="Horizontal" HorizontalAlignment="Center">
                <ComboBox Name="hoursCb" MinWidth="40" ItemsSource="{Binding hours}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50" SelectionChanged="hoursChanged" Style="{StaticResource  ComboBoxStyle3}" HorizontalContentAlignment="Center"/>
                <TextBlock Text=":" VerticalAlignment="Center" FontSize="50" Foreground="White"/>
                <ComboBox Name="minutesCb" MinWidth="40" ItemsSource="{Binding minutes}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50" SelectionChanged="minutesChanged"
                      Style="{StaticResource  ComboBoxStyle3}" HorizontalContentAlignment="Center"/>
                <TextBlock Text=":" VerticalAlignment="Center" FontSize="50" Foreground="White"/>
                <ComboBox Name="secondsCb" MinWidth="40" ItemsSource="{Binding seconds}" SelectedIndex="0" VerticalContentAlignment="Center" MinHeight="50"  SelectionChanged="secondsChanged"
                      Style="{StaticResource  ComboBoxStyle3}" HorizontalContentAlignment="Center"/>
            </StackPanel>
        </Border>
    </Grid>
</UserControl>

3.3 修改调用

  <tp:TimePicker x:Name="tp" Height="50" Width="200" TimeChanged="tp_TimeChanged" />

3.4 效果

默认:
在这里插入图片描述

修改时间
在这里插入图片描述
选中
在这里插入图片描述

修改事件触发
在这里插入图片描述

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: WPF是一种强大的UI框架,它提供了许多控件,其中包括时间选择器控件。但是,由于在特定的应用程序中,需要自定义时间选择器以满足一些个性化需求。在这种情况下,可以使用WPF自定义控件来创建自己的时间选择器控件。 首先,在WPF中创建时间选择器控件,需要使用Calendar控件和TimePicker控件。Calendar控件用于显示日期,而TimePicker控件用于选择时间时间选择器控件的主体是StackPanel控件。在StackPanel控件中添加了两个控件Calendar和TimePicker,以实现时间选择器的基本功能。 然后,需要在时间选择器控件中定义一些附加属性,例如:选定日期、选定时间等等,以实现一些高级功能。 最后,为时间选择器控件添加样式,并实现一些触发器和动画效果,以使其外观和功能与应用程序的主题相匹配。 实现WPF自定义时间选择器可能需要一些时间和经验,但对于需要一个不寻常的时间选择器的应用程序来说,是值得的。 这样的时间选择器是用户友好的,具有很好的设计和功能,并且以C#编写,可以很容易地与WPF应用程序集成。 ### 回答2: WPF自定义时间选择器是一种功能强大、灵活性高的工具,它可以根据需求自行设计不同的选择器,可以实现小时、分钟、秒数的选择等多种功能。 首先,我们需要使用WPF自带的DatePicker控件和TimePicker控件来实现时间选择器。接下来,我们可以自定义控件的样式和模板,使其更符合我们的设计需求。 在自定义控件的样式时,我们需要设置控件的各个属性,比如控件的边框、背景、字体等。同时,我们可以通过设置样式来调整控件的布局和显示效果。 在时间选择器的实现中,需要涉及到一些比较复杂的计算,比如计算时间的差值、时间的格式转换等。我们可以使用C#中的DateTime类和TimeSpan类来实现这些功能。 最后对于自定义时间选择器控件事件,需要自定义一些控件事件,使其更加符合我们的设计需求。比如增加或减少系统时钟里的时间。 总而言之,实现WPF自定义时间选择器需要对WPF控件、样式、模板、计算和控件事件等各个方面有深入的了解。只有掌握了这些知识,才能够设计出优秀的时间选择器,满足用户的需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值