17

ScrollVieweraのプロパティをListBoxC#から変更したいのですが。

この質問はStackoverflowで見つかりました。私は受け入れられた答えのアドバイスを受け取りScrollViewer、サブクラスのプロパティとしてを公開しました。ただし、以下に示す例では、これは機能していないようです。その質問のコメントのいくつかは、このテクニックが機能しなかったとも述べています。

XAML:

<Window x:Class="StackoverflowListBoxScrollViewer.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525">

</Window>

C#:

using System;
using System.Windows;
using System.Windows.Controls;

namespace StackoverflowListBoxScrollViewer
{
    public class MyListBox : ListBox
    {
        public ScrollViewer ScrollViewer
        { get { return (ScrollViewer)GetTemplateChild("ScrollViewer"); } }
    }

    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            var myListBox = new MyListBox();

            Content = myListBox;

            myListBox.Items.Add(new Button() { Content = "abc" });
            myListBox.Items.Add(new Button() { Content = "abc" });
            myListBox.Items.Add(new Button() { Content = "abc" });
            myListBox.Items.Add(new Button() { Content = "abc" });
            myListBox.Items.Add(new Button() { Content = "abc" });

            var button = new Button() { Content = "Check ScrollViewer" };
            button.Click += (s, e) =>
                {
                    if (myListBox.ScrollViewer == null)
                        Console.WriteLine("null");
                };
            myListBox.Items.Add(button);
        }
    }
}

「ScrollViewerの確認」ボタンをクリックすると、「null」と表示されます。つまり、ScrollViewerは取得されませんでした。

どうすればそのくそにたどり着くことができScrollViewerますか?:-)

4

6 に答える 6

27

この小さなヘルパー関数を試すことができます

利用方法

var scrollViewer = GetDescendantByType(yourListBox, typeof(ScrollViewer)) as ScrollViewer;

ヘルパー機能

public static Visual GetDescendantByType(Visual element, Type type)
{
  if (element == null) {
    return null;
  }
  if (element.GetType() == type) {
    return element;
  }
  Visual foundElement = null;
  if (element is FrameworkElement) {
    (element as FrameworkElement).ApplyTemplate();
  }
  for (int i = 0; i < VisualTreeHelper.GetChildrenCount(element); i++) {
    Visual visual = VisualTreeHelper.GetChild(element, i) as Visual;
    foundElement = GetDescendantByType(visual, type);
    if (foundElement != null) {
      break;
    }
  }
  return foundElement;
}

お役に立てれば

于 2012-04-24T07:31:14.573 に答える
12

標準のリストボックスを使用する場合は、ゲッターを次のリストに変更できます。

public class MyListBox : ListBox
{
    public ScrollViewer ScrollViewer
    {
        get 
        {
            Border border = (Border)VisualTreeHelper.GetChild(this, 0);

            return (ScrollViewer)VisualTreeHelper.GetChild(border, 0);
        }
    }
}
于 2012-04-24T07:34:17.483 に答える
9

@ punker76のすばらしい答えを変更して、Visualの拡張メソッドを作成し、明示的な戻り型を提供しました。

   public static class Extensions
   {
      public static T GetDescendantByType<T>(this Visual element) where T:class
      {
         if (element == null)
         {
            return default(T);
         }
         if (element.GetType() == typeof(T))
         {
            return element as T;
         }
         T foundElement = null;
         if (element is FrameworkElement)
         {
            (element as FrameworkElement).ApplyTemplate();
         }
         for (var i = 0; i < VisualTreeHelper.GetChildrenCount(element); i++)
         {
            var visual = VisualTreeHelper.GetChild(element, i) as Visual;
            foundElement = visual.GetDescendantByType<T>();
            if (foundElement != null)
            {
               break;
            }
         }
         return foundElement;
      }

   }

これで、SomeVisual.GetDescendantByTypeで呼び出すことができ、すでに正しい型のScrollViewerまたはnull(デフォルト(T))のいずれかが返されます。

于 2013-07-25T06:45:46.983 に答える
1

私の場合、ScrollViewerをプロパティとして公開することは悪い考えです。まず、ScrollViewerがテンプレートに存在するという保証はありません。次に、ScrollViewerはItemsPanelおよびItemContainerGeneratorと同期して動作します。これをオーバーライドすることは、珍しい動作への直接的な方法です。

WPFコントロールは別のパターンを使用します。それらのクラスは、外部の論理的使用法と内部の視覚的表現の間の仲介者のようなものです。ListBoxは、テンプレート内のScrollViewerで使用できるプロパティを公開する必要がありますが、ScrollViewerでは使用できません。これを行うことで、WPF標準に違反し、制御を特定のテンプレートに制限し、ユーザーコードが内部のListBox実装をハッキングできるようにします。

于 2012-04-24T07:58:57.513 に答える
1

ScrollViewerのプロパティは、リストボックスに「アタッチ」されています(https://docs.microsoft.com/en-us/dotnet/framework/wpf/advanced/attached-properties-overviewを参照)。次の関数を使用して、依存関係プロパティとして取得または設定できます。

public object GetValue (System.Windows.DependencyProperty dp);

public void SetValue (System.Windows.DependencyProperty dp, object value);

たとえば、リストボックス'lb'の場合、次のように記述できます。

lb.GetValue(ScrollViewer.ActualHeightProperty); また lb.SetValue(ScrollViewer.HorizontalScrollBarVisibilityProperty, ScrollBarVisibility.Visible);

于 2019-09-24T12:36:08.133 に答える
0

これは、C#6に対する@punker76の回答の別の作り直された一般的なバージョンです。

public static class VisualExtensions
{
    public static T FindVisualDescendant<T>(this Visual element) where T : Visual
    {
        if (element == null)
            return null;

        var e = element as T;

        if (e != null)
            return e;

        (element as FrameworkElement)?.ApplyTemplate();

        var childrenCount = VisualTreeHelper.GetChildrenCount(element);

        for (var i = 0; i < childrenCount; i++)
        {
            var visual = VisualTreeHelper.GetChild(element, i) as Visual;

            var foundElement = visual.FindVisualDescendant<T>();

            if (foundElement != null)
                return foundElement;
        }

        return null;
    }
}
于 2017-02-02T22:01:24.007 に答える