你可以這樣做: 從可視化樹中刪除按鈕並將其放置在裝飾器上。 裝飾者關閉後再次將其附加到原始父級。 我認爲這比修剪任何幾何圖形更靈活(例如,您可以在裝飾器上放置像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