你可以这样做: 从可视化树中删除按钮并将其放置在装饰器上。 装饰者关闭后再次将其附加到原始父级。 我认为这比修剪任何几何图形更灵活(例如,您可以在装饰器上放置像usercontrols这样的复杂内容)
以下示例使用Panel作为按钮的容器。
XAML中(窗口):
<Grid Margin="50" x:Name="myGrid" Background="LightBlue">
<Button x:Name="myButton" Width="80" Height="30" Click="myButton_Click">Show popup</Button>
代码背后: 私人FrameworkElementAdorner _adorner;私人面板__ __ __ __ __ __ __ __ __ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
private void myButton_Click(object sender, RoutedEventArgs e)
{
if (_adorner == null)
{
_adorner = new FrameworkElementAdorner(myGrid);
}
// remove the button from the parent panel and attach it to the adorner
// otherwise remove from adorner and attach to original parent again
if (_adorner.IsVisible)
{
AdornerLayer adornerLayer = AdornerLayer.GetAdornerLayer(myGrid);
adornerLayer.Remove(_adorner);
Panel parent = VisualTreeHelper.GetParent(myButton) as Panel;
if (parent != null)
{
parent.Children.Remove(myButton);
}
_originalParent.Children.Add(myButton);
}
else
{
_originalParent = VisualTreeHelper.GetParent(myButton) as Panel;
if (_originalParent != null)
{
_originalParent.Children.Remove(myButton);
}
// Create the Adorner with the original button in it
_adorner.Child = CreateAdornerContent(myButton);
AdornerLayer adornerLayer = AdornerLayer.GetAdornerLayer(myGrid);
adornerLayer.Add(_adorner);
}
}
/// <summary>
/// Creates some dummy content for the adorner
/// </summary>
private FrameworkElement CreateAdornerContent(Button myButton)
{
Grid g = new Grid();
g.Background = new SolidColorBrush(Colors.Yellow);
TextBlock tb = new TextBlock();
tb.Text = "I am the Adorner";
g.Children.Add(tb);
g.Children.Add(myButton);
return g;
}
在这里,简单的装饰器刚刚显示FrameworkElement的: 类FrameworkElementAdorner:装饰器 { 私人FrameworkElement的_child;
public FrameworkElementAdorner(UIElement adornedElement)
: base(adornedElement)
{
}
protected override int VisualChildrenCount
{
get
{
return 1;
}
}
protected override Visual GetVisualChild(int index)
{
if (index != 0) throw new ArgumentOutOfRangeException();
return _child;
}
public FrameworkElement Child
{
get
{
return _child;
}
set
{
if (_child != null)
{
RemoveVisualChild(_child);
}
_child = value;
if (_child != null)
{
AddVisualChild(_child);
}
}
}
protected override Size ArrangeOverride(Size finalSize)
{
_child.Arrange(new Rect(new Point(0, 0), finalSize));
return new Size(_child.ActualWidth, _child.ActualHeight);
}
}
我也可以,如果你喜欢上载满SLN。这在某种程度上可能吗?
您是否试图创建一个弹出窗口,以便灰色层不覆盖它?如果是这样,您可以通过使用Z顺序来控制组件的显示顺序: http://blogs.msdn.com/b/wpfsdk/archive/2006/06/13/controlling-zorder-using-the- zindex-property.aspx – stuartmclark
我会用Adorner/AdornerLayer或Popup控件。这两个元素都旨在显示其他控件之上的信息,并且应该覆盖您需要在视觉内容上方显示一个Popup而不需要修剪任何内容。 请参阅 http://msdn.microsoft.com/en-us/library/ms749018.aspx 和 http://msdn.microsoft.com/en-us/library/ms747117.aspx – SvenG
你们两个都是对的,对于弹出窗口来说,使用Z-Index或控件已经符合我的需要会容易得多。我会尝试使用其中一种可能性。但是,源代码控制呢?被点击的按钮也不应该被弹出式覆盖图覆盖。 – MatthiasG