Java Swing - Nimbus L&F переопределяет пользовательский значок в заголовке JTable после применения сортировки

Я пытаюсь создать собственный модуль визуализации ячеек, который будет отображать изображение в ячейке заголовка JTable. Я получил исходный код для работы с Metal L&F, но у меня возникли проблемы с Nimbus. В нормальных условиях Nimbus нормально отображает изображение. Однако, когда таблица отсортирована, Nimbus отобразит значок сортировки вместо значка, который я указал. Это отличается от Metal L&F, так как он всегда будет отображать значок, который я предоставил.

Пример изображения, демонстрирующего ошибку в Nimbus L&F и Metal L&F

Кто-нибудь знает, как заставить Nimbus рисовать изображение, даже если столбец отсортирован?

Я использую Java 6.29 и Nimbus. Я не могу изменить выпуск Java или L&F.

Кроме того, я пытался сделать некоторые другие обходные пути, такие как изменение метки для использования HTML и тега img для отображения изображения, но это дает странный визуальный эффект. РЕДАКТИРОВАТЬ Текст и изображение не выровнены должным образом (даже с тегом выравнивания HTML в теге img). Вот изображение, обратите внимание, как текст в столбце Temp Hi не выравнивается:

пример изображения решения с тегом HMTL и img

import java.awt.Component;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.TableCellRenderer;


public class ImageChangeDemo extends JFrame {
    public static void main(String args[]) {
        //comment out the code below to try in Metal L&F
        try {
            for(javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.
                    getInstalledLookAndFeels()) {
                if("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        }
        catch(Exception ex) {
            ex.printStackTrace();
        }

        java.awt.EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                new ImageChangeDemo().setVisible(true);
            }
        });
    }

    public ImageChangeDemo(){
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        JScrollPane pane = new javax.swing.JScrollPane();
        JTable table = new javax.swing.JTable();
        table.setAutoCreateRowSorter(true);
        table.setModel(new javax.swing.table.DefaultTableModel(
            new Object [][] {
                {"a", "q", "h", "v"},
                {"b", "m", "l", "h"},
                {"d", "c", "a", "d"},
                {"j", "o", "y", "e"}
            },
            new String [] {
                "Col 1", "Col 2", "Col 3", "Col 4"
            }
        ) {
            Class[] types = new Class [] {
                String.class, String.class, String.class, String.class
            };
            @Override
            public Class getColumnClass(int columnIndex) {
                return types [columnIndex];
            }
        });
        pane.setViewportView(table);
        this.add(pane);

        table.getTableHeader().setDefaultRenderer(new ImageRenderer(table));

        pack();
    }

    public class ImageRenderer extends DefaultTableCellRenderer{
        TableCellRenderer orig;
        ImageIcon icon;
        ImageRenderer(JTable table){
            orig = table.getTableHeader().getDefaultRenderer();
        }
        @Override
        public Component getTableCellRendererComponent(final JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
            Component c = 
                    orig.getTableCellRendererComponent(
                        table, value, isSelected, hasFocus, row, column);
            if(c instanceof JLabel){
                if(true){
                    JLabel label = (JLabel)c;
                    label.setIcon(makeIcon());
                }
            }
            return c;
        }

        public ImageIcon makeIcon(){
            if(icon == null)
            icon = new ImageIcon(
                    ImageChangeDemo.class.getResource("/resources/green_triangle_down.png"));
            return icon;
        }
    }
}

РЕДАКТИРОВАТЬ: Вот пример сценария того, что должно делать мое реальное приложение: если столбец таблицы содержит определенные данные (например, любые строки, начинающиеся с определенного слова), отобразите значок предупреждения рядом с именем столбца в заголовке таблицы. Я получил это, чтобы работать нормально. Теперь, если пользователь сортирует столбец с изображением, Nimbus удаляет изображение и заменяет его значком сортировки — я все еще хочу, чтобы отображался исходный значок предупреждения.


person Gregory Peck    schedule 06.06.2012    source источник
comment
это производит странный визуальный эффект Странно, как ЛСД? (Небольшой) скриншот рисует тысячу слов.   -  person Andrew Thompson    schedule 06.06.2012
comment
Текст и изображение плохо выровнены (даже с тегом выравнивания HTML в теге img). Я не включил много подробностей по этому поводу, так как это не мой предпочтительный способ решения проблемы - в реальном приложении многие заголовки таблиц отформатированы с помощью HTML. Таким образом, добавление тега img в этой ситуации вызывает множество проблем с выравниванием. Я обновлю свой пост, чтобы включить немного больше деталей. Если у меня будет время, я постараюсь опубликовать различные скриншоты.   -  person Gregory Peck    schedule 06.06.2012
comment
@Грегори Пек, вы хотите заменить встроенные значки для RowSorter своими собственными ???   -  person mKorbel    schedule 06.06.2012
comment
Не совсем. Я хочу добавить значок в заголовки определенных столбцов на основе условия (не в моем источнике, но является частью реальной программы). Когда я сортирую столбец, Nimbus заменяет значок, который я установил, на значок по возрастанию/убыванию по умолчанию.   -  person Gregory Peck    schedule 06.06.2012
comment
@ Андрей Я отредактировал пример с картинками.   -  person Gregory Peck    schedule 06.06.2012
comment
этот код не работает и с экземпляром Metal, т.к. заменены иконки из RowSorter   -  person mKorbel    schedule 06.06.2012
comment
вы пробовали getIcon от RowSorter f.e. получить порядок сортировки ()   -  person mKorbel    schedule 06.06.2012
comment
@mKorbel Спасибо за помощь, но когда я запускаю код, в Metal L&F он работает так, как хотелось бы. Я ищу, чтобы значки сортировки по умолчанию были заменены значком, который я предоставил.   -  person Gregory Peck    schedule 06.06.2012
comment
@mKorbel извините, если возникла путаница, я не собираюсь настраивать значок порядка сортировки, я хочу использовать свой собственный значок. В примерах изображений, которые я разместил, это зеленый треугольник, но в реальном приложении он будет индикатором данных, содержащихся в столбце (например, значок предупреждения, предупреждающий пользователя о неверных данных).   -  person Gregory Peck    schedule 06.06.2012
comment
вам нужно заменить значок, а не добавлять значок в компонент / JComponent / JLabel, возвращающий Renderer, я опубликую краткий ответ   -  person mKorbel    schedule 06.06.2012


Ответы (2)


  • не воссоздавай Icon внутри Renderer, подготовь это заранее, иначе ты будешь воссоздавать Icon в сумасшедшие периоды

  • не добавлять Icon к Component / JComponent / JLabel возвращает Renderer

  • поставить на Renderer

код, сделанный Дэррилом или Робом

protected Icon getIcon(JTable table, int column) {
    SortKey sortKey = getSortKey(table, column);
    if (sortKey != null && table.convertColumnIndexToView(
        sortKey.getColumn()) == column) {
        switch (sortKey.getSortOrder()) {
            case ASCENDING:
                return UIManager.getIcon("Table.ascendingSortIcon");
            case DESCENDING:
                return UIManager.getIcon("Table.descendingSortIcon");
        }
    }
    return null;
}

РЕДАКТИРОВАТЬ

благодаря Renderer от @trashgod, UNSORTED не требуется переопределять для Renderer, попробуйте и наслаждайтесь

первоначальный вид

введите здесь описание изображения

ПО ВОЗРАСТАНИЮ

введите здесь описание изображения

ПО Убыванию

введите здесь описание изображения

НЕСОРТИРОВКА

введите здесь описание изображения

import java.awt.Component;
import java.util.ArrayList;
import java.util.List;
import javax.swing.Icon;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.RowSorter;
import javax.swing.SortOrder;
import javax.swing.UIManager;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableModel;
import javax.swing.table.TableRowSorter;

public class ImageChangeDemo extends JFrame {

    private static final long serialVersionUID = 1L;
    private JTable table = new javax.swing.JTable();

    public static void main(String args[]) {
        //comment out the code below to try in Metal L&F
        try {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
                if ("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        java.awt.EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new ImageChangeDemo().setVisible(true);
            }
        });
    }

    public ImageChangeDemo() {
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        JScrollPane pane = new javax.swing.JScrollPane();



        table.setModel(new javax.swing.table.DefaultTableModel(
                new Object[][]{
                    {"a", "q", "h", "v"},
                    {"b", "m", "l", "h"},
                    {"d", "c", "a", "d"},
                    {"j", "o", "y", "e"}
                },
                new String[]{
                    "Col 1", "Col 2", "Col 3", "Col 4"
                }) {

            private static final long serialVersionUID = 1L;
            Class[] types = new Class[]{
                String.class, String.class, String.class, String.class
            };

            @Override
            public Class getColumnClass(int columnIndex) {
                return types[columnIndex];
            }
        });
        TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(table.getModel()) {

            @Override
            public void toggleSortOrder(int column) {
                if (column >= 0 && column < getModelWrapper().getColumnCount() && isSortable(column)) {
                    List<SortKey> keys = new ArrayList<SortKey>(getSortKeys());
                    if (!keys.isEmpty()) {
                        SortKey sortKey = keys.get(0);
                        if (sortKey.getColumn() == column && sortKey.getSortOrder() == SortOrder.DESCENDING) {
                            setSortKeys(null);
                            return;
                        }
                    }
                }
                super.toggleSortOrder(column);
            }
        };
        table.setRowSorter(sorter);
        table.setPreferredScrollableViewportSize(table.getPreferredSize());
        table.setDefaultRenderer(ImageChangeDemo.class, new HeaderRenderer(table));
        pane.setViewportView(table);
        add(pane);
        pack();
    }

    class HeaderRenderer implements TableCellRenderer {

       final TableCellRenderer renderer;

        public HeaderRenderer(JTable table) {
            renderer = table.getTableHeader().getDefaultRenderer();
        }

        @Override
        public Component getTableCellRendererComponent(
                JTable table, Object value, boolean isSelected,
                boolean hasFocus, int row, int col) {
            return renderer.getTableCellRendererComponent(
                    table, value, isSelected, hasFocus, row, col);
        }

        public Icon getIcon(JTable table, int column) {
            for (RowSorter.SortKey sortKey : table.getRowSorter().getSortKeys()) {
                if (sortKey.getColumn() == column) {
                    switch (sortKey.getSortOrder()) {
                        case ASCENDING:
                            return (UIManager.getIcon("Table.ascendingSortIcon"));
                        case DESCENDING:
                            return (UIManager.getIcon("Table.descendingSortIcon"));
                    }
                }
            }
            return null;
        }
    }
}

РЕДАКТИРОВАТЬ 2

затем установить значок непосредственно в UIManager

введите здесь описание изображения

введите здесь описание изображения

import java.awt.Color;
import java.awt.Component;
import java.awt.Graphics;
import java.util.ArrayList;
import java.util.List;
import javax.swing.Icon;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.RowSorter.SortKey;
import javax.swing.SortOrder;
import javax.swing.UIManager;
import javax.swing.table.TableModel;
import javax.swing.table.TableRowSorter;

public class ImageChangeDemo extends JFrame {

    private static final long serialVersionUID = 1L;
    private JTable table = new javax.swing.JTable();

    public static void main(String args[]) {
        //comment out the code below to try in Metal L&F
        try {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
                if ("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    UIManager.getLookAndFeelDefaults().put("Table.ascendingSortIcon", new BevelArrowIcon(BevelArrowIcon.UP, false, false));
                    UIManager.getLookAndFeelDefaults().put("Table.descendingSortIcon", new BevelArrowIcon(BevelArrowIcon.DOWN, false, false));
                    break;
                }
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        java.awt.EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new ImageChangeDemo().setVisible(true);
            }
        });
    }

    public ImageChangeDemo() {
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        JScrollPane pane = new javax.swing.JScrollPane();
        //table.setAutoCreateRowSorter(true);
        table.setModel(new javax.swing.table.DefaultTableModel(
                new Object[][]{
                    {"a", "q", "h", "v"},
                    {"b", "m", "l", "h"},
                    {"d", "c", "a", "d"},
                    {"j", "o", "y", "e"}
                },
                new String[]{
                    "Col 1", "Col 2", "Col 3", "Col 4"
                }) {

            private static final long serialVersionUID = 1L;
            Class[] types = new Class[]{
                String.class, String.class, String.class, String.class
            };

            @Override
            public Class getColumnClass(int columnIndex) {
                return types[columnIndex];
            }
        });
        TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(table.getModel()) {

            @Override
            public void toggleSortOrder(int column) {
                if (column >= 0 && column < getModelWrapper().getColumnCount() && isSortable(column)) {
                    List<SortKey> keys = new ArrayList<SortKey>(getSortKeys());
                    if (!keys.isEmpty()) {

                        SortKey sortKey = keys.get(0);
                        if (sortKey.getColumn() == column && sortKey.getSortOrder() == SortOrder.DESCENDING) {
                            setSortKeys(null);
                            return;
                        }
                    }
                }
                super.toggleSortOrder(column);
            }
        };
        table.setRowSorter(sorter);
        //table.getTableHeader().setDefaultRenderer(new DefaultTableHeaderCellRenderer());
        //table.setDefaultRenderer(ImageChangeDemo.class, new HeaderRenderer(table));
        table.setPreferredScrollableViewportSize(table.getPreferredSize());
        pane.setViewportView(table);
        add(pane);
        pack();
    }

   static class BevelArrowIcon implements Icon {

        public static final int UP = 0;         // direction
        public static final int DOWN = 1;
        private static final int DEFAULT_SIZE = 11;
        private Color edge1;
        private Color edge2;
        private Color fill;
        private int size;
        private int direction;

        public BevelArrowIcon(int direction, boolean isRaisedView, boolean isPressedView) {
            if (isRaisedView) {
                if (isPressedView) {
                    init(UIManager.getColor("controlLtHighlight"), UIManager.getColor("controlDkShadow"), UIManager.getColor("controlShadow"), DEFAULT_SIZE, direction);
                } else {
                    init(UIManager.getColor("controlHighlight"), UIManager.getColor("controlShadow"), UIManager.getColor("control"), DEFAULT_SIZE, direction);
                }
            } else {
                if (isPressedView) {
                    init(UIManager.getColor("controlDkShadow"), UIManager.getColor("controlLtHighlight"), UIManager.getColor("controlShadow"), DEFAULT_SIZE, direction);
                } else {
                    init(UIManager.getColor("controlShadow"), UIManager.getColor("controlHighlight"), UIManager.getColor("control"), DEFAULT_SIZE, direction);
                }
            }
        }

        public BevelArrowIcon(Color edge1, Color edge2, Color fill, int size, int direction) {
            init(edge1, edge2, fill, size, direction);
        }

        @Override
        public void paintIcon(Component c, Graphics g, int x, int y) {
            switch (direction) {
                case DOWN:
                    drawDownArrow(g, x, y);
                    break;
                case UP:
                    drawUpArrow(g, x, y);
                    break;
            }
        }

        @Override
        public int getIconWidth() {
            return size;
        }

        @Override
        public int getIconHeight() {
            return size;
        }

        private void init(Color edge1, Color edge2, Color fill, int size, int direction) {
            edge1 = Color.red;
            edge2 = Color.blue;
            this.edge1 = edge1;
            this.edge2 = edge2;
            this.fill = fill;
            this.size = size;
            this.direction = direction;
        }

        private void drawDownArrow(Graphics g, int xo, int yo) {
            g.setColor(edge1);
            g.drawLine(xo, yo, xo + size - 1, yo);
            g.drawLine(xo, yo + 1, xo + size - 3, yo + 1);
            g.setColor(edge2);
            g.drawLine(xo + size - 2, yo + 1, xo + size - 1, yo + 1);
            int x = xo + 1;
            int y = yo + 2;
            int dx = size - 6;
            while (y + 1 < yo + size) {
                g.setColor(edge1);
                g.drawLine(x, y, x + 1, y);
                g.drawLine(x, y + 1, x + 1, y + 1);
                if (0 < dx) {
                    g.setColor(fill);
                    g.drawLine(x + 2, y, x + 1 + dx, y);
                    g.drawLine(x + 2, y + 1, x + 1 + dx, y + 1);
                }
                g.setColor(edge2);
                g.drawLine(x + dx + 2, y, x + dx + 3, y);
                g.drawLine(x + dx + 2, y + 1, x + dx + 3, y + 1);
                x += 1;
                y += 2;
                dx -= 2;
            }
            g.setColor(edge1);
            g.drawLine(xo + (size / 2), yo + size - 1, xo + (size / 2), yo + size - 1);
        }

        private void drawUpArrow(Graphics g, int xo, int yo) {
            g.setColor(edge1);
            int x = xo + (size / 2);
            g.drawLine(x, yo, x, yo);
            x--;
            int y = yo + 1;
            int dx = 0;
            while (y + 3 < yo + size) {
                g.setColor(edge1);
                g.drawLine(x, y, x + 1, y);
                g.drawLine(x, y + 1, x + 1, y + 1);
                if (0 < dx) {
                    g.setColor(fill);
                    g.drawLine(x + 2, y, x + 1 + dx, y);
                    g.drawLine(x + 2, y + 1, x + 1 + dx, y + 1);
                }
                g.setColor(edge2);
                g.drawLine(x + dx + 2, y, x + dx + 3, y);
                g.drawLine(x + dx + 2, y + 1, x + dx + 3, y + 1);
                x -= 1;
                y += 2;
                dx += 2;
            }
            g.setColor(edge1);
            g.drawLine(xo, yo + size - 3, xo + 1, yo + size - 3);
            g.setColor(edge2);
            g.drawLine(xo + 2, yo + size - 2, xo + size - 1, yo + size - 2);
            g.drawLine(xo, yo + size - 1, xo + size, yo + size - 1);
        }
    }
}
person mKorbel    schedule 06.06.2012
comment
Спасибо за помощь, но я не понимаю, куда бы я поместил это в свой исходный код. Не могли бы вы уточнить? - person Gregory Peck; 06.06.2012
comment
case UNSORTED:? См. также этот ответ. - person trashgod; 06.06.2012
comment
@mKorbel Я изменил изображения в своем вопросе, так что, может быть, это немного понятнее - я не собираюсь менять значок сортировки диспетчера пользовательского интерфейса. Я хочу добавить дополнительный значок в заголовок таблицы. - person Gregory Peck; 06.06.2012
comment
@trashgod Спасибо за ссылку, но я уже это делаю. Проблема в моем вопросе заключается в том, что Nimbus заменяет значок, который я установил, значком сортировки. - person Gregory Peck; 06.06.2012
comment
Я отредактирую свой ответ здесь, как упомянул @trashgod, я добавлю UNSORTED (я думаю, что в этом случае не требуется переопределять Renderer), и, используя его код, я опубликую SSCCE - person mKorbel; 06.06.2012
comment
@Грегори Пек, чтобы заменить return (UIManager.getIcon("Table.ascendingSortIcon")); любым Icon - person mKorbel; 06.06.2012
comment
@mKorbel Спасибо, но у меня все еще проблемы с настройкой изображения. Похоже, ваше новое обновление изменяет только порядок сортировки. Меня не интересует порядок сортировки, вместо этого я помещаю пользовательский значок в строку заголовка JTable. Кроме того, при запуске вашего нового источника с помощью отладчика getIcon() никогда не вызывается - person Gregory Peck; 06.06.2012
comment
@mKorbel Я пытаюсь добавить изображение в любой заголовок строки. Моя проблема в том, что Nimbus стирает мое пользовательское изображение со значком сортировки. Есть ли у вас какие-либо мысли о том, как отобразить пользовательское изображение в любой ячейке заголовка строки? (Хотя мне очень нравится ваш BevelIcon) - person Gregory Peck; 06.06.2012
comment
@Грегори Пек, может быть другая проблема Nimbus UIManager и Nimbus Defaults говоря об иконке в смешном разрешении 7x7 в пикселях, разве это не твоя проблема? - person mKorbel; 06.06.2012
comment
(Я думаю) для Standard L&F есть TableHeaderRenderers, созданные великими умами, не уверен, что be работает (лень сегодня что-то тестировать) под Nimbus L&F - person mKorbel; 07.06.2012
comment
Большое спасибо за помощь, это очень сложный вопрос. Что я решил сделать, так это создать средство визуализации, которое возвращает JLayeredPane, содержащее исходный заголовок, а также изображение, которое я хочу отобразить на более высоком уровне. Я опубликую то, что у меня получилось, но это займет некоторое время. - person Gregory Peck; 07.06.2012
comment
никогда не использовать для этого JLayeredPane, используйте GlassPane или возможно, лучше было бы JLayer (Java7) на основе JXLayer (Java6), - person mKorbel; 07.06.2012
comment
@mKorbel Почему бы мне не использовать для этого JLayerpedPane? Я думал об использовании GlassPane, но знаю, что в финальном приложении другие классы используют стеклянную панель, и я не хотел им мешать. - person Gregory Peck; 07.06.2012
comment
этот компонент до сих пор помнит динозавров - person mKorbel; 07.06.2012
comment
В вашем первом примере код для getIcon никогда не используется... поэтому я не могу эффективно менять значки. Я могу прокомментировать этот метод, и он работает - person francogp; 11.07.2018

Итак, после долгих проб и ошибок я смог найти способ разместить свой собственный значок в строке заголовка, даже если столбец отсортирован. По сути, я сделал так, чтобы средство визуализации возвращало пользовательскую панель, содержащую 2 дочерних элемента, изображение в JLabel и компонент, который изначально был создан средством визуализации по умолчанию. (Обратите внимание, что этот обходной путь необходим только для Nimbus L&F, а исходный код примера отлично работает в Metal L&F)

Этот код использует StackLayout, созданный Роменом Гаем, как показано в его книге Filthy Rich Clients (см. стр. 245). Вот источник для StackLayout

Вот код, который я создал для рендерера. Обязательно загрузите StackLayout, иначе он не скомпилируется.

import java.awt.Component;
import java.awt.Dimension;
import java.awt.FontMetrics;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.TableCellRenderer;


public class ImageChangeDemo extends JFrame {
    public static void main(String args[]) {
        //comment out the code below to try in Metal L&F
        try {
            for(javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.
                    getInstalledLookAndFeels()) {
                if("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        }
        catch(Exception ex) {
            ex.printStackTrace();
        }

        java.awt.EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                new ImageChangeDemo().setVisible(true);
            }
        });
    }

    public ImageChangeDemo(){
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        JScrollPane pane = new javax.swing.JScrollPane();
        JTable table = new javax.swing.JTable();
        table.setAutoCreateRowSorter(true);
        table.setModel(new javax.swing.table.DefaultTableModel(
            new Object [][] {
                {"a", "q", "h", "v"},
                {"b", "m", "l", "h"},
                {"d", "c", "a", "d"},
                {"j", "o", "y", "e"}
            },
            new String [] {
                "Col 1", "Col 2", "Col 3", "Col 4"
            }
        ) {
            Class[] types = new Class [] {
                String.class, String.class, String.class, String.class
            };
            @Override
            public Class getColumnClass(int columnIndex) {
                return types [columnIndex];
            }
        });
        pane.setViewportView(table);
        this.add(pane);


        pack();
        //set renderer after pack so header row has correct default height
        table.getTableHeader().setDefaultRenderer(new ImageRenderer(table));


    }

    public class ImageRenderer extends DefaultTableCellRenderer{
        TableCellRenderer orig;
        private final ImageIcon icon = new ImageIcon(
                    ImageChangeDemo.class.getResource("/resources/exclamation-icon.png"));;
        private JPanel jp = new JPanel(new StackLayout());
        private final JLabel pic = new JLabel(icon);
        { //extra initialization for PIC
            pic.setHorizontalAlignment(JLabel.LEADING); //so it isn't centered in stack layout
        }

        ImageRenderer(JTable table){
            orig = table.getTableHeader().getDefaultRenderer();
        }

        @Override
        public Component getTableCellRendererComponent(final JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
            Component c = 
                    orig.getTableCellRendererComponent(
                        table, value, isSelected, hasFocus, row, column);
            if(true){
                int width  = table.getColumnModel().getColumn(column).getWidth();
                int height = table.getTableHeader().getSize().height;
                System.out.println("height"+height);

                jp.removeAll();                //clean the JPanel

                //move text in label to the left so it isn't covered by the icon
                if(c instanceof JLabel){
                    JLabel l = (JLabel) c;
                    l.setPreferredSize(new Dimension(width, height));

                    FontMetrics fontMetrics = l.getFontMetrics(l.getFont());
                    int sizeOfSpace = fontMetrics.charWidth(' ');
                    int numSpaces = (int)Math.round(icon.getIconWidth() / (double)sizeOfSpace);
                    StringBuilder sb = new StringBuilder();
                    for(int i = 0; i < numSpaces; i++)
                        sb.append(' ');

                    //account for HTML in header messages
                    if(l.getText().toLowerCase().startsWith("<html>")){
                        l.setText(  l.getText().substring(0, "<html>".length()) +
                                    sb.toString() +
                                    l.getText().substring("<html>".length()));
                    }
                    else
                        l.setText(sb.toString()+l.getText());
                }


                //Add components to the JPanel & return it.
                jp.add(c, StackLayout.BOTTOM);  //will contain modifications for spacing.
                jp.add(pic, StackLayout.TOP);
                return jp;

            }
            else
                return c;
        }
    }
}
person Gregory Peck    schedule 07.06.2012
comment
setHorizontalAlignment(CENTER) и setHorizontalTextPosition(LEFT) и setVerticalAlignment(BOTTOM) - person mKorbel; 07.06.2012
comment
@mKorbel Спасибо за предложение о выравнивании, я бы выбрал этот подход, за исключением того, что в реальном приложении разные столбцы будут выравниваться по правому / левому краю. Я не хотел глобально менять выравнивание только для того, чтобы добавить изображение. Таким образом, решение, которое я придумал (добавление пробелов в текст), является хакерским и не очень надежным, но оно допускает различное выравнивание в разных столбцах. - person Gregory Peck; 07.06.2012
comment
вы грязно усложняете простые вещи, аааааахххх, почему бы вам не установить значки в UIManager, вы можете создать 100 таблиц в одном JFrame, и каждый из них будет иметь разные значки, сразу после создания нового JTable вы будете для вызова UIMAnager.put(....), а затем SwingUtilities.updateComponentTreeUI(mytable), - person mKorbel; 07.06.2012
comment
Я не могу сделать то же самое, что и с JPanel, может быть, я что-то упускаю, в результате в этом случае оба JTable имеют одинаковые значки, аааааааааааааааааааааааааааааааааааааааааааааааааааааа - person mKorbel; 07.06.2012
comment
@mKorbel - это хорошее предложение, о котором я не подумал. В реальной версии этого класса у меня есть рендерер в своем собственном классе со статичным значком, поэтому для всех таблиц должен быть только один экземпляр. В демонстрационном коде я не мог создать значок как статический, так как это внутренний класс. Код, который я предоставил, представляет собой урезанное решение, а не более надежное, которое у меня есть в моем реальном проекте. - person Gregory Peck; 07.06.2012
comment
@mKorbel Для моих целей (и для чего это будет использоваться) один и тот же значок будет использоваться во всем моем приложении, поэтому не проблема иметь один и тот же значок везде. - person Gregory Peck; 07.06.2012
comment
:-) Да, насколько я знаю, с Нимбусом очень сложно иметь дело. - person Gregory Peck; 07.06.2012
comment
хммм есть один или два Custom Look and Feels на основе Nimbus, один из них декларировал чистый доступ к темам UIManager и Colors :-) - person mKorbel; 07.06.2012