- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我发誓...我希望这是我最后一个这样问的问题,但我快要发疯了。
我有一个使用自定义 TableCellRenderer 的 JTable,它使用 JEditorPane 在 JTable 的各个单元格中显示 html。我如何处理点击 JEditorPane 中显示的链接?
我知道 HyperlinkListener,但没有鼠标事件通过 JTable 到达 EditorPane 以处理任何 HyperlinkEvents。
如何在 JTable 中的 JEditorPane 中处理超链接?
最佳答案
EditorPane 没有接收到任何事件,因为从 TableCellRenderer 返回的组件只允许显示,而不是拦截事件,这使得它几乎与图像相同,但不允许任何行为。因此,即使注册了监听器,返回的组件也永远不会“意识到”任何事件。解决方法是在 JTable 上注册一个 MouseListener,并从那里拦截所有相关事件。
这是我过去创建的一些类,用于允许 JButton 翻转在 JTable 中工作,但您也应该能够重新使用其中的大部分来解决您的问题。对于需要它的每个单元格,我都有一个单独的 JButton。这样,这个 ActiveJComponentTableMouseListener 就会计算出鼠标事件发生在哪个单元格中,并将事件分派(dispatch)给相应的组件。 ActiveJComponentTableCellRenderer 的工作是通过 Map 跟踪组件。
它足够智能,可以知道何时已触发事件,因此您不会积压冗余事件。为超文本实现这一点应该没有什么不同,您可能仍然希望翻转。这是类(class)
public class ActiveJComponentTableMouseListener extends MouseAdapter implements MouseMotionListener {
private JTable table;
private JComponent oldComponent = null;
private TableCell oldTableCell = new TableCell();
public ActiveJComponentTableMouseListener(JTable table) {
this.table = table;
}
@Override
public void mouseMoved(MouseEvent e) {
TableCell cell = new TableCell(getRow(e), getColumn(e));
if (alreadyVisited(cell)) {
return;
}
save(cell);
if (oldComponent != null) {
dispatchEvent(createMouseEvent(e, MouseEvent.MOUSE_EXITED), oldComponent);
oldComponent = null;
}
JComponent component = getComponent(cell);
if (component == null) {
return;
}
dispatchEvent(createMouseEvent(e, MouseEvent.MOUSE_ENTERED), component);
saveComponent(component);
save(cell);
}
@Override
public void mouseExited(MouseEvent e) {
TableCell cell = new TableCell(getRow(e), getColumn(e));
if (alreadyVisited(cell)) {
return;
}
if (oldComponent != null) {
dispatchEvent(createMouseEvent(e, MouseEvent.MOUSE_EXITED), oldComponent);
oldComponent = null;
}
}
@Override
public void mouseEntered(MouseEvent e) {
forwardEventToComponent(e);
}
private void forwardEventToComponent(MouseEvent e) {
TableCell cell = new TableCell(getRow(e), getColumn(e));
save(cell);
JComponent component = getComponent(cell);
if (component == null) {
return;
}
dispatchEvent(e, component);
saveComponent(component);
}
private void dispatchEvent(MouseEvent componentEvent, JComponent component) {
MouseEvent convertedEvent = (MouseEvent) SwingUtilities.convertMouseEvent(table, componentEvent, component);
component.dispatchEvent(convertedEvent);
// This is necessary so that when a button is pressed and released
// it gets rendered properly. Otherwise, the button may still appear
// pressed down when it has been released.
table.repaint();
}
private JComponent getComponent(TableCell cell) {
if (rowOrColumnInvalid(cell)) {
return null;
}
TableCellRenderer renderer = table.getCellRenderer(cell.row, cell.column);
if (!(renderer instanceof ActiveJComponentTableCellRenderer)) {
return null;
}
ActiveJComponentTableCellRenderer activeComponentRenderer = (ActiveJComponentTableCellRenderer) renderer;
return activeComponentRenderer.getComponent(cell);
}
private int getColumn(MouseEvent e) {
TableColumnModel columnModel = table.getColumnModel();
int column = columnModel.getColumnIndexAtX(e.getX());
return column;
}
private int getRow(MouseEvent e) {
int row = e.getY() / table.getRowHeight();
return row;
}
private boolean rowInvalid(int row) {
return row >= table.getRowCount() || row < 0;
}
private boolean rowOrColumnInvalid(TableCell cell) {
return rowInvalid(cell.row) || columnInvalid(cell.column);
}
private boolean alreadyVisited(TableCell cell) {
return oldTableCell.equals(cell);
}
private boolean columnInvalid(int column) {
return column >= table.getColumnCount() || column < 0;
}
private MouseEvent createMouseEvent(MouseEvent e, int eventID) {
return new MouseEvent((Component) e.getSource(), eventID, e.getWhen(), e.getModifiers(), e.getX(), e.getY(), e.getClickCount(), e.isPopupTrigger(), e.getButton());
}
private void save(TableCell cell) {
oldTableCell = cell;
}
private void saveComponent(JComponent component) {
oldComponent = component;
}}
public class TableCell {
public int row;
public int column;
public TableCell() {
}
public TableCell(int row, int column) {
this.row = row;
this.column = column;
}
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (getClass() != obj.getClass()) {
return false;
}
final TableCell other = (TableCell) obj;
if (this.row != other.row) {
return false;
}
if (this.column != other.column) {
return false;
}
return true;
}
@Override
public int hashCode() {
int hash = 7;
hash = 67 * hash + this.row;
hash = 67 * hash + this.column;
return hash;
}}
public class ActiveJComponentTableCellRenderer<T extends JComponent> extends AbstractCellEditor implements TableCellEditor, TableCellRenderer {
private Map<TableCell, T> components;
private JComponentFactory<T> factory;
public ActiveJComponentTableCellRenderer() {
this.components = new HashMap<TableCell, T>();
}
public ActiveJComponentTableCellRenderer(JComponentFactory<T> factory) {
this();
this.factory = factory;
}
public T getComponent(TableCell key) {
T component = components.get(key);
if (component == null && factory != null) {
// lazy-load component
component = factory.build();
initialiseComponent(component);
components.put(key, component);
}
return component;
}
/**
* Override this method to provide custom component initialisation code
* @param component passed in component from getComponent(cell)
*/
protected void initialiseComponent(T component) {
}
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
return getComponent(new TableCell(row, column));
}
@Override
public Object getCellEditorValue() {
return null;
}
@Override
public Component getTableCellEditorComponent(JTable table, Object value, boolean isSelected, int row, int column) {
return getComponent(new TableCell(row, column));
}
public void setComponentFactory(JComponentFactory factory) {
this.factory = factory;
}}
public interface JComponentFactory<T extends JComponent> {
T build();
}
要使用它,您需要将监听器注册为表格上的鼠标和运动监听器,并在适当的单元格上注册渲染器。如果你想拦截 actionPerformed 类型的事件,像这样覆盖 ActiveJComponentTableCellRenderer.initialiseComponent() :
protected void initialiseComponent(T component) {
component.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
stopCellEditing();
}
});
}
关于java - JTable 中 JEditorPane 中的超链接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1203734/
ChatGUI 我使用 2 个 JEditorPane 将文本从一个 Pane 传输到另一个 Pane 。 一旦我传输了数据,我就会执行以下操作: JEditorPane.setText(null);
我试图在我的应用程序中实现聊天功能。我使用了 2 个 JEditorPane。一个用于保存聊天记录,另一个用于将聊天发送到前一个 JEditorPane。 JEditorPane 是 text/h
如何向 JEditorPane 添加建议列表(当用户键入某个单词的一部分以建议以该部分开头的所有标记的列表,例如 Eclipse 中的智能感知)?我已经实现了返回字符串 ArrayList 的方法,但
我已经在网上搜索了一段时间,但我没有找到任何关于它的真实信息......我想知道无论如何可以使用文件完整目录路径打开文件(如java文件) ? 我目前正在使用 FileReader reader =
我试图每 5 秒刷新一次 JEditorPane,如果新数据添加到数据库表中,它应该得到反射(reflect),但它不会重新绘制。 首先我有一个 JFrame,如下所示,在构造函数内调用 public
我想知道是否有一种方式或方法可以让我获取当前显示的 JEditorPane。例如,我有一个 JFrame,可以在其中创建多个选项卡。每当创建选项卡时,都会创建一个新的 JEditorPane 对象,并
我必须编写基于 JEditorPane 的 WebBrowser 应用程序 - 去学校。 我遇到了一点问题,超链接不起作用。 这是我的代码: public class BrowserFrame ext
在 JEditorPane 中,我想搜索一个字符串,我知道 scrollToReference 不起作用,所以我想知道如何迭代 Pane 并查找指定的字符串。 如何迭代 JEditorPane 来查找
我有一个程序将一些 URL 输出到 JEditorPane。我希望 URL 成为超链接。该程序基本上会将 URLS 输出到 JEditorPane,就好像它是日志一样。 我已经得到了一些工作,但它没有
我无法获得 JEditorPane 以将 HTML img 标记呈现为图像。显示的只是一个占位符图形。下面是我的代码。提前致谢。 我看到的: 我的代码: import java.awt.*; impo
public class EditorPane extends javax.swing.JPanel { /** * Creates new form EditorPane */ public E
谁知道如何更改 JEditorPane 的换行符属性? 我无法在我的 JTextPane 文本中找到换行符(它既不是 \n 也不是 \r),所以我可以' 正确计算此文本的行数。我想更改 \n 的换行符
我想知道我们是否可以使用 sethighlighter(new Highlighter()) 方法将两个荧光笔设置为 JEditorpane,其中一个是默认荧光笔,另一个是下划线荧光笔。 最佳答案 R
在下面的代码中,我想在 JEditorPane 上写入 0 到 10000 的数字。但是,JEditorPane 不显示任何内容,除非它完全完成(一次打印所有 0 到 10000)或者操作系统给它时间
我有这个代码: package web.test; import java.awt.Dimension; import java.io.IOException; import javax.swing.
我在使用 JEditorPane 时遇到问题。我想在同一行上左对齐和右对齐文本。 这是我的代码: INFO_AREA = new JEditorPane(); INFO_AREA.setBor
我正在尝试在 JEditorPane 中使用超链接对于某些单词,类似于单击单词以获取其定义的想法。当在编辑器 Pane 中输入单词时,程序会根据列表检查它们并用 some word 替换列出的单词。标
我正在编写一个使用 JEditorPane 制作简单编辑器的程序,它使用超链接允许用户使用简单的超链接监听器在不同页面之间跳转。 问题是我希望能够让用户选择一些文本并将其转换为链接。我发现有很多这样的
我正在用 Java 创建一个 Web 浏览器。在我使用的这个浏览器中 浏览器窗口的 JEditorPane。我正在使用“setPage(String url)”方法 显示页面。浏览器可以显示页面但是有
我正在尝试在 Java JEditorPane 中显示内联图像。下面的代码使用 HTML 内容,可以在 Firefox 中正确显示图像,但不能在 JEditorPane 中显示。任何想法为什么?谢谢。
我是一名优秀的程序员,十分优秀!