WPF Viewport3D 解决透视模式时窗体模糊
最近折腾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" />


点
|
说明
|
A
|
视点
|
|
|
线
|
说明
|
BC
|
查看的目标宽度
|
bc
|
展示的视窗宽度
|
AD
|
视点到查看目标的距离
|
ad
|
视点到视窗的距离
|
|
|
角度
|
说明
|
cAb
|
视角(FieldOfView)
|
|
|
<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 解决透视模式时窗体模糊的更多相关文章
- 如何解决卸载McAfee时出现“处于托管模式时无法删除”问题(转)
问题现象: 这几天在为客户终端换装杀毒软件时出现这么一个问题:在控制面板的添加或删除程序里面将“McAfee VirusScan Enterprise和 McAfee AntiSpyware Ente ...
- WPF 在TextBox失去焦点时检测数据,出错重新获得焦点解决办法
WPF 在TextBox失去焦点时检测数据,出错重新获得焦点解决办法 在WPF的TextBox的LostFocus事件中直接使用Focus()方法会出现死循环的问题 正确的使用方式有2中方法: 方法一 ...
- [WPF疑难]避免窗口最大化时遮盖任务栏
原文 [WPF疑难]避免窗口最大化时遮盖任务栏 [WPF疑难]避免窗口最大化时遮盖任务栏 周银辉 WPF窗口最大化时有个很不好的现象是:如果窗口的WindowStyle被直接或间接地设置为None后( ...
- WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参
原文:WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参 ContextMenu无论定义在.cs或.xaml文件中,都不继承父级的DataC ...
- WPF中ItemsControl应用虚拟化时找到子元素的方法
原文:WPF中ItemsControl应用虚拟化时找到子元素的方法 wpf的虚拟化技术会使UI的控件只初始化看的到的子元素, 而不是所有子元素都被初始化,这样会提高UI性能. 但是我们经常会遇到一个 ...
- WPF Prism MVVM 中 弹出新窗体. 放入用户控件
原文:WPF Prism MVVM 中 弹出新窗体. 放入用户控件 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/qq_37214567/artic ...
- C# WPF 一直保持多个Topmost窗体的置顶顺序
原文:C# WPF 一直保持多个Topmost窗体的置顶顺序 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/m0_37862405/article/ ...
- 解决Gradle编译时出现: 编码GBK的不可映射字符
解决Gradle编译时出现: 编码GBK的不可映射字符 在build.gradle文件中加入如下内容: [compileJava, compileTestJava]*.options*.encodin ...
- SpringMVC学习系列-后记 解决GET请求时中文乱码的问题
SpringMVC学习系列-后记 解决GET请求时中文乱码的问题 之前项目中的web.xml中的编码设置: <filter> <filter-name>CharacterEnc ...
随机推荐
- Redis实战——redis主从复制和集群实现原理
出自:https://blog.csdn.net/nuli888/article/details/52136822 redis主从复制redis主从配置比较简单,基本就是在从节点配置文件加上:slav ...
- 转 CentOS下面安装RVM+ruby+Rails
CentOS6.2下面安装RVM+ruby+Rails (1)RVM官方网站应该是改版过一次, 使用 curl -L https://get.rvm.io | bash -s stable 下载并安装 ...
- 安卓开发之不通过USB数据线调试的方法
搞安卓开发的程序员肯定都需要用又笨又碍手碍脚的USB数据线连接安卓设备进行调试,是不是觉得非常麻烦? 然后为了避免这种麻烦,我们一般通过网络上的各种方法,比如AS的插件ADB WIFI,不知道大家是如 ...
- jena读取和解析本体文件
使用jena开发本体应用程序时,首先需要对我们利用本体构建工具,如protege等,构建的本体文件,如owl.rdf等读取并解析得到本体模型.下面给出相应的代码,不对的地方请指正. (基于jena 2 ...
- 201671010140. 2016-2017-2 《Java程序设计》java学习第十五周
java学习第十五周 Java的GUI界面设计,框架以及主要部件填充,归置,布局管理,在第十一章和第十二章进行了系统的学习,在这两章的知识奠基下,可以简单的构造一个GUI用户界面,在两周的学习后,可以 ...
- ActiveMQ集群整体认识
出自:https://segmentfault.com/a/1190000014592517 前言 最终需要掌握 Replicated LevelDB Store部署方式,这种部署方式是基于ZooKe ...
- HttpServletResponse返回页面弹窗
下载方法: @RequestMapping(value = "/download.htm") public void downLoadFile(String id,HttpServ ...
- Linux Debian 下LNMP服务器——nginx+mysql+php环境搭建及配置
昨天刚给公司服务器装了LNMP服务器环境,在这里简单记录一下过程备忘. 这里我在安装的时候是用的Dotdeb源,仅供参考. 1.导入Dotdeb源,据说Dotdeb源里的软件版本比较新. 在向源中导入 ...
- Simple Cubemap Reflection
[Simple Cubemap Reflection] Cubemap加在MainTex上,所以Property需要按如下定义: 注意_Cubemap的类型是CUBE. 使用Cubemap,需要计算反 ...
- 欢迎加入本人建的QQ群,讨论技术,生活及每天都有招聘信息
银狐AS3游戏2 3438131 银狐AS3Help 1803321 银狐U3D 167024653 银狐AS3转H5 467565102 本人在MSDN上的帐号 https://social.msd ...