【发布时间】:2016-06-30 14:27:18
【问题描述】:
我正在使用 GreatMaps.NET 库 (https://greatmaps.codeplex.com/) 的面向 MVVM 的分支,但我的问题归结为一个基本问题,即使用自定义 Canvas 作为 ItemsPanel 的 ItemsControl。本质上,地图是一个 ItemsControl,其中一个 MapCanvas 将其子项放置如下:
protected override Size ArrangeOverride(Size arrangeSize)
{
foreach (UIElement child in Children)
{
PointLatLng position = new PointLatLng(GetTop(child), GetLeft(child));
GMapControl map = Owner as GMapControl;
if (map != null)
{
GPoint p = map.FromLatLngToLocal(position);
p.Offset(-(long)(map.MapTranslateTransform.X + child.DesiredSize.Width * 0.5), -(long)(map.MapTranslateTransform.Y + child.DesiredSize.Height * 0.5));
Rect rect = new Rect(p.X, p.Y, child.DesiredSize.Width, child.DesiredSize.Height);
child.Arrange(rect);
}
}
return arrangeSize;
}
之所以有效,是因为我的 ItemContainerStyle 为每个地图项绑定到 ViewModel 中的纬度和经度,如下所示(连同 ZIndex,我已将其设置为 99 作为占位符):
<Setter Property="Canvas.Left" Value="{Binding Longitude}" />
<Setter Property="Canvas.Top" Value="{Binding Latitude}" />
<Setter Property="Canvas.ZIndex" Value="{Binding ZIndex}" />
我正在使用带有 DataType 字段的 DataTemplates 来调整每个元素的显示方式,并且该部分工作正常。
有些元素(例如地图上的路线线)我不需要作为点元素,而是作为子元素序列。为此,我只是按照模式制作了一个 DataTemplate,在其中我使用另一个 MapCanvas 绑定到与其所有者相同的 MapControl(这里,MapOverlay 只是 MapCanvas 的一个副本,它覆盖 OnRender 以在其子项之间绘制线条):
<DataTemplate DataType="{x:Type viewModels:RouteViewModel}">
<ItemsControl ItemsSource="{Binding Locations}">
<ItemsControl.ItemsPanel>
<ItemsPanelTemplate>
<wpf:MapOverlay Name="MapOverlay" Owner="{Binding Path=., RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type wpf:GMapControl}}}"/>
</ItemsPanelTemplate>
</ItemsControl.ItemsPanel>
<ItemsControl.ItemContainerStyle>
<Style>
<Setter Property="Canvas.Left" Value="{Binding Longitude}" />
<Setter Property="Canvas.Top" Value="{Binding Latitude}" />
<Setter Property="Canvas.ZIndex" Value="{Binding ZIndex}" />
</Style>
</ItemsControl.ItemContainerStyle>
<ItemsControl.ItemTemplate>
<DataTemplate>
<Ellipse Fill="Blue" Stroke="Black" Width="10" Height="10"></Ellipse>
</DataTemplate>
</ItemsControl.ItemTemplate>
</ItemsControl>
</DataTemplate>
为了保持绑定正常工作,RouteViewModel 仍然有自己的 Lat/Lng,这似乎仅在我将它们定位在沿路线的点列表的初始点时才有效:
public double Latitude => Locations.Select(loc => loc.Latitude).FirstOrDefault();
public double Longitude => Locations.Select(loc => loc.Longitude).FirstOrDefault();
总结:
- 我有一个 GMapControl ItemsControl,它使用 MapCanvas 对元素进行地理空间定位。 DataTypes 用于选择每个元素使用哪个 DataTemplate。这适用于标记等基于点的元素。
- 其中一个 DataTemplates 是 ItemsControl,它使用另一个 MapCanvas 变体 (MapOverlay) 作为其 ItemsPanel,以在子元素之间绘制线条。这在启动时起作用,当它的 ArrangeOverride 执行时。
- 虽然在 MapOverlay 中没有调用 ArrangeOverride,但可以平移地图。
问题:虽然我的元素最初显示正确定位,但缩放地图(通过地图控件触发 InvalidateVisual 和 UpdateLayout)不会导致在嵌套 MapOverlay 上调用 ArrangeOverride。这会导致路线不再正确定位 - 它无法缩放。
关于为什么排列失效没有渗透到嵌套的 MapOverlay 和/或有关如何修复它的提示的任何建议?
附录: 仅当地图最初位于其第一个元素时,路线才被正确放置 - 整个路线的纬度/经度是一个单独的问题,我欢迎提出意见.
【问题讨论】:
-
如果您有兴趣,我可以告诉您如何使用我的XAML MapControl 轻松显示路线(即与线相连的圆圈)。当然是 MVVM。
-
我已经看到您提到的项目——这将非常有帮助!谢谢!
-
你如何缩放?我会在 RenderTransform 中使用 TransformGroup 作为地图的根,它具有控制缩放和平移的 ScaleTransform 和 TranslateTransform。就像在渲染过程中完成的那样,它根本不会影响布局。此外,您可以使用来自不同来源的许多缩放控件来控制该方面,而无需担心地图中的布局。
标签: wpf canvas mvvm itemscontrol gmap.net