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