JScrollPane и JList с автоматической прокруткой

У меня есть следующий код:

    listModel = new DefaultListModel();
    listModel.addElement(dateFormat.format(new Date()) + ": Msg1");
    messageList = new JList(listModel);
    messageList.setLayoutOrientation(JList.VERTICAL);

    messageScrollList = new JScrollPane(messageList);
    messageScrollList.setPreferredSize(new Dimension(500, 200));

    messageScrollList.getVerticalScrollBar().addAdjustmentListener(new AdjustmentListener() {  
        public void adjustmentValueChanged(AdjustmentEvent e) {  
            e.getAdjustable().setValue(e.getAdjustable().getMaximum());  
        }
    }); 

Это автоматически прокручивается вниз. Но если я попытаюсь прокрутить назад вверх, чтобы перечитать сообщение, оно заставит прокрутить вниз. Как я могу это исправить?

7 ответов

Решение

При добавлении нового сообщения вызывать scrollRectToVisible() на JList используя Rectangle имеющие те же размеры, что и предпочитаемый размер панели сообщений. Учитывая вертикальную ориентацию, может быть удобно сделать предпочтительный размер JScrollPane "s JViewportцелое кратное высоты панели сообщений. Смотрите также: Как использовать панели прокрутки.

Приложение: Это убедительное обсуждение прокрутки текстовой области также может быть полезным.

this.list = blah blah... 
this.list.setSelectedValue(whatever);   
final JScrollPane sp = new JScrollPane(this.list); // needs to be after the parent is the sp 
this.list.ensureIndexIsVisible(this.list.getSelectedIndex());

Я нашел это действительно полезным: http://forums.sun.com/thread.jspa?threadID=623669 (сообщение от 'inopia')
Работает отлично

Как он говорит: "Проблема здесь в том, что может стать немного трудно найти событие, которое запускается после обновления ListModel, JList и JScrollPane".

Я использовал JScrollPane с JTextArea, Я избежал принудительной прокрутки вниз, прокручивая только когда JScrollBar max значение изменилось:

        JScrollPane scrollPane = new JScrollPane(jTextArea);

        verticalScrollBarMaximumValue = scrollPane.getVerticalScrollBar().getMaximum();
        scrollPane.getVerticalScrollBar().addAdjustmentListener(
                e -> {
                    if ((verticalScrollBarMaximumValue - e.getAdjustable().getMaximum()) == 0)
                        return;
                    e.getAdjustable().setValue(e.getAdjustable().getMaximum());
                    verticalScrollBarMaximumValue = scrollPane.getVerticalScrollBar().getMaximum();
                });

Я полагаю, есть лучшее решение для фильтрации событий без дополнительных переменных, и было бы полезно, если бы кто-то опубликовал это.

@ вопросник. Пожалуйста, измените ваш код с

messageScrollList.getVerticalScrollBar().addAdjustmentListener(new AdjustmentListener() {  
        public void adjustmentValueChanged(AdjustmentEvent e) {  
            e.getAdjustable().setValue(e.getAdjustable().getMaximum());  
        }
    });

в

messageScrollList.getVerticalScrollBar().addAdjustmentListener(new AdjustmentListener() {  
        public void adjustmentValueChanged(AdjustmentEvent e) {  
            e.getAdjustable().setValue(e.getAdjustable().getValue());  
        }
    }); 

Это изменение getMaximum() в getValue() Тогда это работает как требуется. getMaximum() берет скроллер на максимум; в то время как getValue() берет это везде, где происходит событие.

Вы можете вообще не использовать этот кусок кода, если поставите одну строку

messageScrollList.setViewportView(messageList);

Это работает как add(component) для jList и получает присущее ему поведение.

Для автоматической прокрутки я использую очень простую концепцию статической переменной и list.makeVisible(int index)

    public class autoScrollExample
    {
     static int index=0;
     private void someFunction()
     /*   
     * 
     */
     list1.add("element1");
     list1.makeVisible(index++);
     /*
     *
     */

     private void someOtherFunction()
     /*   
     * 
     */
     list1.add("element2");
     list1.makeVisible(index++);
     /*
     *
     */


    }

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

Adjustable sb = messageScrollList.getVerticalScrollBar()
boolean onBottom = sb.getValue() == sb.getMaximum();
//
// add your message to the JList.
//
if(onBottom)  sb.setValue(sb.getMaximum());

В противном случае вам нужно будет сказать, была ли корректировка вызвана изменением модели или мышью, и просматривая документы API, я не уверен, есть ли способ сделать это легко. Хотя вы могли видеть, если AdjustmentEvent.getAdjustmentType() в этих случаях возвращает разные значения, если это правда, тогда вы можете просто использовать оператор if в своем анонимном внутреннем классе.

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

Другие вопросы по тегам