- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
首先,我对标题的措辞感到抱歉;我真的不确定如何简洁地陈述我的困惑而不听起来很煽动。
我对使用 WPF 制作 UI 还很陌生(两天)。到目前为止,我绝对喜欢它的大部分内容。但是,在 Canvas 上绘图对我来说似乎很不直观。我希望这里有人可以解释 a) 为什么这样更好,或者 b) 我只是用错了。
我在Java的Graphics2D Canvas 上学习了计算机绘图,类似于GDI+。要在空间中的特定坐标处绘制一个矩形,我调用类似 canvas.drawRectangle(x, y, w, h, color)
的方法。然而,对于 WPF,我注意到 Rectangle 对象在它们的构造函数中没有参数,这意味着它需要更多的线来绘制一个:
Rectangle rect = new Rectangle();
rect.Width = w;
rect.Height = h;
rect.Color = Colors.Black;
... // a whole line for each parameter
Canvas.SetLeft(rect, x);
Canvas.Setleft(rect, y);
c.Children.Add(rect);
它似乎不必要地冗长,这让我觉得我错过了一些东西(好处或捷径)。我了解 WPF 相对于 GDI+ 的优势以及为什么有些东西需要以某种方式构建,但我希望能够做这样的事情:
Rectangle rect = new Rectangle(w, h, Colors.Black);
c.AddChild(rect, x, y);
很抱歉这篇文章很长,这不是标准的“我该怎么做_?”所以问题。我真的希望这不会让人觉得发牢骚;我只是想学习一门新技术,没想到会有如此不同。
最佳答案
在不判断数据绑定(bind)是否适合您的情况下,肯定有可能:
// Shapes.cs
public class Shape
{
public double X { get; set; }
public double Y { get; set; }
public double W { get; set; }
public double H { get; set; }
}
public class Rectangle : Shape { }
public class Circle : Shape { }
public class Triangle : Shape
{
public double Angle { get; set; }
}
<!-- Window.xaml -->
<!-- Itemscontrol will take each item in a bound collection (in this case, shapes) and create a visual based off of data templates -->
<ItemsControl ItemsSource="{Binding shapes}">
<ItemsControl.Resources>
<!-- data templates let us control the visual representation of a given class -->
<!-- DataType=Rectangle causes the contents of this datatemplate to be shown for -->
<!-- each instance of a rectangle found in the shapes collection. -->
<!-- Since we do not specify a size, the contents will fill to their parent, which -->
<!-- we are constraining in the ItemsContainerStyle -->
<DataTemplate DataType="{x:Type local:Rectangle}">
<Rectangle Fill="Red" />
</DataTemplate>
<DataTemplate DataType="{x:Type local:Circle}">
<Ellipse Fill="Blue" />
</DataTemplate>
<DataTemplate DataType="{x:Type local:Triangle}">
<!-- Since we are specifying a specific geometry for the triangle, we want it -->
<!-- to scale to the dimensions supplied. Viewbox does that for us. -->
<Viewbox>
<Polygon Points="0,0 80,50, 0,100" Stroke="Black" Fill="Black">
<Polygon.LayoutTransform>
<!-- We also transform the coordinate space by rotating it by the -->
<!-- angle specified in the Triangle instance -->
<RotateTransform Angle="{Binding Angle}" />
</Polygon.LayoutTransform>
</Polygon>
</Viewbox>
</DataTemplate>
</ItemsControl.Resources>
<!-- Here we are swapping out the standard stack-panel for a canvas -->
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<Canvas />
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
<!-- And we set the positioning of the items within the itemscontrol. -->
<!-- These must be set via style, rather than directly, since the items control -->
<!-- wraps each item in a ContentPresenter (ItemsContainerStyle.Template) -->
<ItemsControl.ItemContainerStyle>
<Style TargetType="FrameworkElement">
<Setter Property="Canvas.Left" Value="{Binding X}"/>
<Setter Property="Canvas.Top" Value="{Binding Y}"/>
<Setter Property="Width" Value="{Binding W}" />
<Setter Property="Height" Value="{Binding H}" />
</Style>
</ItemsControl.ItemContainerStyle>
</ItemsControl>
// Window.xaml.cs
public MainWindow()
{
shapes = new ObservableCollection<Shape>();
InitializeComponent();
this.DataContext = this;
shapes.Add(new Rectangle() { X = 50, Y = 43, H = 15, W = 59 });
shapes.Add(new Rectangle() { X = 67, Y = 43, H = 20, W = 30 });
shapes.Add(new Circle() { X = 50, Y = 43, H = 15, W = 59 });
shapes.Add(new Triangle() { X = 100, Y = 100, H = 40, W = 40, Angle = 20 });
}
public ObservableCollection<Shape> shapes { get; set; }
产出:
好处是基于矢量的、可数据绑定(bind)的、可样式化的模板优点。有关数据绑定(bind)的更多信息,see MSDN .数据模板是 explained here.
如果你只是想像 GDI 一样绘制到位图表面,直接使用 RenderContext
(the analog to Graphics
) , 你可以 implement your own FrameworkElement与您覆盖以前的 OnPaint
方法的方式大致相同:
class DrawingElement : FrameworkElement
{
private List<Shape> shapes;
private DrawingVisual visual;
public DrawingElement(IEnumerable<Shape> shapesToDraw)
: base()
{
this.shapes = new List<Shape>(shapesToDraw);
this.visual = new DrawingVisual();
this.AddVisualChild(visual);
render();
}
private void render()
{
var dc = this.visual.RenderOpen();
foreach (var shape in shapes)
{
if (shape is Rectangle)
{
var rect = (Rectangle)shape;
dc.DrawRectangle(Brushes.Red, new Pen(Brushes.Black, 1), new Rect(rect.X, rect.Y, rect.W, rect.H));
}
else if (shape is Circle)
{
var rect = (Circle)shape;
dc.DrawEllipse(Brushes.Blue, new Pen(Brushes.Black, 1),
new Point(rect.X + rect.W / 2, rect.Y + rect.H / 2),
rect.W / 2, rect.H / 2);
}
else if (shape is Triangle)
{
// trig goes here
}
}
dc.Close();
}
protected override Visual GetVisualChild(int index)
{
if (index != 0)
throw new ArgumentOutOfRangeException();
return visual;
}
protected override int VisualChildrenCount { get { return 1; } }
}
关于c# - 我发现 WPF 的 Canvas 绘图设计得很奇怪。为什么它更喜欢无参数构造函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21128179/
有谁知道蓝牙设备如何获取范围内可发现设备的设备 ID? 理想情况下,我正在寻找涉及蓝牙协议(protocol)最小实现的最简单解决方案。 一个起点会很好,我只是想创建一个设备,它可以以最小的功耗存储附
我有用于搜索Kibana中特定事件的查询,该查询与其他搜索一起保存,是否可以通过REST调用以编程方式更改它? 最佳答案 正如@Mohammad所说,所有与Kibana相关的元数据都存储在elasti
我正在使用带有这些注释的基本集成测试: @ExtendWith(SpringExtension::class) @SpringBootTest(classes = [SomeApplication::
以下是我的代码 HTML: Hello !! Javascript: $(function() { $('#content .child').click(function() {
我试图避免在每个 GDB session 中输入相同的命令。为此,我遵循了 rust discovery book 中的说明。但是当我通过 cargo run 运行程序时,程序没有像书中提到的那样工作
好的,我记得有一些命令可以放在 settings.py 文件中,这样基本上当您将 django 项目移动到另一个目录时,它就不会启动 foo-bar . 我知道我可以通过在它提到主目录的任何地方设置一
假设我正在制作一份注册表单。现在我希望它突出显示四个字段中的空白字段。现在我可以只执行一堆 if-else 语句,但这将花费很长时间。 假设我有以下代码: Javascript: if($firstn
我试图理解 C++ 中 regex 的逻辑 std::string s ("Ni Ni Ni NI"); std::regex e ("(Ni)"); std::smatch sm; std::re
运行时: vim /tmp/blah :q echo $? 我的退出状态为 1 .这破坏了包括 Git 在内的各种东西。如果我在没有 vimrc 的情况下运行 vim: vim -u NONE /tm
我无法通过“查找”来查找旧文件。我将我的发现链接到一个声明中,所有其他部分都运行良好。这是我所拥有的精简版。它搜索 $path 的目录树,并为找到的每个项目创建仅包含换行符的单独临时文件:所有文件、超
我已经多次看到这个问题,但没有一个答案对我有用。 我的 DotNet Core 应用程序失败 编码: public static void Main(string[] args) {
已解决见编辑 2 你好, 我一直在编写一个 Perl 程序来处理本地(专有)程序的自动升级(对于我工作的公司)。 基本上,它通过 cron 运行,不幸的是有内存泄漏(或类似的东西)。问题是泄漏只发生在
在 icCube 中创建到 Oracle 数据库的连接时,“选择现有数据库表”返回一个空的表列表。 连接很好,我可以查询模式创建 SQL 查询。 最佳答案 用户用作模式名称,但 Oracle 使用大写
我正在使用 VBA 循环遍历两个工作表上的行,如果它们匹配,则将工作表 2 中的行复制到工作表 1 中。 我的代码应该: 打开第二个工作簿 将所有信息复制到新工作表上的原始工作簿中 然后循环遍历原始工
当我尝试同步我的数据库时出现这个奇怪的错误: Unhandled rejection Error: Cyclic dependency found. roles is dependent of its
以编程方式发现 perl 模块具有的所有子例程的最佳方法是什么?这可以是一个模块、一个类(没有@EXPORT)或任何介于两者之间的东西。 编辑:下面的所有方法看起来都可以工作。我可能会在生产中使用 C
如何在“讨论”按钮左侧添加“共享”按钮。我希望该按钮与当前的“讨论”按钮具有相同的样式/颜色。 我从https://atmospherejs.com/joshowens/shareit添加了包 我将
我最近从 Netbeans 切换到 Eclipse,Eclipse 在我的项目中发现了许多 Netbeans 没有的语法错误,我不知道为什么。据可以看出,两个 IDE 都设置为使用 java 1.6。
我必须为我的项目设置一些不接受错误网址的规则。我为此使用正则表达式。 我的网址是“http://some/resource/location”。 此网址的开头、中间或结尾不应留有空格。 例如,这些空格
问题:鉴于作为 VMSS 的一部分启动的 N 个实例,我希望每个 Azure 实例上的应用程序代码能够发现其他对等实例的 IP 地址。我该怎么做? 总体目的是对实例进行集群,以提供主动被动 HA 或保
我是一名优秀的程序员,十分优秀!