Можно ли использовать JXBusyLabel в качестве ячейки, отображаемой в JTable? - PullRequest
4 голосов
/ 20 мая 2011

Я хотел бы использовать JXBusyLabel в ячейке, чтобы уведомить пользователя о том, что в данный момент происходит событие для строки, в которой находится JXBusyLabel. Например, двойной щелчок по строке, чтобы открыть ее, вызовет произвольную запись JXBusyLabel.

Имеет ли это смысл?

Если вам интересно, что такое JXBusyLabel, посмотрите здесь .

Спасибо!

[ПРАВИТЬ] Решение на основе ответа @kleopatra :

import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.Timer;
import javax.swing.table.DefaultTableModel;

import org.jdesktop.swingx.JXTable;
import org.jdesktop.swingx.decorator.AbstractHighlighter;
import org.jdesktop.swingx.decorator.HighlightPredicate;
import org.jdesktop.swingx.decorator.PainterHighlighter;
import org.jdesktop.swingx.painter.BusyPainter;

public class TableBusyLabelTest {

    private JFrame frame;
    private JXTable table;
    private JButton button;

    private BusyPainter busyPainter;
    private AbstractHighlighter highlighter;
    private Timer timer;

    private boolean on = false;

    /**
     * Launch the application.
     */
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    TableBusyLabelTest window = new TableBusyLabelTest();
                    window.frame.setVisible(true);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }

    /**
     * Create the application.
     */
    public TableBusyLabelTest() {
        initialize();
    }

    /**
     * Initialize the contents of the frame.
     */
    private void initialize() {
        frame = new JFrame();
        frame.setBounds(100, 100, 450, 300);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        busyPainter = new BusyPainter(15);

        timer = new Timer(100, getTimerActionListener());

        highlighter = new PainterHighlighter(HighlightPredicate.NEVER,
                busyPainter);

        table = new JXTable();
        table.setModel(getTableExampleModel());
        // Tell which column will use the highlighter
        table.getColumnExt(0).addHighlighter(highlighter);

        button = new JButton("Start / Stop busy thing");
        button.addActionListener(getButtonActionListener());

        frame.getContentPane().add(table, BorderLayout.CENTER);
        frame.getContentPane().add(button, BorderLayout.SOUTH);
    }

    private DefaultTableModel getTableExampleModel() {
        return new DefaultTableModel(new Object[][] { { null, "Test1" },
                { null, "Test2" }, }, new String[] { "busy", "Name" });
    }

    private ActionListener getTimerActionListener() {
        return new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                int frame = busyPainter.getFrame();
                frame = (frame + 1) % busyPainter.getPoints();
                busyPainter.setFrame(frame);
            }

        };
    }

    private ActionListener getButtonActionListener() {
        return new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {

                if (on) {
                    on = false;
                } else {
                    on = true;
                }

                // on a change that should toggle the busy-ness on/off
                if (on) {
                    highlighter
                            .setHighlightPredicate(HighlightPredicate.ALWAYS);
                    timer.start();
                } else {
                    highlighter.setHighlightPredicate(HighlightPredicate.NEVER);
                    timer.stop();
                }
            }
        };
    }

}

Ответы [ 2 ]

2 голосов
/ 14 июня 2011

В зависимости от вашего конкретного случая использования вы не захотите использовать JXBusyLabel в качестве компонента рендеринга (вы не получите анимацию, так как это рендер ), все, что вам нужно, это PainterHighlighterнастроен с BusyPainter, свойство кадра которого контролируется таймером.Независимо от того, является ли художник видимым, должно быть привязано к какому-либо свойству ваших данных, которое запускает включение / выключение подсветки HighlightPredicate.

Например, см. PainterVisualCheck interactiveAnimatedBusyPainterHighlight в иерархии тестов swingx, средство визуализации пакетов.Что-то вроде:

    BusyPainter painter = new BusyPainter();
    ActionListener l = new ActionListener() {
        public void actionPerformed(ActionEvent e) {
            int frame = busyPainter.getFrame();
            frame = (frame+1)%busyPainter.getPoints();
            busyPainter.setFrame(frame);
        }

    };
    Timer timer = new Timer(delay, l);
    AbstractHighlighter hl = new PainterHighlighter(HighlightPredicate.NEVER, painter);
    table.getColumnExt().addHighlighter(hl);

    // on a change that should toggle the busy-ness on/off
    if (on) {
         hl.setHighlightPredicate(HighlightPredicate.ALWAYS);
         timer.start();
    } else {
         hl.setHighlightPredicate(HighlightPredicate.NEVER);
         timer.stop(); 
    }

Редактировать (отвечая на расширенный вопрос):

, чтобы активировать подсветку занятости только для определенного условия, реализовать пользовательский HighlightPredicate и установить его вместо ВСЕГДА.Fi конкретная строка в столбце:

    HighlightPredicate predicate = new HighlightPredicate() {

        @Override
        public boolean isHighlighted(Component renderer,
                ComponentAdapter adapter) {

            return 
               adapter.convertRowIndexToModel(adapter.row) == mySpecialRow;
        }

    };
1 голос
/ 27 мая 2011

Да, это возможно.Я тоже использовал это в своем столе.Выглядит очень круто для отложенной загрузки.

private class YourCellRenderer implements TableCellRenderer {
private final JLabel okLabel;
private final JXBusyLabel busyLabel;

public YourCellRenderer() {
  busyLabel = new JXBusyLabel(new Dimension(50, 50));
  busyLabel.setHorizontalAlignment(JXLabel.CENTER);
  busyLabel.setBusy(true);

  okLabel = new JLabel();
}

@Override
public Component getTableCellRendererComponent(final JTable table,
    final Object value, final boolean isSelected, final boolean hasFocus,
    final int row, final int column) {

  if (((YourClass)value).isBusy()) {
    return busyLabel;
  } else {
    okLabel.set(((YourClass)value).getText());
    return okLabel;
  }

  return null;
}

Установите новый рендерер в качестве рендерера по умолчанию для YourClass:

table.setDefaultRenderer(YourClass.class, new YourCellRenderer());

Ваша модель таблицы, конечно, должна возвращать экземпляры YourClass.

...