Почему MeshGeometry3D Plane невидима на определенных позициях во время Неаффинного Преобразования?
Добрый день всем, это мой первый вопрос здесь. Так что не стесняйтесь давать мне CC:)
О проекте. Я хочу не аффинно трансформироваться из плоской многоугольной плоскости без касания ее по оси Z (0,0,0 0,1,0 1,0,0 1,1,0). Мне нужно сделать это, разместив 4 угловых точки. Я достиг этого, используя "Неаффинные преобразования в 2D" Чарльза Петцольда ( http://www.charlespetzold.com/blog/2007/08/250638.html).
Пока все хорошо. И не исчезающий самолет.
Потому что мне нужны кисти / текстуры для расширения по краям плоскости. Я решил добавить треугольники, пока не получу сетку 5х5. (вся геометрия по-прежнему трансформируется путем выбора угловых точек первой плоскости (это точка посередине.)
К сожалению, я не могу публиковать фотографии.
https://i.imgur.com/oG3gKRD.png
Светло-синий прямоугольник - это тот, который имеет выбираемые углы, а пурпурный цвет - остальная его часть (для снимков я накрыл другую плоскость (которая не связана с большой, просто чтобы показать, что нет эффекта сумасшедшего преобразования). разница 1 в значении (независимо от угла)
https://i.imgur.com/wSkywVk.png
Насколько я могу судить, это не проблема с камерой. Изменение значений на многих вещах, таких как камера / ближний и дальний план. Такое расположение всегда приводит к невидимой геометрии (и это не единственная позиция, ведущая к ней, их несколько, но я не вижу связи), если я уменьшу сетку до плоскости 3х3, у меня та же проблема Только на других позициях. Объект все еще там, он просто не виден (я все еще могу выбрать угловые точки, и он будет сразу же виден снова)
Кто-нибудь знает, что может привести к этому? Или кто-то знает, как я могу отлаживать подобные "ошибки"?
может быть, это неправильный подход к выращиванию трансформируемой плоскости вообще? Если у кого есть идея, дайте мне знать!
Настройка для геометрии с обеими плоскостями (два пользовательских класса "scalingCanvas" и "CenterOnPoint", вы можете использовать обычный холст с нормальным положением)
<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:HtMatrix">
<Style TargetType="local:HtProjektor">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="local:HtProjektor">
<Grid Background="DarkOrange">
<Viewport3D Name="viewport3d" Width="{TemplateBinding ActualWidth}" Height="{TemplateBinding ActualHeight}" >
<ModelVisual3D>
<ModelVisual3D.Content>
<Model3DGroup>
<GeometryModel3D>
<GeometryModel3D.Geometry>
<MeshGeometry3D x:Name="mesh"
Positions="0 0 0, 0 1 0, 1 0 0, 1 1 0,
0 1 0, 0 2 0, 1 1 0, 1 2 0,
0 -1 0, 0 0 0, 1 -1 0, 1 0 0,
1 1 0, 1 2 0, 2 1 0, 2 2 0,
1 0 0, 1 1 0, 2 0 0, 2 1 0,
1 -1 0, 1 0 0, 2 -1 0, 2 0 0,
-1 0 0, -1 1 0, 0 0 0, 0 1 0,
-1 1 0, -1 2 0, 0 1 0, 0 2 0,
-1 -1 0, -1 0 0, 0 -1 0, 0 0 0,
-2 2 0, -2 3 0, -1 2 0, -1 3 0,
-1 2 0, -1 3 0, 0 2 0, 0 3 0,
0 2 0, 0 3 0, 1 2 0, 1 3 0,
1 2 0, 1 3 0, 2 2 0, 2 3 0,
2 2 0, 2 3 0, 3 2 0, 3 3 0
-2 1 0, -2 2 0, -1 1 0, -1 2 0,
2 1 0, 2 2 0, 3 1 0, 3 2 0,
-2 0 0, -2 1 0, -1 0 0, -1 1 0,
2 0 0, 2 1 0, 3 0 0, 3 1 0
-2 -1 0, -2 0 0, -1 -1 0, -1 0 0,
2 -1 0, 2 0 0, 3 -1 0, 3 0 0,
-2 -2 0, -2 -1 0, -1 -2 0, -1 -1 0,
-1 -2 0, -1 -1 0, 0 -2 0, 0 -1 0,
0 -2 0, 0 -1 0, 1 -2 0, 1 -1 0,
1 -2 0, 1 -1 0, 2 -2 0, 2 -1 0,
2 -2 0, 2 -1 0, 3 -2 0, 3 -1 0
"
TextureCoordinates="0 0, 0 1, 1 0, 1 1,
0 1, 0 2, 1 1, 1 2,
0 -1, 0 0, 1 -1, 1 0,
1 1, 1 2, 2 1, 2 2,
1 0, 1 1, 2 0, 2 1,
1 -1, 1 0, 2 -1, 2 0,
-1 0, -1 1, 0 0, 0 1,
-1 1, -1 2, 0 1, 0 2,
-1 -1, -1 0, 0 -1, 0 0,
-2 2, -2 3, -1 2, -1 3,
-1 2, -1 3, 0 2, 0 3,
0 2, 0 3, 1 2, 1 3,
1 2, 1 3, 2 2, 2 3,
2 2, 2 3, 3 2, 3 3
-2 1, -2 2, -1 1, -1 2,
2 1, 2 2, 3 1, 3 2,
-2 0, -2 1, -1 0, -1 1,
2 0, 2 1, 3 0, 3 1,
-2 -1, -2 0, -1 -1, -1 0,
2 -1, 2 0, 3 -1, 3 0
-2 -2, -2 -1, -1 -2, -1 -1,
-1 -2, -1 -1, 0 -2, 0 -1,
0 -2, 0 -1, 1 -2, 1 -1,
1 -2, 1 -1, 2 -2, 2 -1,
2 -2, 2 -1, 3 -2, 3 -1
"
TriangleIndices="0 2 1, 1 2 3,
4 6 5, 5 6 7,
8 10 9, 9 10 11,
12 14 13, 13 14 15,
16 18 17, 17 18 19,
20 22 21, 21 22 23,
24 26 25, 25 26 27,
28 30 29, 29 30 31,
32 34 33, 33 34 35,
36 38 37, 37 38 39,
40 42 41, 41 42 43,
44 46 45, 45 46 47,
48 50 49, 49 50 51,
52 54 53, 53 54 55
56 58 57, 57 58 59,
60 62 61, 61 62 63,
64 66 65, 65 66 67,
68 70 69, 69 70 71,
72 74 73, 73 74 75,
76 78 77, 77 78 79,
80 82 81, 81 82 83,
84 86 85, 85 86 87,
88 90 89, 89 90 91,
92 94 93, 93 94 95,
96 98 97, 97 98 99
" />
</GeometryModel3D.Geometry>
<GeometryModel3D.Material>
<DiffuseMaterial>
<DiffuseMaterial.Brush>
<VisualBrush>
<VisualBrush.Visual>
<local:ScalingCanvas Background="Gray" Width="{TemplateBinding ActualWidth}" Height="{TemplateBinding ActualHeight}">
<Grid Width="{TemplateBinding ActualWidth}" Height="{TemplateBinding ActualHeight}" ClipToBounds="True">
<local:ScalingCanvas Width="{Binding RelativeSource={RelativeSource TemplatedParent}}" Height="{Binding RelativeSource={RelativeSource TemplatedParent}}" Background="PaleVioletRed" Opacity="0.5">
<local:ScalingCanvas.RenderTransform>
<ScaleTransform ScaleX="{Binding ScaleX, RelativeSource={RelativeSource TemplatedParent}}" ScaleY="{Binding ScaleY, RelativeSource={RelativeSource TemplatedParent}}"/>
</local:ScalingCanvas.RenderTransform>
<Rectangle Height="768" Width="1024"/>
<Rectangle local:ScalingCanvas.Top="200" local:ScalingCanvas.Left="200" Fill="Aqua" Width="100" Height="100" />
<Rectangle local:CenterOnPoint.CenterPoint="100,100" Fill="Blue" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,0" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="1024,0" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,768" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="1024,768" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, 0" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="600, 768" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, 868" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, -100" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,350" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="1000,384" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="512,384" Fill="YellowGreen" Height="153.6" Width="204.8" Opacity="0.5"/>
<Image local:CenterOnPoint.CenterPoint="512,384" Source="IMG_4831.JPG" Width="100" Height="150" Stretch="Fill" />
</local:ScalingCanvas>
</Grid>
</local:ScalingCanvas>
</VisualBrush.Visual>
</VisualBrush>
</DiffuseMaterial.Brush>
</DiffuseMaterial>
</GeometryModel3D.Material>
<GeometryModel3D.Transform >
<Transform3DGroup>
<MatrixTransform3D x:Name="xform" />
</Transform3DGroup>
</GeometryModel3D.Transform>
</GeometryModel3D>
<GeometryModel3D>
<GeometryModel3D.Geometry>
<MeshGeometry3D x:Name="mesh1"
Positions="0 0 0, 0 1 0, 1 0 0, 1 1 0"
TextureCoordinates="0 0, 0 1, 1 0, 1 1"
TriangleIndices="0 2 1, 2 3 1" />
</GeometryModel3D.Geometry>
<GeometryModel3D.Material>
<DiffuseMaterial>
<DiffuseMaterial.Brush>
<VisualBrush>
<VisualBrush.Visual>
<local:ScalingCanvas Background="Gray" Width="{TemplateBinding ActualWidth}" Height="{TemplateBinding ActualHeight}">
<Grid Width="{TemplateBinding ActualWidth}" Height="{TemplateBinding ActualHeight}" ClipToBounds="True">
<local:ScalingCanvas Width="{Binding RelativeSource={RelativeSource TemplatedParent}}" Height="{Binding RelativeSource={RelativeSource TemplatedParent}}" Background="AliceBlue" Opacity="0.5">
<local:ScalingCanvas.RenderTransform>
<ScaleTransform ScaleX="{Binding ScaleX, RelativeSource={RelativeSource TemplatedParent}}" ScaleY="{Binding ScaleY, RelativeSource={RelativeSource TemplatedParent}}"/>
</local:ScalingCanvas.RenderTransform>
<Rectangle Height="768" Width="1024"/>
<Rectangle local:ScalingCanvas.Top="200" local:ScalingCanvas.Left="200" Fill="Aqua" Width="100" Height="100" />
<Rectangle local:CenterOnPoint.CenterPoint="100,100" Fill="Blue" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,0" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="1024,0" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,768" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="1024,768" Fill="Yellow" Width="100" Height="100" Opacity="0.7"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, 0" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="600, 768" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, 868" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="500, -100" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="0,350" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="1000,384" Fill="Black" Height="100" Width="100"/>
<Rectangle local:CenterOnPoint.CenterPoint="512,384" Fill="YellowGreen" Height="153.6" Width="204.8" Opacity="0.5"/>
<Image local:CenterOnPoint.CenterPoint="512,384" Source="IMG_4831.JPG" Width="100" Height="150" Stretch="Fill" />
</local:ScalingCanvas>
</Grid>
</local:ScalingCanvas>
</VisualBrush.Visual>
</VisualBrush>
</DiffuseMaterial.Brush>
</DiffuseMaterial>
</GeometryModel3D.Material>
<GeometryModel3D.Transform >
<Transform3DGroup>
<MatrixTransform3D x:Name="xform2" />
</Transform3DGroup>
</GeometryModel3D.Transform>
</GeometryModel3D>
<!--Light source.-->
<AmbientLight Color="White" />
</Model3DGroup>
</ModelVisual3D.Content>
</ModelVisual3D>
<!-- Camera. -->
<Viewport3D.Camera>
<OrthographicCamera Position="0.5 0.5 1"
LookDirection="0 0 -1"
UpDirection="0 1 0"
Width="1"
/>
</Viewport3D.Camera>
</Viewport3D>
</Grid>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ResourceDictionary>
Класс Projektor будет опубликован как ответ из-за ограничения характера. Если вам нужно что-то прояснить, просто дайте мне знать! И этот вопрос вообще большой? Приветствую конструктивного критика!:)
1 ответ
Класс Projektor с расчетами! Можно ли связать комментарий с основным вопросом?
HtProjektor.cs
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Media3D;
namespace HtMatrix
{
public class HtProjektor : Control
{
/// <summary>
/// Breite in mm
/// </summary>
public int Breite
{
get { return (int)GetValue(BreiteProperty); }
set { SetValue(BreiteProperty, value); }
}
/// <summary>
/// The <see cref="Breite"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty BreiteProperty = DependencyProperty.Register("Breite", typeof(int), typeof(HtProjektor), new PropertyMetadata(1024, _DisplayScaleChanged));
/// Höhe in mm
public int Hoehe
{
get { return (int)GetValue(HoeheProperty); }
set { SetValue(HoeheProperty, value); }
}
/// The <see cref="Hoehe"/> DependencyProperty.
public static readonly DependencyProperty HoeheProperty = DependencyProperty.Register("Hoehe", typeof(int), typeof(HtProjektor), new PropertyMetadata(768, _DisplayScaleChanged));
private static void _DisplayScaleChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
projektor.ScaleX = projektor.ActualWidth / projektor.Breite;
projektor.ScaleY = projektor.ActualHeight / projektor.Hoehe;
}
}
/// Skalierung X
public double ScaleX
{
get { return (double)GetValue(ScaleXProperty); }
set { SetValue(ScaleXProperty, value); }
}
/// The <see cref="ScaleX"/> DependencyProperty.
public static readonly DependencyProperty ScaleXProperty = DependencyProperty.Register("ScaleX", typeof(double), typeof(HtProjektor), new PropertyMetadata(0.0));
/// Skalierung Y
public double ScaleY
{
get { return (double)GetValue(ScaleYProperty); }
set { SetValue(ScaleYProperty, value); }
}
/// The <see cref="ScaleY"/> DependencyProperty.
public static readonly DependencyProperty ScaleYProperty = DependencyProperty.Register("ScaleY", typeof(double), typeof(HtProjektor), new PropertyMetadata(0.0));
/// The Top Left point
public Point CurrentMousePosition
{
get { return (Point)GetValue(CurrentMousePositionProperty); }
set { SetValue(CurrentMousePositionProperty, value); }
}
/// The <see cref="TopLeftPoint"/> DependencyProperty.
public static readonly DependencyProperty CurrentMousePositionProperty = DependencyProperty.Register("CurrentMousePosition", typeof(Point), typeof(HtProjektor));
/// The Top Left point
public Point TopLeftPoint
{
get { return (Point)GetValue(TopLeftPointProperty); }
set { SetValue(TopLeftPointProperty, value); }
}
/// The <see cref="TopLeftPoint"/> DependencyProperty.
public static readonly DependencyProperty TopLeftPointProperty = DependencyProperty.Register("TopLeftPoint", typeof(Point), typeof(HtProjektor), new PropertyMetadata(_Coordinate_Changed));
/// The Top Right point
public Point TopRightPoint
{
get { return (Point)GetValue(TopRightPointProperty); }
set { SetValue(TopRightPointProperty, value); }
}
/// The <see cref="TopRightPoint"/> DependencyProperty.
public static readonly DependencyProperty TopRightPointProperty = DependencyProperty.Register("TopRightPoint", typeof(Point), typeof(HtProjektor), new PropertyMetadata(_Coordinate_Changed));
/// The Bottom Left point
public Point BottomLeftPoint
{
get { return (Point)GetValue(BottomLeftPointProperty); }
set { SetValue(BottomLeftPointProperty, value); }
}
/// The <see cref="BottomLeftPoint"/> DependencyProperty.
public static readonly DependencyProperty BottomLeftPointProperty = DependencyProperty.Register("BottomLeftPoint", typeof(Point), typeof(HtProjektor), new PropertyMetadata(_Coordinate_Changed));
/// The Bottom Right point
public Point BottomRightPoint
{
get { return (Point)GetValue(BottomRightPointProperty); }
set { SetValue(BottomRightPointProperty, value); }
}
/// The <see cref="BottomRightPoint"/> DependencyProperty.
public static readonly DependencyProperty BottomRightPointProperty = DependencyProperty.Register("BottomRightPoint", typeof(Point), typeof(HtProjektor), new PropertyMetadata(_Coordinate_Changed));
private static void _Coordinate_Changed(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
Point pointTl = projektor.Simple3Dto2D(projektor._Viewport3D, projektor.pointsTransformed[1]);
Point pointTr = projektor.Simple3Dto2D(projektor._Viewport3D, projektor.pointsTransformed[3]);
Point pointBl = projektor.Simple3Dto2D(projektor._Viewport3D, projektor.pointsTransformed[0]);
Point pointBr = projektor.Simple3Dto2D(projektor._Viewport3D, projektor.pointsTransformed[2]);
projektor.TLX = (int)pointTl.X;
projektor.TLY = (int)pointTl.Y;
projektor.TRX = (int)pointTr.X;
projektor.TRY = (int)pointTr.Y;
projektor.BLX = (int)pointBl.X;
projektor.BLY = (int)pointBl.Y;
projektor.BRX = (int)pointBr.X;
projektor.BRY = (int)pointBr.Y;
}
}
/// <summary>
/// The top left x coordinate
/// </summary>
public int TLX
{
get { return (int)GetValue(TLXProperty); }
set { SetValue(TLXProperty, value); }
}
/// <summary>
/// The <see cref="TLX"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty TLXProperty = DependencyProperty.Register("TLX", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Tl_CoordinateChanged));
/// <summary>
/// The top left y coordinate
/// </summary>
public int TLY
{
get { return (int)GetValue(TLYProperty); }
set { SetValue(TLYProperty, value); }
}
/// <summary>
/// The <see cref="TLY"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty TLYProperty = DependencyProperty.Register("TLY", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Tl_CoordinateChanged));
private static void _Pixel_Tl_CoordinateChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
projektor.pointsTransformed[1] = projektor.Simple2Dto3D(projektor._Viewport3D, new Point(projektor.TLX, projektor.TLY)); //links oben
projektor._MatrixTransform3D.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
projektor._MatrixTransform3D02.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
}
}
/// <summary>
/// The top right X coordinate
/// </summary>
public int TRX
{
get { return (int)GetValue(TRXProperty); }
set { SetValue(TRXProperty, value); }
}
/// <summary>
/// The <see cref="TRX"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty TRXProperty = DependencyProperty.Register("TRX", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Tr_CoordinateChanged));
/// <summary>
/// The top right y coordinate
/// </summary>
public int TRY
{
get { return (int)GetValue(TRYProperty); }
set { SetValue(TRYProperty, value); }
}
/// <summary>
/// The <see cref="TRY"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty TRYProperty = DependencyProperty.Register("TRY", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Tr_CoordinateChanged));
private static void _Pixel_Tr_CoordinateChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
projektor.pointsTransformed[3] = projektor.Simple2Dto3D(projektor._Viewport3D, new Point(projektor.TRX, projektor.TRY)); //rechts oben
projektor._MatrixTransform3D.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
projektor._MatrixTransform3D02.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
}
}
/// <summary>
/// The bottom left x coordinate
/// </summary>
public int BLX
{
get { return (int)GetValue(BLXProperty); }
set { SetValue(BLXProperty, value); }
}
/// <summary>
/// The <see cref="BLX"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty BLXProperty = DependencyProperty.Register("BLX", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Bl_CoordinateChanged));
/// <summary>
/// The bottom left y coordinate
/// </summary>
public int BLY
{
get { return (int)GetValue(BLYProperty); }
set { SetValue(BLYProperty, value); }
}
/// <summary>
/// The <see cref="BLY"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty BLYProperty = DependencyProperty.Register("BLY", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Bl_CoordinateChanged));
private static void _Pixel_Bl_CoordinateChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
projektor.pointsTransformed[0] = projektor.Simple2Dto3D(projektor._Viewport3D, new Point(projektor.BLX, projektor.BLY)); //links unten
projektor._MatrixTransform3D.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
projektor._MatrixTransform3D02.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
}
}
/// <summary>
/// The bottom right x coordinate
/// </summary>
public int BRX
{
get { return (int)GetValue(BRXProperty); }
set { SetValue(BRXProperty, value); }
}
/// <summary>
/// The <see cref="BRX"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty BRXProperty = DependencyProperty.Register("BRX", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Br_CoordinateChanged));
/// <summary>
/// The bottom right y coordinate
/// </summary>
public int BRY
{
get { return (int)GetValue(BRYProperty); }
set { SetValue(BRYProperty, value); }
}
/// <summary>
/// The <see cref="BRY"/> DependencyProperty.
/// </summary>
public static readonly DependencyProperty BRYProperty = DependencyProperty.Register("BRY", typeof(int), typeof(HtProjektor), new PropertyMetadata(0, _Pixel_Br_CoordinateChanged));
private static void _Pixel_Br_CoordinateChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
{
if (dependencyObject is HtProjektor projektor)
{
projektor.pointsTransformed[2] = projektor.Simple2Dto3D(projektor._Viewport3D, new Point(projektor.BRX, projektor.BRY)); //rechts unten
projektor._MatrixTransform3D.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
projektor._MatrixTransform3D02.Matrix = projektor.CalculateNonAffineTransform(projektor.pointsTransformed);
}
}
#endregion
// ##########################################################################################
// Private Properties
// ##########################################################################################
private MeshGeometry3D _MeshGeometry3D;
// private MeshGeometry3D _MeshGeometry3D02;
private MatrixTransform3D _MatrixTransform3D;
private MatrixTransform3D _MatrixTransform3D02;
private Viewport3D _Viewport3D;
#endregion
// ##############################################################################################################################
// Konstruktor
// ##############################################################################################################################
#region Konstruktor
public HtProjektor()
{
DefaultStyleKey = typeof(HtProjektor);
if (!DesignerProperties.GetIsInDesignMode(this))
{
Loaded += _OnLoaded;
}
}
private void _OnLoaded(object sender, RoutedEventArgs routedEventArgs)
{
_MeshGeometry3D = GetTemplateChild("mesh") as MeshGeometry3D;
// _MeshGeometry3D02 = GetTemplateChild("mesh1") as MeshGeometry3D;
_MatrixTransform3D = GetTemplateChild("xform") as MatrixTransform3D;
_MatrixTransform3D02 = GetTemplateChild("xform2") as MatrixTransform3D;
_Viewport3D = GetTemplateChild("viewport3d") as Viewport3D;
for (int i = 0; i < 4; i++)
pointsTransformed[i] = _MatrixTransform3D.Matrix.Transform(_MeshGeometry3D.Positions[i]);
for (int i = 0; i < 4; i++)
pointsTransformed[i] = _MatrixTransform3D02.Matrix.Transform(_MeshGeometry3D.Positions[i]);
TopLeftPoint = new Point(pointsTransformed[1].X, pointsTransformed[1].Y);
TopRightPoint = new Point(pointsTransformed[3].X, pointsTransformed[3].Y);
BottomLeftPoint = new Point(pointsTransformed[0].X, pointsTransformed[0].Y);
BottomRightPoint = new Point(pointsTransformed[2].X, pointsTransformed[2].Y);
ScaleX = ActualWidth / Breite;
ScaleY = ActualHeight / Hoehe;
}
protected override Size ArrangeOverride(Size arrangeBounds)
{
FixToBounds(arrangeBounds.Width, arrangeBounds.Height);
return base.ArrangeOverride(arrangeBounds);
}
#endregion
bool isDragging;
int indexDragging;
Point3D[] pointsTransformed = new Point3D[4];
protected override void OnMouseLeftButtonDown(MouseButtonEventArgs args)
{
Point pt = args.GetPosition(_Viewport3D);
// Obtain the Visual3D objects under the mouse pointer.
HitTestResult result = VisualTreeHelper.HitTest(_Viewport3D, pt);
// Cast result parameter to RayMeshGeometry3DHitTestResult.
RayMeshGeometry3DHitTestResult resultMesh = result as RayMeshGeometry3DHitTestResult;
// This should not happen, but play it safe anyway.
if (resultMesh == null)
return;
// Obtain clicked ModelVisual3D.
ModelVisual3D vis = resultMesh.VisualHit as ModelVisual3D;
// This should not happen, but play it safe anyway.
if (vis == null)
return;
// Determine which vertex the mouse is closest to.
if (resultMesh.VertexWeight1 < resultMesh.VertexWeight2)
{
if (resultMesh.VertexWeight2 < resultMesh.VertexWeight3)
indexDragging = resultMesh.VertexIndex3;
else
indexDragging = resultMesh.VertexIndex2;
}
else if (resultMesh.VertexWeight3 > resultMesh.VertexWeight1)
indexDragging = resultMesh.VertexIndex3;
else
indexDragging = resultMesh.VertexIndex1;
// Get current transformed points.
for (int i = 0; i < 4; i++)
pointsTransformed[i] = _MatrixTransform3D.Matrix.Transform(_MeshGeometry3D.Positions[i]);
// Obtain new transform and commence dragging operation.
pointsTransformed[indexDragging] = Simple2Dto3D(_Viewport3D, pt);
_MatrixTransform3D.Matrix = CalculateNonAffineTransform(pointsTransformed);
_MatrixTransform3D02.Matrix = CalculateNonAffineTransform(pointsTransformed);
isDragging = true;
CaptureMouse();
args.Handled = true;
}
protected override void OnMouseMove(MouseEventArgs args)
{
base.OnMouseMove(args);
if (isDragging)
{
Point ptMouse = args.GetPosition(_Viewport3D);
//if (ptMouse.Y <= 0.0)
// ptMouse.Y = 0.0;
//else if (ptMouse.X <= 0.0)
// ptMouse.X = 0.0;
//if (ptMouse.Y >= ActualHeight)
// ptMouse.Y = ActualHeight;
//else if (ptMouse.X >= ActualWidth)
// ptMouse.X = ActualWidth;
pointsTransformed[indexDragging] = Simple2Dto3D(_Viewport3D, ptMouse);
_MatrixTransform3D.Matrix = CalculateNonAffineTransform(pointsTransformed);
_MatrixTransform3D02.Matrix = CalculateNonAffineTransform(pointsTransformed);
args.Handled = true;
TopLeftPoint = new Point(pointsTransformed[1].X, pointsTransformed[1].Y);
TopRightPoint = new Point(pointsTransformed[3].X, pointsTransformed[3].Y);
BottomLeftPoint = new Point(pointsTransformed[0].X, pointsTransformed[0].Y);
BottomRightPoint = new Point(pointsTransformed[2].X, pointsTransformed[2].Y);
}
CurrentMousePosition = args.GetPosition(_Viewport3D);
}
protected override void OnMouseLeftButtonUp(MouseButtonEventArgs args)
{
base.OnMouseUp(args);
if (isDragging)
{
isDragging = false;
ReleaseMouseCapture();
args.Handled = true;
}
}
private void FixToBounds(double width, double height)
{
if(_Viewport3D?.Camera == null)return;
pointsTransformed[0] = Simple2Dto3D(_Viewport3D, new Point(0.0, height)); //links unten
pointsTransformed[1] = Simple2Dto3D(_Viewport3D, new Point(0.0, 0.0)); //links oben
pointsTransformed[2] = Simple2Dto3D(_Viewport3D, new Point(width, height)); //rechts unten
pointsTransformed[3] = Simple2Dto3D(_Viewport3D, new Point(width, 0.0)); //rechts oben
_MatrixTransform3D.Matrix = CalculateNonAffineTransform(pointsTransformed);
_MatrixTransform3D02.Matrix = CalculateNonAffineTransform(pointsTransformed);
#region Private Methods
// The input array of points describes a 2D rectangle
// (with Z assumed to be zero) in the order
// lower-left, upper-left, lower-right, upper-right.
// The returned transform maps the points (0, 0, 0),
// (0, 1, 0), (1, 0, 0), and (1, 1, 0) to these points.
Matrix3D CalculateNonAffineTransform(Point3D[] points)
{
// Affine transform
// ----------------
// This matrix maps (0, 0) --> (x0, y0)
// (0, 1) --> (x1, y1)
// (1, 0) --> (x2, y2)
// (1, 1) --> (x2 + x1 + x0, y2 + y1 + y0)
Matrix3D A = new Matrix3D();
A.M11 = points[2].X - points[0].X;
A.M12 = points[2].Y - points[0].Y;
A.M21 = points[1].X - points[0].X;
A.M22 = points[1].Y - points[0].Y;
A.OffsetX = points[0].X;
A.OffsetY = points[0].Y;
// Calculate point (a, b) that get mapped by the affine transform to (x3, y3)
double den = A.M11 * A.M22 - A.M12 * A.M21;
double a = (A.M22 * points[3].X - A.M21 * points[3].Y +
A.M21 * A.OffsetY - A.M22 * A.OffsetX) / den;
double b = (A.M11 * points[3].Y - A.M12 * points[3].X +
A.M12 * A.OffsetX - A.M11 * A.OffsetY) / den;
// Non-affine transform
// --------------------
// This matrix maps (0, 0) --> (0, 0)
// (0, 1) --> (0, 1)
// (1, 0) --> (1, 0)
// (1, 1) --> (a, b)
Matrix3D B = new Matrix3D();
B.M11 = a / (a + b - 1);
B.M22 = b / (a + b - 1);
B.M14 = B.M11 - 1;
B.M24 = B.M22 - 1;
return B * A;
}
// The following two methods only work with OrthographicCamera,
// with LookDirection of (0, 0, -1) and UpDirection of (0, 1, 0).
// More advanced conversion routines can be found in the
// Petzold.Media3D library.
// Converts a 2D point in device-independent coordinates relative
// to Viewport3D to 3D space.
Point3D Simple2Dto3D(Viewport3D vp, Point pt)
{
OrthographicCamera cam = CheckRestrictions(vp);
double scale = cam.Width / vp.ActualWidth;
double x = scale * (pt.X - vp.ActualWidth / 2) + cam.Position.X;
double y = scale * (vp.ActualHeight / 2 - pt.Y) + cam.Position.Y;
return new Point3D(x, y, 0);
}
// Converts a 3D point to 2D in device-independent coordinates
// relative to Viewport3D.
Point Simple3Dto2D(Viewport3D vp, Point3D point)
{
OrthographicCamera cam = CheckRestrictions(vp);
double scale = vp.ActualWidth / cam.Width;
// double scale2 = vp.ActualHeight / cam.Width;
double x = vp.ActualWidth / 2 + scale * (point.X - cam.Position.X);
double y = vp.ActualHeight / 2 - scale * (point.Y - cam.Position.Y);
return new Point(x, y);
}
OrthographicCamera CheckRestrictions(Viewport3D vp)
{
OrthographicCamera cam = vp.Camera as OrthographicCamera;
if (cam == null)
throw new ArgumentException("Camera must be OrthographicCamera");
if (cam.LookDirection != new Vector3D(0, 0, -1))
throw new ArgumentException("Camera LookDirection must be (0, 0, -1)");
if (cam.UpDirection != new Vector3D(0, 1, 0))
throw new ArgumentException("Camera UpDirection must be (0, 1, 0)");
return cam;
}
#endregion
}
}