Как сделать WPF ScrollViewer прокруткой по нажатию средней кнопки? - PullRequest
12 голосов
/ 25 марта 2011

Если щелкнуть среднюю кнопку мыши (она же колесико мыши), а затем слегка переместить мышь, пользователи смогут прокручивать IE и большинство приложений Windows. Такое поведение отсутствует в элементах управления WPF по умолчанию? Есть ли какие-то настройки, обходные пути или что-то очевидное, чего мне не хватает?

Ответы [ 2 ]

12 голосов
/ 27 марта 2011

Я нашел, как этого добиться, используя 3 события мыши (MouseDown, MouseUp, MouseMove).Их обработчики прикреплены к элементу ScrollViewer в xaml ниже:

<Grid>
    <ScrollViewer MouseDown="ScrollViewer_MouseDown" MouseUp="ScrollViewer_MouseUp" MouseMove="ScrollViewer_MouseMove">
            <StackPanel x:Name="dynamicLongStackPanel">

            </StackPanel>
    </ScrollViewer>
    <Canvas x:Name="topLayer" IsHitTestVisible="False" />
</Grid>

Было бы лучше написать поведение вместо событий в коде позади, но не у всех есть необходимая библиотека, а такжеЯ не знаю, как связать его с Canvas.

Обработчики событий:

    private bool isMoving = false;                  //False - ignore mouse movements and don't scroll
    private bool isDeferredMovingStarted = false;   //True - Mouse down -> Mouse up without moving -> Move; False - Mouse down -> Move
    private Point? startPosition = null;
    private double slowdown = 200;                  //The number 200 is found from experiments, it should be corrected



    private void ScrollViewer_MouseDown(object sender, MouseButtonEventArgs e)
    {
        if (this.isMoving == true) //Moving with a released wheel and pressing a button
                this.CancelScrolling();
        else if (e.ChangedButton == MouseButton.Middle && e.ButtonState == MouseButtonState.Pressed)
        {
            if (this.isMoving == false) //Pressing a wheel the first time
            {
                this.isMoving = true;
                this.startPosition = e.GetPosition(sender as IInputElement);
                this.isDeferredMovingStarted = true; //the default value is true until the opposite value is set

                this.AddScrollSign(e.GetPosition(this.topLayer).X, e.GetPosition(this.topLayer).Y);
            }
        }
    }

    private void ScrollViewer_MouseUp(object sender, MouseButtonEventArgs e)
    {
        if (e.ChangedButton == MouseButton.Middle && e.ButtonState == MouseButtonState.Released && this.isDeferredMovingStarted != true)
            this.CancelScrolling();
    }

    private void CancelScrolling()
    {
        this.isMoving = false;
        this.startPosition = null;
        this.isDeferredMovingStarted = false;
        this.RemoveScrollSign();
    }

    private void ScrollViewer_MouseMove(object sender, MouseEventArgs e)
    {
        var sv = sender as ScrollViewer;

        if (this.isMoving && sv != null)
        {
            this.isDeferredMovingStarted = false; //standard scrolling (Mouse down -> Move)

            var currentPosition = e.GetPosition(sv);
            var offset = currentPosition - startPosition.Value;
            offset.Y /= slowdown;
            offset.X /= slowdown;

            //if(Math.Abs(offset.Y) > 25.0/slowdown)  //Some kind of a dead space, uncomment if it is neccessary
            sv.ScrollToVerticalOffset(sv.VerticalOffset + offset.Y);
            sv.ScrollToHorizontalOffset(sv.HorizontalOffset + offset.X);
        }
    }

Если удалить вызовы методов AddScrollSign и RemoveScrollSign, этот пример будетРабота.Но я расширил его двумя способами, которые устанавливают значок прокрутки:

    private void AddScrollSign(double x, double y)
    {
        int size = 50;
        var img = new BitmapImage(new Uri(@"d:\middle_button_scroll.png"));
        var adorner = new Image() { Source = img, Width = size, Height = size };
        //var adorner = new Ellipse { Stroke = Brushes.Red, StrokeThickness = 2.0, Width = 20, Height = 20 };

        this.topLayer.Children.Add(adorner);
        Canvas.SetLeft(adorner, x - size / 2);
        Canvas.SetTop(adorner, y - size / 2);
    }

    private void RemoveScrollSign()
    {
        this.topLayer.Children.Clear();
    }

Пример значков: enter image description hereenter image description here

И еще одно замечание: есть некоторые проблемы с способомPress -> Immediately Release -> Move.Предполагается отменить прокрутку, если пользователь нажимает левую кнопку мыши или любую клавишу клавиатуры или приложение теряет фокус.Есть много событий, и у меня нет времени обрабатывать их все.

Но стандартный способ Press -> Move -> Release работает без проблем.

2 голосов
/ 28 марта 2011

vorrtex опубликовал хорошее решение, , пожалуйста, сообщите ему !

У меня есть некоторые предложения по его решению, которые слишком длинны, чтобы вместить их всех вкомментарии, поэтому я публикую отдельный ответ и направлю его ему!

Вы упоминаете о проблемах с Press-> Release-> Move.Вы должны использовать MouseCapturing, чтобы получить MouseEvents, даже когда Mouse больше не находится над ScrollViewer.Я не проверял его, но я думаю, что ваше решение также не работает в Press->Move->Move outside of ScrollViewer->Release, MouseCapturing также позаботится об этом.

Также вы упомянули использование Поведения.Я бы предпочел прикрепленное поведение , которое не требует дополнительных зависимостей.

Вам определенно не следует использовать дополнительный холст, но делайте это в Adorner.

Сам ScrollViewer содержит ScrollContentPresenter, который определяет AdornerLayer.Вы должны вставить Adorner там.Это устраняет необходимость в какой-либо дополнительной зависимости, а также сохраняет присоединенное поведение таким простым, как IsMiddleScrollable="true".

...