最近折腾Viewport3D玩,遇到了一些诡异的问题,研究一下略有心得,特此和大家分享~

三维图形概述:

概要

三维坐标系

 
       二维图形的 WPF 坐标系将原点定位在呈现区域(通常是屏幕)的左上角。 在二维系统中,x 轴上的正值朝右,y 轴上的正值朝下。 但是,在三维坐标系中,原点位于呈现区域的中心,x 轴上的正值朝右,但是 y 轴上的正值朝上,z 轴上的正值从原点向外朝向观察者。

照相机

透视投影和正投影
 

关于TextureCoordinates(纹理坐标):

 
<MeshGeometry3D Positions="-250,-150,0 250,-150,0 250,150,0 -250,150,0"

                TextureCoordinates="0,1 1,1 1,0 0,0"

                TriangleIndices="0,1,2 0,2,3" />
 
上面这段MeshGeometry3D(用于生成3-D形状的三角形基元)对照图2三维坐标系会发现是个从左下角逆时针绘制的矩形。如果要将一个窗体UserControl纹理应用到Viewport2DVisual3D上,我们就需要合理的设置TextureCoordinates(纹理坐标)。对照图1二维坐标系,纹理坐标同样是由左下开始绘制的矩形。这样设置,纹理就可以显示正常了。
 
关于视角和距离的关系:
  
 
含义:
说明
A
视点
 
 
线
说明
BC
查看的目标宽度
bc
展示的视窗宽度
AD
视点到查看目标的距离
ad
视点到视窗的距离
 
 
角度
说明
cAb
视角(FieldOfView)
 
 
结论:
BD/bd=AD/Ad
 
我们如果想看到一个原样大小的窗体,就需要设置我的角cAb为。同时设置AD长度为查看目标的一半宽度。
 
 
 
 
好不容易设置好Viewport3D,将UserControl作为窗体的查看对象,却发现虽然窗体大小一致,但是显示出的UserControl字体和控件却很模糊。当我们以3D模型渲染窗体时,WPF应该做了性能优化,然而我并不清楚如何开启高清模式= =。
怎么解决这个问题呢?我试了好久,发现并没有比较简单快捷的解决方法。没办法,只能采取一些小技巧来解决啦。
 
具体思路:
第一步:窗体呈现时,将UserControl实例化后直接呈现到页面最前端的展示面板上,此时Viewport3D内容为空。
第二部:触发页面翻转动画时,将最前的展示面板置空,UserControl实例剪切到Viewport3D里面。
第三步:动画执行完毕后再从Viewport3D里面将UserControl剪切到页面最前端的展示面板中。
 
 示例代码:
<Window x:Class="App3DWindow.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:app3DWindow="clr-namespace:App3DWindow"
xmlns:asyncDelegate="clr-namespace:AsyncDelegate"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="MainWindow"
Width="500"
Height="310"
AllowsTransparency="True"
Background="Transparent"
Loaded="MainWindow_OnLoaded"
WindowStyle="None"
WindowStartupLocation="CenterScreen"
mc:Ignorable="d">
<Window.Resources>
<Storyboard x:Key="Turn2ConfigViewStoryboard">
<Rotation3DAnimationUsingKeyFrames Storyboard.TargetName="containerUIElement3D" Storyboard.TargetProperty="(Visual3D.Transform).(Transform3DGroup.Children)[2].(RotateTransform3D.Rotation)">
<EasingRotation3DKeyFrame KeyTime="0:0:2">
<EasingRotation3DKeyFrame.EasingFunction>
<QuinticEase EasingMode="EaseOut" />
</EasingRotation3DKeyFrame.EasingFunction>
<EasingRotation3DKeyFrame.Value>
<AxisAngleRotation3D Angle="180" Axis="0,1,0" />
</EasingRotation3DKeyFrame.Value>
</EasingRotation3DKeyFrame>
</Rotation3DAnimationUsingKeyFrames>
</Storyboard>
<Storyboard x:Key="Turn2DownloadViewStoryboard">
<Rotation3DAnimationUsingKeyFrames Storyboard.TargetName="containerUIElement3D" Storyboard.TargetProperty="(Visual3D.Transform).(Transform3DGroup.Children)[2].(RotateTransform3D.Rotation)">
<EasingRotation3DKeyFrame KeyTime="0:0:2">
<EasingRotation3DKeyFrame.EasingFunction>
<QuinticEase EasingMode="EaseOut" />
</EasingRotation3DKeyFrame.EasingFunction>
<EasingRotation3DKeyFrame.Value>
<AxisAngleRotation3D Angle="360" Axis="0,1,0" />
</EasingRotation3DKeyFrame.Value>
</EasingRotation3DKeyFrame>
</Rotation3DAnimationUsingKeyFrames>
</Storyboard>
<!-- 相机 -->
<PerspectiveCamera x:Key="PerspectiveCamera"
FieldOfView="90"
LookDirection="0,0,-1"
NearPlaneDistance="1"
Position="0,0,250"
UpDirection="0,1,0" /> <OrthographicCamera x:Key="OrthographicCamera"
Width="500"
LookDirection="0,0,-1"
Position="0,0,250"
UpDirection="0,1,0" />
</Window.Resources>
<Grid Background="Transparent">
<Viewport3D Margin="0">
<Viewport3D.Camera>
<!-- 相机 -->
<PerspectiveCamera x:Name="PerspectiveCamera"
FieldOfView="90"
LookDirection="0,0,-1"
NearPlaneDistance="1"
Position="0,0,250"
UpDirection="0,1,0" />
</Viewport3D.Camera>
<Viewport3D.Children>
<ContainerUIElement3D x:Name="containerUIElement3D">
<ContainerUIElement3D.Transform>
<Transform3DGroup>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<ScaleTransform3D ScaleX="1"
ScaleY="1"
ScaleZ="1" />
<RotateTransform3D d:EulerAngles="0,0,0">
<RotateTransform3D.Rotation>
<AxisAngleRotation3D Angle="0" Axis="0,1,0" />
</RotateTransform3D.Rotation>
</RotateTransform3D>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
</Transform3DGroup>
</ContainerUIElement3D.Transform> <!-- 正面视角 -->
<Viewport2DVisual3D x:Name="FrontViewport2DVisual3D">
<Viewport2DVisual3D.Transform>
<Transform3DGroup>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<ScaleTransform3D ScaleX="1"
ScaleY="1"
ScaleZ="1" />
<RotateTransform3D d:EulerAngles="0,0,0">
<RotateTransform3D.Rotation>
<AxisAngleRotation3D Angle="0" Axis="0,0,0" />
</RotateTransform3D.Rotation>
</RotateTransform3D>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
</Transform3DGroup>
</Viewport2DVisual3D.Transform>
<Viewport2DVisual3D.Geometry>
<MeshGeometry3D Positions="-250,-150,0 250,-150,0 250,150,0 -250,150,0"
TextureCoordinates="0,1 1,1 1,0 0,0"
TriangleIndices="0,1,2 0,2,3" />
</Viewport2DVisual3D.Geometry>
<Viewport2DVisual3D.Material>
<DiffuseMaterial Viewport2DVisual3D.IsVisualHostMaterial="True" />
</Viewport2DVisual3D.Material>
<Viewport2DVisual3D.Visual>
<app3DWindow:DownloadWindow x:Name="DownloadWindowPart"
Width="500"
Height="300"
Background="White"
SnapsToDevicePixels="True" />
</Viewport2DVisual3D.Visual>
</Viewport2DVisual3D>
<!-- 背面视角 -->
<Viewport2DVisual3D x:Name="BackViewport2DVisual3D">
<Viewport2DVisual3D.Transform>
<Transform3DGroup>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<ScaleTransform3D ScaleX="1"
ScaleY="1"
ScaleZ="1" />
<RotateTransform3D d:EulerAngles="0,0,0">
<RotateTransform3D.Rotation>
<AxisAngleRotation3D Angle="0" Axis="0,0,0" />
</RotateTransform3D.Rotation>
</RotateTransform3D>
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
<TranslateTransform3D OffsetX="0"
OffsetY="0"
OffsetZ="0" />
</Transform3DGroup>
</Viewport2DVisual3D.Transform>
<Viewport2DVisual3D.Geometry>
<MeshGeometry3D Positions="-250,-150,0 250,-150,0 250,150,0 -250,150,0"
TextureCoordinates="1,1 0,1 0,0 1,0 "
TriangleIndices="0,3,2 0,2,1" />
</Viewport2DVisual3D.Geometry>
<Viewport2DVisual3D.Material>
<DiffuseMaterial Viewport2DVisual3D.IsVisualHostMaterial="True" />
</Viewport2DVisual3D.Material>
<Viewport2DVisual3D.Visual>
<app3DWindow:ProxyConfigView x:Name="ProxyConfigViewPart"
Width="500"
Height="300"
Background="White"
SnapsToDevicePixels="True" />
</Viewport2DVisual3D.Visual>
</Viewport2DVisual3D>
</ContainerUIElement3D> <ModelVisual3D>
<ModelVisual3D.Content>
<!-- 光源 -->
<AmbientLight x:Name="ViewLight" />
</ModelVisual3D.Content>
</ModelVisual3D>
</Viewport3D.Children>
</Viewport3D>
<Grid x:Name="DisplayGrid">
<Grid.RowDefinitions>
<RowDefinition Height="40" />
<RowDefinition Height="*" />
</Grid.RowDefinitions>
<UserControl x:Name="DisPlayControl"
Grid.Row="0"
Grid.RowSpan="2" />
<!--<Border x:Name="Part_Drag"
Grid.Row="0"
Background="AliceBlue"
PreviewMouseLeftButtonDown="DisPlayControl_OnPreviewMouseLeftButtonDown" />-->
</Grid>
</Grid>
</Window>
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;
using System.Windows.Media.Animation; namespace App3DWindow
{
/// <summary>
/// MainWindow.xaml 的交互逻辑
/// </summary>
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
/// <summary>
/// 载入时进行模糊情况特殊处理
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void MainWindow_OnLoaded(object sender, RoutedEventArgs e)
{
//return;
SetDownloadWindowDisplayMode();
var turn2ConfigViewStoryboard = this.TryFindResource("Turn2ConfigViewStoryboard") as Storyboard;
if (turn2ConfigViewStoryboard != null)
{
turn2ConfigViewStoryboard.Completed += (s, e1) =>
{//动画执行完成切换展示模式
SetConfigViewDisplayMode();
SetDownloadWindowTurningMode();
};
turn2ConfigViewStoryboard.CurrentTimeInvalidated += (s, e1) =>
{//动画执行时切换成翻转模式
SetDownloadWindowTurningMode();
};
};
var turn2DownloadViewStoryboard = this.TryFindResource("Turn2DownloadViewStoryboard") as Storyboard;
if (turn2DownloadViewStoryboard != null)
{
turn2DownloadViewStoryboard.Completed += (s, e1) =>
{//动画执行完成切换展示模式
SetDownloadWindowDisplayMode();
SetConfigViewTurningMode();
};
turn2DownloadViewStoryboard.CurrentTimeInvalidated += (s, e1) =>
{//动画执行时切换成翻转模式
SetConfigViewTurningMode();
};
};
}
/// <summary>
/// 设置下载页面进入展示模式
/// </summary>
private void SetDownloadWindowDisplayMode()
{
//var container = DownloadWindowPart.Parent as Viewport2DVisual3D;
//if (container == null) return;
//container.Visual = null;
if (FrontViewport2DVisual3D.Visual==(DownloadWindowPart))
{
FrontViewport2DVisual3D.Visual = null;
}
DisPlayControl.Content = DownloadWindowPart;
}
/// <summary>
/// 设置配置页面进入展示模式
/// </summary>
private void SetConfigViewDisplayMode()
{
//var container = ProxyConfigViewPart.Parent as Viewport2DVisual3D;
//if (container == null) return;
//container.Visual = null;
if (BackViewport2DVisual3D.Visual==(ProxyConfigViewPart))
{
BackViewport2DVisual3D.Visual = null;
}
DisPlayControl.Content = ProxyConfigViewPart;
}
/// <summary>
/// 设置下载页面进入翻转模式
/// </summary>
private void SetDownloadWindowTurningMode()
{
var container = DownloadWindowPart.Parent as UserControl;
if (container == null) return;
container.Content = null;
FrontViewport2DVisual3D.Visual = DownloadWindowPart;
}
/// <summary>
/// 设置配置页面进入翻转模式
/// </summary>
private void SetConfigViewTurningMode()
{
var container = ProxyConfigViewPart.Parent as UserControl;
if (container == null) return;
container.Content = null;
BackViewport2DVisual3D.Visual = ProxyConfigViewPart;
}
/// <summary>
/// 支持拖拽
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void DisPlayControl_OnPreviewMouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
this.DragMove();
}
}
}
 呈现效果:
效果见下图,翻转动画执行完毕后不再模糊。
 
 
 

WPF Viewport3D 解决透视模式时窗体模糊的更多相关文章

  1. 如何解决卸载McAfee时出现“处于托管模式时无法删除”问题(转)

    问题现象: 这几天在为客户终端换装杀毒软件时出现这么一个问题:在控制面板的添加或删除程序里面将“McAfee VirusScan Enterprise和 McAfee AntiSpyware Ente ...

  2. WPF 在TextBox失去焦点时检测数据,出错重新获得焦点解决办法

    WPF 在TextBox失去焦点时检测数据,出错重新获得焦点解决办法 在WPF的TextBox的LostFocus事件中直接使用Focus()方法会出现死循环的问题 正确的使用方式有2中方法: 方法一 ...

  3. [WPF疑难]避免窗口最大化时遮盖任务栏

    原文 [WPF疑难]避免窗口最大化时遮盖任务栏 [WPF疑难]避免窗口最大化时遮盖任务栏 周银辉 WPF窗口最大化时有个很不好的现象是:如果窗口的WindowStyle被直接或间接地设置为None后( ...

  4. WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参

    原文:WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参 ContextMenu无论定义在.cs或.xaml文件中,都不继承父级的DataC ...

  5. WPF中ItemsControl应用虚拟化时找到子元素的方法

    原文:WPF中ItemsControl应用虚拟化时找到子元素的方法  wpf的虚拟化技术会使UI的控件只初始化看的到的子元素, 而不是所有子元素都被初始化,这样会提高UI性能. 但是我们经常会遇到一个 ...

  6. WPF Prism MVVM 中 弹出新窗体. 放入用户控件

    原文:WPF Prism MVVM 中 弹出新窗体. 放入用户控件 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/qq_37214567/artic ...

  7. C# WPF 一直保持多个Topmost窗体的置顶顺序

    原文:C# WPF 一直保持多个Topmost窗体的置顶顺序 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/m0_37862405/article/ ...

  8. 解决Gradle编译时出现: 编码GBK的不可映射字符

    解决Gradle编译时出现: 编码GBK的不可映射字符 在build.gradle文件中加入如下内容: [compileJava, compileTestJava]*.options*.encodin ...

  9. SpringMVC学习系列-后记 解决GET请求时中文乱码的问题

    SpringMVC学习系列-后记 解决GET请求时中文乱码的问题 之前项目中的web.xml中的编码设置: <filter> <filter-name>CharacterEnc ...

随机推荐

  1. 5.Redis 发布订阅

    转自:http://www.runoob.com/redis/redis-tutorial.html Redis 发布订阅(pub/sub)是一种消息通信模式:发送者(pub)发送消息,订阅者(sub ...

  2. __slots__

    __slots__ 由于Python是动态语言,任何实例在运行期都可以动态地添加属性. 如果要限制添加的属性,例如,Student类只允许添加 name.gender和score 这3个属性,就可以利 ...

  3. python开发最受欢迎的十款工具

    python开发最受欢迎的十款工具 dreamyla3个月前 今天小编给正在学习python开发的朋友介绍十款最受欢迎的开发工具,因为在学习python开发过程中少不了IDE或者代码编辑器,想要提高开 ...

  4. 关于 Apache Shiro 详解

    1.1  简介 Apache Shiro是Java的一个安全框架.目前,使用Apache Shiro的人越来越多,因为它相当简单,对比Spring Security,可能没有Spring Securi ...

  5. Python常用的一些内建函数和math模块函数

    一:Python内建函数 # abs取绝对值 num = -10 print(abs(num)) # max 求最大值 print(max(6, 9, 2, 12, 8)) # min求最小值 pri ...

  6. 编码总结,以及对BOM的理解

    一.前言 在跨平台.跨操作系统或者跨区域之间,经常会涉及到编码的问题,因为前段时间在项目中,遇到了因为编码而产生乱码的问题,以前对编码也是一知半解,所以决定对编码有一个更为深入的了解,因此才有了这篇自 ...

  7. 万能头文件#include<bits/stdc++.h>

    最近在打cf时赛后翻阅别人的代码总是会发现一个陌生而奇怪的头文件#include<bits/stdc++.h> 奇怪之处就在于基本上所有的代码只要用了这个头文件就不再写其他头文件了. 百度 ...

  8. OpenCV之设计模式

    参考资料: http://hahack.com/codes/opencv-and-mvc/ http://blog.csdn.net/yzhang6_10/article/details/508084 ...

  9. Setuptool+pip安装

    https://pypi.python.org/pypi/setuptools 1. 下载ez_setup.py文件,cmd进入安装目录: 2. python setup.py install htt ...

  10. java之yield(),sleep(),wait()区别详解-备忘笔记[转]

    1.sleep() 使当前线程(即调用该方法的线程)暂停执行一段时间,让其他线程有机会继续执行,但它并不释放对象锁.也就是说如果有synchronized同步快,其他线程仍然不能访问共享数据.注意该方 ...