0

ビューポートがあり、変更リスナーをアタッチしました。ビューポートをスクロールするたびに、変更リスナーが約 4 回呼び出されます。これを回避する方法がわかりません。呼び出しを 1 回だけにしたいのですが?

4

2 に答える 2

2

これを回避する方法はありません。多くのプロパティへの変更に関する通知を提供しているため、JViewportいくつかのイベントが発生します...stateChanged

JavaDocsから...

ChangeListenerビューのサイズ、位置、またはビューポートのエクステント サイズが変更されるたびに通知されるリストに を追加します。

この時点では、あなたが何を達成しようとしているのかわからないため、何を提案すればよいかわかりにくいですが、 を使用する必要がある場合はChangeListener、合体メカニズムを設定できます。つまり、各イベントに応答するのではなく、基本的にイベント間に十分な遅延が発生するまで待機してから応答します...

例えば...

public class DelayedChangeHandler implements ChangeListener {

    private Timer timer;
    private ChangeEvent last;

    public DelayedChangeHandler() {
        timer = new Timer(250, new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                stableStateChanged();
            }
        });
        timer.setRepeats(false);
    }

    @Override
    public void stateChanged(ChangeEvent e) {
        last = e;
        timer.restart();
    }

    protected void stableStateChanged() {
        System.out.println("Finally...");
    }

}

基本的に、これは短い遅延でChangeListener非繰り返しを使用する実装です。javax.swing.Timerが呼び出されるたびstateChangedに、タイマーが再起動されます。最後に、タイマーの「ティック」が許可されるとstableStateChanged、最後のイベントが発生してから十分な時間が経過したことを示す呼び出しが行われます。

これは、イベントの原因についてはあまり気にせず、イベントが発生したことだけを前提としています...

実行可能な例...

import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.Timer;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

public class TestViewport {

    public static void main(String[] args) {
        new TestViewport();
    }

    public TestViewport() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                }

                JPanel pane = new JPanel() {
                    @Override
                    public Dimension getPreferredSize() {
                        return new Dimension(1000, 1000);
                    }
                };

                JScrollPane sp = new JScrollPane(pane);
                sp.getViewport().addChangeListener(new DelayedChangeHandler());
                sp.getViewport().addPropertyChangeListener(new PropertyChangeListener() {
                    @Override
                    public void propertyChange(PropertyChangeEvent evt) {
                        System.out.println(evt.getPropertyName());
                    }
                });

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(sp);
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class DelayedChangeHandler implements ChangeListener {

        private Timer timer;
        private ChangeEvent last;

        public DelayedChangeHandler() {
            timer = new Timer(250, new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    stableStateChanged();
                }
            });
            timer.setRepeats(false);
        }

        @Override
        public void stateChanged(ChangeEvent e) {
            last = e;
            timer.restart();
        }

        protected void stableStateChanged() {
            System.out.println("Finally...");
        }

    }

}
于 2014-01-23T06:59:05.400 に答える