本文主要写我开发遇到常(zhi)见(zhang)问题。

真的有那么多问题?其实我就想记 ListViewItem 的问题,我自己都不记得在什么时候写如何设置他不压缩。后面越记录就越多啦

本文记录的内容适用于 UWP 和 WinUI 和 UNO 框架

设置 ListView 列表项宽度撑开

默认 ListViewItem 是宽度压缩。想要实现宽度撑开,可在 ListView 里,编写 ItemContainerStyle 样式,核心是设置 HorizontalContentAlignment 为 Stretch 的值,代码如下

  <ListView.ItemContainerStyle>
     <Style TargetType="ListViewItem">
         <Setter Property="HorizontalContentAlignment" Value="Stretch"/>
     </Style>
  </ListView.ItemContainerStyle>

更多关于 ListView 请看 win10 UWP ListView

如果需要设置 WPF 的 ListView 宽度,可以使用HorizontalContentAlignment="Stretch"

UWP 设置文本使用文字图标

在 UWP 很经常看到如下图这样的图标

这就是 Segoe MDL2 icons 字体图标,可以使用下面不同的方式在 xaml 使用这些图标

SymbolIcon

可以使用的 SymbolIcon 直接给名字,相对可读性比较好

<SymbolIcon Symbol="GlobalNavigationButton"/>

FontIcon

<FontIcon FontFamily="Segoe MDL2 Assets" Glyph="&#xE700;"/>

TextBlock

使用 TextBlock 和 FontIcon 差不多

<TextBlock FontFamily="Segoe MDL2 Assets" Text="&#xE700;"/>

参见:Segoe MDL2 icon guidelines - UWP app developer

设置 TargetType 样式

编译时 VisualStudio 告诉 Style object must specify a String value for the TargetType property 就是存在样式没有设置 TargetType 属性的值,在UWP所有的样式都需要添加 TargetType 才可以使用。

      Error WMC0080: Style object must specify a String value for the TargetType property (107, 10)

原来的代码

        <Style x:Key="KguypnDitzbbi">
            <Setter Property="FontFamily" Value="Segoe MDL2 Assets" />
            <Setter Property="Width" Value="38"></Setter>
        </Style>

修改后的代码


        <Style x:Key="KguypnDitzbbi" TargetType="TextBlock">
            <Setter Property="FontFamily" Value="Segoe MDL2 Assets" />
            <Setter Property="Width" Value="38"></Setter>
        </Style>

使用 VisualStateManager 编写简单的按钮样式

代码例子如下

  <Page.Resources>
    <Style x:Key="Style.Button.FooButtonStyle" TargetType="Button">
      <Setter Property="Template">
        <Setter.Value>
          <ControlTemplate TargetType="Button">
            <Grid x:Name="RootGrid" Background="{TemplateBinding Background}">
              <!-- 由于 VisualStateManager 必须放在容器里面,因此这里必须需要有一个容器 想省一点的话,可以用 Border 代替 Grid 做容器-->
              <VisualStateManager.VisualStateGroups>
                <VisualStateGroup x:Name="CommonStates">
                  <VisualState x:Name="Normal">
                    <Storyboard>
                      <!--啥都不用做,清空状态即可-->
                      <!-- 正常状态需要放在第一个 -->
                    </Storyboard>
                  </VisualState>
                  <VisualState x:Name="PointerOver">
                    <!-- Hover 效果 -->
                    <VisualState.Setters>
                      <Setter Target="ButtonContentPath.Stroke" Value="Transparent"></Setter>
                      <Setter Target="ButtonContentPath.Fill" Value="Blue"></Setter>
                    </VisualState.Setters>
                  </VisualState>
                  <VisualState x:Name="Pressed">
                    <!-- 按下状态 -->
                    <Storyboard>
                      <!-- 颜色用 ColorAnimation 也可以 -->
                      <!-- <ColorAnimation To=""></ColorAnimation> -->
                      <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath" Storyboard.TargetProperty="Stroke" >
                        <DiscreteObjectKeyFrame KeyTime="0" Value="Transparent"/>
                      </ObjectAnimationUsingKeyFrames>

                      <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath" Storyboard.TargetProperty="Fill" >
                        <DiscreteObjectKeyFrame KeyTime="0" Value="#FF666666"/>
                      </ObjectAnimationUsingKeyFrames>
                    </Storyboard>
                  </VisualState>
                  <VisualState x:Name="Disabled">
                    <!-- 不可用的状态 -->
                  </VisualState>
                </VisualStateGroup>
              </VisualStateManager.VisualStateGroups>

              <Path x:Name="ButtonContentPath" StrokeThickness="2" Stroke="#FF666666" Data="M7,15C6.85289858,15.5677816,6.85289858,16.4322348,7,17L22,29C22.7348015,29.3762198,24,28.8227297,24,28L24,4C24,3.1772867,22.7348015,2.62379657,22,3L7,15z"></Path>
            </Grid>
          </ControlTemplate>
        </Setter.Value>
      </Setter>
    </Style>

  </Page.Resources>
  <StackPanel
    HorizontalAlignment="Center"
    VerticalAlignment="Center">
    <Border BorderBrush="Black" BorderThickness="0">
      <Button Style="{StaticResource Style.Button.FooButtonStyle}"
            Background="Transparent" Padding="0">

      </Button>
    </Border>
  </StackPanel>

想要让 VisualStateManager 的 VisualStateGroups 生效,必须放在容器里面,直接写在 ControlTemplate 下面是不行的

第一个 Normal 的 VisualState 需要放在最前面,里面可以不写任何的代码,将会自动清空状态

设置属性的值时候,既可以使用 Setters 的方式,也可以使用动画的方式。使用 Setters 的代码比较短

 <VisualState x:Name="PointerOver">
   <!-- Hover 效果 -->
   <VisualState.Setters>
     <Setter Target="ButtonContentPath.Stroke" Value="Transparent"></Setter>
     <Setter Target="ButtonContentPath.Fill" Value="Blue"></Setter>
   </VisualState.Setters>
 </VisualState>

使用 Setters 时不需要管 Property 属性,只需要保证 Target 是 对象.属性 的写法就好了

使用动画的例子如下

 <VisualState x:Name="Pressed">
   <!-- 按下状态 -->
   <Storyboard>
     <!-- 颜色用 ColorAnimation 也可以 -->
     <!-- <ColorAnimation To=""></ColorAnimation> -->
     <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath" Storyboard.TargetProperty="Stroke" >
       <DiscreteObjectKeyFrame KeyTime="0" Value="Transparent"/>
     </ObjectAnimationUsingKeyFrames>

     <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath" Storyboard.TargetProperty="Fill" >
       <DiscreteObjectKeyFrame KeyTime="0" Value="#FF666666"/>
     </ObjectAnimationUsingKeyFrames>
   </Storyboard>
 </VisualState>

动画代码比较多,但是可以实现比较柔和的效果。因为 Setters 是立刻变化的,动画可以实现慢慢变化。对于视觉效果比较大的范围,推荐使用动画

具体一个控件有哪些 VisualState 可以设置,需要查阅文档。自己编写 VisualState 也是没问题的

官方文档: https://learn.microsoft.com/en-us/uwp/api/windows.ui.xaml.visualstate

以上代码放在githubgitee 欢迎访问

可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin cdad42f92c6aaa25f4a6abfe14166c72defd3b45

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin cdad42f92c6aaa25f4a6abfe14166c72defd3b45

获取代码之后,进入 UnoDemo/ButtonDemo 文件夹

有些状态设置无效

比如 StrokeThickness 等类似的会导致布局变更的属性,这些属性都是不给设置动画的

当然了,对于 StrokeThickness 还是能够接受 ObjectAnimationUsingKeyFrames 动画的,如以下代码是有效的

 <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath"
                                Storyboard.TargetProperty="StrokeThickness">
   <ObjectAnimationUsingKeyFrames.KeyFrames>
     <DiscreteObjectKeyFrame KeyTime="0:0:0" Value="0.0"></DiscreteObjectKeyFrame>
   </ObjectAnimationUsingKeyFrames.KeyFrames>
 </ObjectAnimationUsingKeyFrames>

但是如下代码是无效的,这是为了性能考虑的原因

 <DoubleAnimation Storyboard.TargetName="ButtonContentPath"
                  Storyboard.TargetProperty="StrokeThickness"
                  Duration="0:0:1"
                  To="0" />

想要在 DoubleAnimation 里面使用,需要明确设置 EnableDependentAnimation 属性,如以下代码

 <DoubleAnimation Storyboard.TargetName="ButtonContentPath"
                  Storyboard.TargetProperty="StrokeThickness"
                  EnableDependentAnimation="True"
                  Duration="0:0:1"
                  To="0" />

以上例子的全部代码如下

  <Page.Resources>
    <Style x:Key="Style.Button.EmptyButtonStyle" TargetType="Button">
      <Setter Property="Template">
        <Setter.Value>
          <ControlTemplate TargetType="Button">
            <Grid>
              <VisualStateManager.VisualStateGroups>
                <VisualStateGroup x:Name="CommonStates">
                  <VisualState x:Name="Normal">
                  </VisualState>
                  <VisualState x:Name="PointerOver">
                    <!-- Hover 效果 -->
                    <Storyboard>
                      <DoubleAnimation Storyboard.TargetName="ButtonContentPath"
                                       Storyboard.TargetProperty="StrokeThickness"
                                       EnableDependentAnimation="True"
                                       Duration="0:0:1"
                                       To="0" />

                      <DoubleAnimation Storyboard.TargetName="ButtonContentPath"
                                       Storyboard.TargetProperty="Opacity"
                                       Duration="0:0:3"
                                       To="0.6" />
                   

                    </Storyboard>
                  </VisualState>

                  <VisualState x:Name="Pressed">
                    <Storyboard>

                      <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath"
                                                     Storyboard.TargetProperty="StrokeThickness">
                        <ObjectAnimationUsingKeyFrames.KeyFrames>
                          <DiscreteObjectKeyFrame KeyTime="0:0:0" Value="0.0"></DiscreteObjectKeyFrame>
                        </ObjectAnimationUsingKeyFrames.KeyFrames>
                      </ObjectAnimationUsingKeyFrames>

                      <ObjectAnimationUsingKeyFrames Storyboard.TargetName="ButtonContentPath"
                                                     Storyboard.TargetProperty="Fill">
                        <ObjectAnimationUsingKeyFrames.KeyFrames>
                          <DiscreteObjectKeyFrame KeyTime="0:0:0" Value="Blue"></DiscreteObjectKeyFrame>
                        </ObjectAnimationUsingKeyFrames.KeyFrames>
                      </ObjectAnimationUsingKeyFrames>

                     
                    </Storyboard>
                  </VisualState>
                </VisualStateGroup>
              </VisualStateManager.VisualStateGroups>

              <Path x:Name="ButtonContentPath" StrokeThickness="5" Stroke="#FF666666"
                    Data="M7,15C6.85289858,15.5677816,6.85289858,16.4322348,7,17L22,29C22.7348015,29.3762198,24,28.8227297,24,28L24,4C24,3.1772867,22.7348015,2.62379657,22,3L7,15z">
              </Path>
            </Grid>
          </ControlTemplate>
        </Setter.Value>
      </Setter>
    </Style>

  </Page.Resources>
  <StackPanel
    HorizontalAlignment="Center"
    VerticalAlignment="Center">
    <Border BorderBrush="Black" BorderThickness="0">
      <Button Style="{StaticResource Style.Button.EmptyButtonStyle}"
              Background="Transparent" Padding="0">

      </Button>
    </Border>
  </StackPanel>

以上代码放在githubgitee 欢迎访问

可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin 68545b360aa349688f13bdb7734b7ad90e7199c3

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin 68545b360aa349688f13bdb7734b7ad90e7199c3

获取代码之后,进入 UnoDemo/ButtonDemo 文件夹

List 添加重复项

有时候需要写一个测试,添加1000个相同项,难道使用 For ?

实际有简单方法

                List<string> Items = Enumerable.Repeat("lindexi 需要重复文字", n/*多少个*/).ToList();

上面不仅是文字,因为使用List<string> ,实际使用什么,就添加什么。

Distinct linq 匿名类去重

http://www.aneasystone.com/archives/2015/04/distinct-in-csharp-linq.html

绑定 DataContext 的所有方法

win10 uwp DataContext

如果觉得ms需要做一些功能,或想去说垃圾ms,请到Welcome to the Windows developer feedback site!

UWP中获取Encoding.Default

    Encoding.GetEncoding(0);

参见:UWP中获取Encoding.Default - yinyue200 - 博客园

UWP 使用 GBK 读取文本

核心是因为 .NET Core 下没有带上 GBK 编码,解决方法请参阅

判断 ctrl 按下

判断 ctrl 按下,可以使用:

    (Window.Current.CoreWindow.GetKeyState(VirtualKey.Control) & CoreVirtualKeyStates.Down) != 0

判断其它的键,如 shift 键,使用方法也一样。

UWP 获取软件版本

在 UWP 获取自己软件的版本可以用来和服务器比较,判断当前是否需要升级。可以使用下面的代码获得 UWP 应用的版本

Windows.ApplicationModel.Package.Current.Id.Version

UWP 应用获取各类系统、用户信息 (1) - 设备和系统的基本信息、应用包信息、用户数据账户信息和用户账户信息

换成判断程序集版本号也是可以的

获取窗口大小

获取主窗口可视大小

Windows.UI.ViewManagement.ApplicationView.GetForCurrentView().VisibleBounds

当前窗口的大小

Window.Current.Bounds

获取当前窗口的其他方法

Window.Current.CoreWindow.Bounds

上面的方法可以获得窗口的 x 和 y 大小

ContentDialog 透明背景

张高兴的 UWP 开发笔记:定制 ContentDialog 样式 - 张高兴 - 博客园

bind 的默认 mode

{x:Bind} 的默认 Mode 是 OneTime。{Binding} 的默认 mode 是 OneWay.

参见 Data binding in depth

常用转换器

布尔转换器

允许自己配置的转换器

using Microsoft.UI.Xaml.Data;

public class BoolToVisibilityConverter : IValueConverter
{
    public Visibility TrueFor { set; get; } = Visibility.Visible;
    public Visibility FalseFor { set; get; } = Visibility.Collapsed;
    public Visibility NullFor { set; get; } = Visibility.Collapsed;

    public object Convert(object value, Type targetType, object parameter, string language)
    {
        if (value is true)
        {
            return TrueFor;
        }
        else if (value is false)
        {
            return FalseFor;
        }
        else
        {
            return NullFor;
        }
    }

    public object? ConvertBack(object value, Type targetType, object parameter, string language)
    {
        if (value is Visibility visibility)
        {
            if (visibility == TrueFor)
            {
                return true;
            }
            else if (visibility == FalseFor)
            {
                return false;
            }
            else if (visibility == NullFor)
            {
                return null;
            }
        }
        return null;
    }
}

布尔不可见转换器

public class BoolTrueToCollapsedConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, string language)
    {
        if (value is true)
        {
            return Visibility.Collapsed;
        }
        return Visibility.Visible;
    }
    
    public object ConvertBack(object value, Type targetType, object parameter, string language)
    {
        throw new NotSupportedException();
    }
}

布尔可见转换器

public class BoolTrueToVisibleConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, string language)
    {
        if (value is true)
        {
            return Visibility.Visible;
        }
        return Visibility.Collapsed;
    }
    
    public object ConvertBack(object value, Type targetType, object parameter, string language)
    {
        throw new NotSupportedException();
    }
}

使用管道通讯

在非 UWP 的桌面程序或控制台编写管道服务端。在 UWP 使用 CreateFileW 作为客户端。但此方式无法通过 Windows App Certification Kit (WACK) 检测。绕过 WACK 方法请参阅 三种方式轻松绕过Windows App Certification Kit (WACK)的API检测(UWP, C++) David Huang’s Blog

以下是非 UWP 的控制台的管道服务端代码

using (var pipe = NamedPipeServerStreamAcl.Create("FooPipe", PipeDirection.InOut, -1, PipeTransmissionMode.Byte, PipeOptions.None, 0, 0, null, HandleInheritability.None, PipeAccessRights.ChangePermissions))
{
    PipeSecurity ps = pipe.GetAccessControl();
    PipeAccessRule clientRule = new PipeAccessRule(
        new SecurityIdentifier("S-1-15-2-1"), // All application packages
        PipeAccessRights.ReadWrite,
        AccessControlType.Allow);
    PipeAccessRule ownerRule = new PipeAccessRule(
        WindowsIdentity.GetCurrent().Owner!,
        PipeAccessRights.FullControl,
        AccessControlType.Allow);
    ps.AddAccessRule(clientRule);
    ps.AddAccessRule(ownerRule);
    pipe.SetAccessControl(ps);
    pipe.WaitForConnection();
    using (var streamReader = new StreamReader(pipe, Encoding.UTF8))
    {
        for (int i = 0; i < int.MaxValue; i++)
        {
            string? message = streamReader.ReadLine();

            Console.WriteLine($"收到消息: {message}");

            var streamWriter = new StreamWriter(pipe, Encoding.UTF8);
            streamWriter.WriteLine($"[{i}] 服务端已收到消息");
            streamWriter.Flush();
        }
    }
}

以下是 UWP 的代码

    public sealed partial class MainPage : Page
    {
        public MainPage()
        {
            this.InitializeComponent();

            Loaded += MainPage_Loaded;
        }

        private void MainPage_Loaded(object sender, RoutedEventArgs e)
        {
            const uint readAndWriteAccess = unchecked((uint) (1 << 30 | 1 << 31));
            var handle = CreateFileW(@"\\.\pipe\FooPipe", readAndWriteAccess, 0/*FILE_SHARE_NONE*/, IntPtr.Zero, 4/*OPEN_ALWAYS*/, 0/*SECURITY_ANONYMOUS*/, IntPtr.Zero);
            using (SafeFileHandle safeHandle = new SafeFileHandle(handle, true))
            {
                using (var fileStream = new FileStream(safeHandle, FileAccess.ReadWrite))
                {
                    using (var streamWriter = new StreamWriter(fileStream, Encoding.UTF8))
                    {
                        var streamReader = new StreamReader(fileStream, Encoding.UTF8);

                        while (true)
                        {
                            streamWriter.WriteLine("测试发送信息");
                            streamWriter.Flush();

                            var message = streamReader.ReadLine();
                            Debug.WriteLine(message);
                        }
                    }
                }
            }
        }

        [DllImport("KERNEL32.dll", ExactSpelling = true, SetLastError = true)]
        [DefaultDllImportSearchPaths(DllImportSearchPath.System32)]
        public static extern IntPtr CreateFileW(
           [MarshalAs(UnmanagedType.LPWStr)] string lpFileName,
            uint dwDesiredAccess,
            uint dwShareMode,
            [Optional] IntPtr lpSecurityAttributes,
            uint dwCreationDisposition,
            uint dwFlagsAndAttributes,
            IntPtr hTemplateFile);
    }

以上代码放在 githubgitee 上,可以使用如下命令行拉取代码。我整个代码仓库比较庞大,使用以下命令行可以进行部分拉取,拉取速度比较快

先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin bf97a4c77400f5921b7e1947b160e23ca3fcf202

以上使用的是国内的 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码,将 gitee 源换成 github 源进行拉取代码。如果依然拉取不到代码,可以发邮件向我要代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin bf97a4c77400f5921b7e1947b160e23ca3fcf202

获取代码之后,进入 Workbench/KochufafallwalljoGelriciwu 文件夹,即可获取到源代码

常见构建失败

提示 TemplateBindingExtension 错误

错误信息大概如下

WMC0615 Type 'TemplateBindingExtension' used after '{' must be a Markup Extension. Error code 0x09c6

原因大概就是用了 TemplateBindingExtension 去绑定,如下面代码,而正确的方法应该是使用 TemplateBinding 作为 XAML 绑定

BorderThickness="{TemplateBindingExtension BorderThickness}"

修复代码是换成 TemplateBinding 去做

BorderThickness="{TemplateBinding BorderThickness}"

协变 逆变

协变 out ,如:string->object (子类到父类的转换) 逆变 in ,如:object->string (父类到子类的转换)

https://haojima.net/Home/Blog/53

有趣故事

开发中,会遇到一些有趣的故事。

有一天vs告诉我,你写的代码,连标点符号我也不信。

我是世界上少见的不需要写单元测试的程序员,因为我写的代码都是bug。

下面,我来告诉你一些有趣的故事:

有趣的“烫烫烫烫”和“屯屯屯屯” - imjustice - 博客园

更多笑话请看:

工具

介绍一下好用的工具

HappyStudio.UwpToolsLibrary.Auxiliarys 1.0.3 UWP工具库的辅助类库

HappyStudio.UwpToolsLibrary.Information 1.0.2 UWP 工具库的信息类库

HappyStudio.UwpToolsLibrary.Control 1.1.0 UWP 工具类库的控件库

HappyStudio.UwpToolsLibrary 1.0.4 给UWP工具类库的其他模块使用的依赖类库

WinRT XAML Toolkit for Windows 10 2.3.0


本文会经常更新,请阅读原文: https://blog.lindexi.com/post/UWP-%E5%BC%80%E5%8F%91%E4%B8%AD-%E9%9C%80%E8%A6%81%E7%9F%A5%E9%81%93%E7%9A%841000%E4%B8%AA%E9%97%AE%E9%A2%98.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者收藏我的博客导航

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

微软最具价值专家


无盈利,不卖课,做纯粹的技术博客

以下是广告时间

推荐关注 Edi.Wang 的公众号

欢迎进入 Eleven 老师组建的 .NET 社区

以上广告全是友情推广,无盈利