2

Adorner是在Image一部分上定义的。所需的行为如下:

  1. 当鼠标悬停在图像区域(包括装饰器区域)上时,会出现装饰器。
  2. 当鼠标离开图像和装饰器区域时,装饰器消失。
  3. 装饰器的出现和消失是通过淡入/淡出动画来实现的。
  4. 点击 Adorner 区域必须引发事件 AdornerClicked
  5. 单击装饰器未隐藏的图像上的区域,必须提升 ImageClicked。

一个天真的实现

在图像的 MouseEnter 和 MouseLeave 事件的 Adorner 不透明度上附加动画,并为每个事件附加 Click 事件。然而,当鼠标直接位于装饰器上方时,这会导致装饰器消失(因为在下图中触发了 MouseLeave),违反了要求 1。

对简单实现的一个可能的修改是在 Adorner 上设置 IsHitTestVisible=false。但是,装饰器没有捕获任何点击,这违反了要求 4。

满足要求的正确模式是什么?

4

1 回答 1

0

一个有点老的问题,但我刚刚遇到了同样的问题,找不到答案,所以这就是我想出的。

所以问题是控件及其装饰器重叠并且将装饰器设置为可见会触发装饰控件上的 MouseLeave,因为它现在被装饰器覆盖。

解决方案是对装饰控件及其装饰器上的每个 MouseEnter 和 MouseLeave 做出反应,并手动进行命中测试。如果它们中的任何一个被击中,那么装饰器应该是可见的,否则会折叠。

因此,您需要能够从装饰控件中获取装饰器,反之亦然。从装饰器获取装饰控件没有问题(使用 AdornedElement 属性),但框架(AFAIK)不提供控件的装饰器,因此我使用字典将我的控件映射到其装饰器列表。

这是我的 Panel 派生类中的代码(包含并排列我的控件及其装饰器):

private readonly Dictionary<Control, List<Adorner>> _controlToAdornersMap;

...

private void CreateMyControl()
{
    var control = new MyControl();
    control.MouseEnter += OnMyControlMouseEnterOrLeave;
    control.MouseLeave += OnMyControlMouseEnterOrLeave;
    Children.Add(control);
    AddAdorners(control);
}

private void AddAdorners(Control control)
{
    var myAdorner = new MyAdorner(control);
    myAdorner.MouseEnter += OnMyAdornerMouseEnterOrLeave;
    myAdorner.MouseLeave += OnMyAdornerMouseEnterOrLeave;

    var adornerLayer = AdornerLayer.GetAdornerLayer(control);
    adornerLayer.Add(myAdorner);

    _controlToAdornersMap[control] = new List<Adorner> {myAdorner};
}

private void OnMyControlMouseEnterOrLeave(object sender, MouseEventArgs e)
{
    HitTestAndSetAdornersVisibility((MyControl)sender, e);
}

private void OnMyAdornerMouseEnterOrLeave(object sender, MouseEventArgs e)
{
    var adorner = (Adorner)sender;
    HitTestAndSetAdornersVisibility((MyControl)adorner.AdornedElement, e);
}

private void HitTestAndSetAdornersVisibility(MyControl control, MouseEventArgs e)
{
    var adorners = _controlToAdornersMap[control];
    var hitTestSubjects = new List<UIElement> { control }.Concat(adorners);
    var hit = hitTestSubjects.Any(i => VisualTreeHelper.HitTest(i, e.GetPosition(i)) != null);

    SetAdornersVisibility(adorners, hit ? Visibility.Visible : Visibility.Collapsed);
}

private static void SetAdornersVisibility(IEnumerable<Adorner> adorners, Visibility visibility)
{
    if (adorners != null)
        foreach (var adorner in adorners)
            adorner.Visibility = visibility;
}
于 2015-12-05T12:03:12.673 回答