diff --git a/bundles/org.eclipse.jface.text/META-INF/MANIFEST.MF b/bundles/org.eclipse.jface.text/META-INF/MANIFEST.MF
index 23023cb5a81..58e89a789d3 100644
--- a/bundles/org.eclipse.jface.text/META-INF/MANIFEST.MF
+++ b/bundles/org.eclipse.jface.text/META-INF/MANIFEST.MF
@@ -10,6 +10,7 @@ Export-Package:
org.eclipse.jface.contentassist.images,
org.eclipse.jface.internal.text;x-internal:=true,
org.eclipse.jface.internal.text.codemining;x-internal:=true,
+ org.eclipse.jface.internal.text.contentassist;x-internal:=true,
org.eclipse.jface.internal.text.html;x-friends:="org.eclipse.ant.ui, org.eclipse.jdt.ui, org.eclipse.ltk.ui.refactoring, org.eclipse.pde.ui, org.eclipse.ui.editors, org.eclipse.xtext.ui",
org.eclipse.jface.internal.text.link.contentassist;x-internal:=true,
org.eclipse.jface.internal.text.revisions;x-internal:=true,
diff --git a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/TableOwnerDrawSupport.java b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/TableOwnerDrawSupport.java
index a3052c84d42..7a180c7824b 100644
--- a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/TableOwnerDrawSupport.java
+++ b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/TableOwnerDrawSupport.java
@@ -25,6 +25,9 @@
import org.eclipse.swt.widgets.Table;
import org.eclipse.swt.widgets.TableItem;
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
+
/**
* Adds owner draw support for tables.
@@ -42,6 +45,10 @@ public class TableOwnerDrawSupport implements Listener {
public static void install(Table table) {
TableOwnerDrawSupport listener= new TableOwnerDrawSupport(table);
+ installListener(table, listener);
+ }
+
+ protected static void installListener(Table table, Listener listener) {
table.addListener(SWT.Dispose, listener);
table.addListener(SWT.MeasureItem, listener);
table.addListener(SWT.EraseItem, listener);
@@ -70,7 +77,7 @@ private static StyleRange[] getStyledRanges(TableItem item, int column) {
return (StyleRange[])item.getData(STYLED_RANGES_KEY + column);
}
- private TableOwnerDrawSupport(Table table) {
+ protected TableOwnerDrawSupport(Table table) {
int orientation= table.getStyle() & (SWT.LEFT_TO_RIGHT | SWT.RIGHT_TO_LEFT);
fSharedLayout= new TextLayout(table.getDisplay());
fSharedLayout.setOrientation(orientation);
@@ -147,7 +154,28 @@ private void performPaint(Event event) {
Color oldForeground= gc.getForeground();
Color oldBackground= gc.getBackground();
- if (!isSelected) {
+ if (isSelected) {
+ Color background= item.getParent().isFocusControl()
+ ? getSelectedRowBackgroundColor()
+ : getSelectedRowBackgroundColorNoFocus();
+ Color foreground= item.getParent().isFocusControl()
+ ? getSelectedRowForegroundColor()
+ : getSelectedRowForegroundColorNoFocus();
+
+ if (background == null) {
+ background= item.getDisplay().getSystemColor(
+ SWT.COLOR_LIST_SELECTION);
+ }
+
+ if (foreground == null) {
+ foreground= item.getDisplay().getSystemColor(
+ SWT.COLOR_LIST_SELECTION_TEXT);
+ }
+
+ gc.setBackground(background);
+ gc.setForeground(foreground);
+ gc.fillRectangle(0, event.y, item.getParent().getBounds().width, event.height);
+ } else {
Color foreground= item.getForeground(index);
gc.setForeground(foreground);
@@ -178,10 +206,54 @@ private void performPaint(Event event) {
gc.drawFocus(focusBounds.x, focusBounds.y, focusBounds.width + fDeltaOfLastMeasure, focusBounds.height);
}
- if (!isSelected) {
- gc.setForeground(oldForeground);
- gc.setBackground(oldBackground);
- }
+ gc.setForeground(oldForeground);
+ gc.setBackground(oldBackground);
+ }
+
+ /**
+ * The color to use when rendering the background of the selected row when the control has the
+ * input focus
+ *
+ * @return the color or null
to use the default
+ */
+ protected Color getSelectedRowBackgroundColor() {
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND"); //$NON-NLS-1$
+ }
+
+ /**
+ * The color to use when rendering the foreground (=text) of the selected row when the control
+ * has the input focus
+ *
+ * @return the color or null
to use the default
+ */
+ protected Color getSelectedRowForegroundColor() {
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND"); //$NON-NLS-1$
+ }
+
+ /**
+ * The color to use when rendering the foreground (=text) of the selected row when the control
+ * has no input focus
+ *
+ * @return the color or null
to use the same used when control has focus
+ * @since 3.4
+ */
+ protected Color getSelectedRowForegroundColorNoFocus() {
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS"); //$NON-NLS-1$
+ }
+
+ /**
+ * The color to use when rendering the background of the selected row when the control has
+ * no input focus
+ *
+ * @return the color or null
to use the same used when control has focus
+ * @since 3.4
+ */
+ protected Color getSelectedRowBackgroundColorNoFocus() {
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS"); //$NON-NLS-1$
}
private void widgetDisposed() {
diff --git a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupport.java b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupport.java
new file mode 100644
index 00000000000..611397a4723
--- /dev/null
+++ b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupport.java
@@ -0,0 +1,63 @@
+/*******************************************************************************
+ * Copyright (c) 2024 SAP SE.
+ *
+ * This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License 2.0
+ * which accompanies this distribution, and is available at
+ * https://www.eclipse.org/legal/epl-2.0/
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ *
+ * Contributors:
+ * SAP SE - initial API and implementation
+ *******************************************************************************/
+package org.eclipse.jface.internal.text.contentassist;
+
+import org.eclipse.swt.custom.StyleRange;
+import org.eclipse.swt.graphics.Color;
+import org.eclipse.swt.widgets.Table;
+import org.eclipse.swt.widgets.TableItem;
+
+import org.eclipse.jface.internal.text.TableOwnerDrawSupport;
+
+/**
+ * When a completion table (for example for code completion) is requested by the user, the user
+ * needs to be able to continue typing in the linked text control. In such cases, the focus is not
+ * on the completion table. To ensure the selected code completion proposal always displays in the
+ * correct color, even if the completion table is not focused, the non-focused colors are overridden
+ * with the focus colors.
+ */
+public class CompletionTableDrawSupport extends TableOwnerDrawSupport {
+
+ public static void install(Table table) {
+ TableOwnerDrawSupport listener= new CompletionTableDrawSupport(table);
+ installListener(table, listener);
+ }
+
+ /**
+ * Stores the styled ranges in the given table item. See {@link TableOwnerDrawSupport}
+ *
+ * @param item table item
+ * @param column the column index
+ * @param ranges the styled ranges or null
to remove them
+ */
+ public static void storeStyleRanges(TableItem item, int column, StyleRange[] ranges) {
+ TableOwnerDrawSupport.storeStyleRanges(item, column, ranges);
+ }
+
+
+ private CompletionTableDrawSupport(Table table) {
+ super(table);
+ }
+
+ @Override
+ protected Color getSelectedRowBackgroundColorNoFocus() {
+ return super.getSelectedRowBackgroundColor();
+ }
+
+ @Override
+ protected Color getSelectedRowForegroundColorNoFocus() {
+ return super.getSelectedRowForegroundColor();
+ }
+
+}
diff --git a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/link/contentassist/CompletionProposalPopup2.java b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/link/contentassist/CompletionProposalPopup2.java
index 6ca009a07d7..067e233936f 100644
--- a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/link/contentassist/CompletionProposalPopup2.java
+++ b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/internal/text/link/contentassist/CompletionProposalPopup2.java
@@ -39,7 +39,7 @@
import org.eclipse.swt.widgets.Table;
import org.eclipse.swt.widgets.TableItem;
-import org.eclipse.jface.internal.text.TableOwnerDrawSupport;
+import org.eclipse.jface.internal.text.contentassist.CompletionTableDrawSupport;
import org.eclipse.jface.preference.JFacePreferences;
import org.eclipse.jface.resource.ColorRegistry;
import org.eclipse.jface.resource.JFaceColors;
@@ -269,7 +269,7 @@ private void createProposalSelector() {
fIsColoredLabelsSupportEnabled= fContentAssistant.isColoredLabelsSupportEnabled();
if (fIsColoredLabelsSupportEnabled)
- TableOwnerDrawSupport.install(fProposalTable);
+ CompletionTableDrawSupport.install(fProposalTable);
fProposalTable.setLocation(0, 0);
if (fAdditionalInfoController != null)
@@ -572,7 +572,7 @@ private void setProposals(ICompletionProposal[] proposals) {
item.setText(displayString);
if (fIsColoredLabelsSupportEnabled)
- TableOwnerDrawSupport.storeStyleRanges(item, 0, styleRanges);
+ CompletionTableDrawSupport.storeStyleRanges(item, 0, styleRanges);
item.setData(p);
diff --git a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/text/contentassist/CompletionProposalPopup.java b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/text/contentassist/CompletionProposalPopup.java
index f426f8b7ce8..7297722a77b 100644
--- a/bundles/org.eclipse.jface.text/src/org/eclipse/jface/text/contentassist/CompletionProposalPopup.java
+++ b/bundles/org.eclipse.jface.text/src/org/eclipse/jface/text/contentassist/CompletionProposalPopup.java
@@ -78,7 +78,7 @@
import org.eclipse.jface.bindings.keys.SWTKeySupport;
import org.eclipse.jface.contentassist.IContentAssistSubjectControl;
import org.eclipse.jface.internal.text.InformationControlReplacer;
-import org.eclipse.jface.internal.text.TableOwnerDrawSupport;
+import org.eclipse.jface.internal.text.contentassist.CompletionTableDrawSupport;
import org.eclipse.jface.preference.JFacePreferences;
import org.eclipse.jface.resource.JFaceColors;
import org.eclipse.jface.resource.JFaceResources;
@@ -613,7 +613,7 @@ void createProposalSelector() {
fIsColoredLabelsSupportEnabled= fContentAssistant.isColoredLabelsSupportEnabled();
if (fIsColoredLabelsSupportEnabled)
- TableOwnerDrawSupport.install(fProposalTable);
+ CompletionTableDrawSupport.install(fProposalTable);
fProposalTable.setLocation(0, 0);
if (fAdditionalInfoController != null)
@@ -904,7 +904,7 @@ private void handleSetData(Event event) {
item.setText(displayString);
if (fIsColoredLabelsSupportEnabled)
- TableOwnerDrawSupport.storeStyleRanges(item, 0, styleRanges);
+ CompletionTableDrawSupport.storeStyleRanges(item, 0, styleRanges);
item.setImage(image);
item.setData(current);
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/FocusCellOwnerDrawHighlighter.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/FocusCellOwnerDrawHighlighter.java
index 99925bdc6f9..5455e92aac3 100644
--- a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/FocusCellOwnerDrawHighlighter.java
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/FocusCellOwnerDrawHighlighter.java
@@ -18,6 +18,8 @@
import org.eclipse.core.runtime.Assert;
import org.eclipse.jface.util.Util;
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
import org.eclipse.swt.SWT;
import org.eclipse.swt.graphics.Color;
import org.eclipse.swt.graphics.GC;
@@ -151,8 +153,8 @@ private void hookListener(final ColumnViewer viewer) {
* @return the color or null
to use the default
*/
protected Color getSelectedCellBackgroundColor(ViewerCell cell) {
- return removeNonFocusedSelectionInformation ? null
- : cell.getItem().getDisplay().getSystemColor(SWT.COLOR_LIST_SELECTION);
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND"); //$NON-NLS-1$
}
/**
@@ -164,7 +166,8 @@ protected Color getSelectedCellBackgroundColor(ViewerCell cell) {
* @return the color or null
to use the default
*/
protected Color getSelectedCellForegroundColor(ViewerCell cell) {
- return null;
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND"); //$NON-NLS-1$
}
/**
@@ -178,7 +181,8 @@ protected Color getSelectedCellForegroundColor(ViewerCell cell) {
* @since 3.4
*/
protected Color getSelectedCellForegroundColorNoFocus(ViewerCell cell) {
- return null;
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS"); //$NON-NLS-1$
}
/**
@@ -192,7 +196,8 @@ protected Color getSelectedCellForegroundColorNoFocus(ViewerCell cell) {
* @since 3.4
*/
protected Color getSelectedCellBackgroundColorNoFocus(ViewerCell cell) {
- return null;
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ return colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS"); //$NON-NLS-1$
}
/**
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/OwnerDrawLabelProvider.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/OwnerDrawLabelProvider.java
index e1710179f94..2454eda1613 100644
--- a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/OwnerDrawLabelProvider.java
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/OwnerDrawLabelProvider.java
@@ -17,6 +17,8 @@
import java.util.HashSet;
import java.util.Set;
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
import org.eclipse.swt.SWT;
import org.eclipse.swt.graphics.Color;
import org.eclipse.swt.graphics.Rectangle;
@@ -166,20 +168,16 @@ public void update(ViewerCell cell) {
}
/**
- * Handle the erase event. The default implementation colors the background
- * of selected areas with {@link SWT#COLOR_LIST_SELECTION} and foregrounds
- * with {@link SWT#COLOR_LIST_SELECTION_TEXT}. Note that this
- * implementation causes non-native behavior on some platforms. Subclasses
- * should override this method and not call the super
+ * Handle the erase event. The default implementation colors the background of
+ * selected areas with "org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND" and
+ * foregrounds with "org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND". Note
+ * that this implementation causes non-native behavior on some platforms.
+ * Subclasses should override this method and not call the super
* implementation.
*
- * @param event
- * the erase event
- * @param element
- * the model object
+ * @param event the erase event
+ * @param element the model object
* @see SWT#EraseItem
- * @see SWT#COLOR_LIST_SELECTION
- * @see SWT#COLOR_LIST_SELECTION_TEXT
*/
protected void erase(Event event, Object element) {
@@ -189,11 +187,16 @@ protected void erase(Event event, Object element) {
Color oldForeground = event.gc.getForeground();
Color oldBackground = event.gc.getBackground();
- event.gc.setBackground(event.item.getDisplay().getSystemColor(
- SWT.COLOR_LIST_SELECTION));
- event.gc.setForeground(event.item.getDisplay().getSystemColor(
- SWT.COLOR_LIST_SELECTION_TEXT));
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ if (event.widget instanceof Control control && control.isFocusControl()) {
+ event.gc.setBackground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND")); //$NON-NLS-1$
+ event.gc.setForeground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND")); //$NON-NLS-1$
+ } else {
+ event.gc.setBackground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS")); //$NON-NLS-1$
+ event.gc.setForeground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS")); //$NON-NLS-1$
+ }
event.gc.fillRectangle(bounds);
+
/* restore the old GC colors */
event.gc.setForeground(oldForeground);
event.gc.setBackground(oldBackground);
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/StyledCellLabelProvider.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/StyledCellLabelProvider.java
index 4d1872cf13b..1b29e15c69e 100644
--- a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/StyledCellLabelProvider.java
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/StyledCellLabelProvider.java
@@ -263,6 +263,7 @@ protected void erase(Event event, Object element) {
// info has been set by 'update': announce that we paint ourselves
event.detail &= ~SWT.FOREGROUND;
}
+ super.erase(event, element);
}
@Override
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TableViewer.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TableViewer.java
index 8966573b3f3..6baf6b0e07a 100644
--- a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TableViewer.java
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TableViewer.java
@@ -20,6 +20,7 @@
package org.eclipse.jface.viewers;
import org.eclipse.core.runtime.Assert;
+import org.eclipse.jface.viewers.internal.ColumnViewerSelectionColorListener;
import org.eclipse.jface.viewers.internal.ExpandableNode;
import org.eclipse.pde.api.tools.annotations.NoExtend;
import org.eclipse.swt.SWT;
@@ -119,6 +120,7 @@ public TableViewer(Composite parent, int style) {
public TableViewer(Table table) {
this.table = table;
hookControl(table);
+ overwriteSelectionColor();
}
@Override
@@ -507,4 +509,13 @@ void handleExpandableNodeClicked(Widget w) {
}
}
+ /**
+ * The color of the selected item is drawn by the OS. On some OS the color might
+ * be not accessible. To fix this issue the background color for selected items
+ * is drawn in a custom method.
+ */
+ private void overwriteSelectionColor() {
+ ColumnViewerSelectionColorListener.addListenerToViewer(this);
+ }
+
}
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TreeViewer.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TreeViewer.java
index 55eae5315c0..5f86b75e1ac 100644
--- a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TreeViewer.java
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/TreeViewer.java
@@ -25,6 +25,7 @@
import java.util.List;
import org.eclipse.jface.util.Policy;
+import org.eclipse.jface.viewers.internal.ColumnViewerSelectionColorListener;
import org.eclipse.jface.viewers.internal.ExpandableNode;
import org.eclipse.pde.api.tools.annotations.NoExtend;
import org.eclipse.swt.SWT;
@@ -137,6 +138,7 @@ public TreeViewer(Tree tree) {
super();
this.tree = tree;
hookControl(tree);
+ overwriteSelectionColor();
}
@Override
@@ -1188,4 +1190,13 @@ Object getLastElement(Widget parent) {
}
return items[length - 1].getData();
}
+
+ /**
+ * The color of the selected item is drawn by the OS. On some OS the color might
+ * be not accessible. To fix this issue the background color for selected items
+ * is drawn in a custom method.
+ */
+ private void overwriteSelectionColor() {
+ ColumnViewerSelectionColorListener.addListenerToViewer(this);
+ }
}
diff --git a/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/internal/ColumnViewerSelectionColorListener.java b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/internal/ColumnViewerSelectionColorListener.java
new file mode 100644
index 00000000000..fbc6e7ea40a
--- /dev/null
+++ b/bundles/org.eclipse.jface/src/org/eclipse/jface/viewers/internal/ColumnViewerSelectionColorListener.java
@@ -0,0 +1,83 @@
+/*******************************************************************************
+ * Copyright (c) 2024 SAP SE.
+ *
+ * This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License 2.0
+ * which accompanies this distribution, and is available at
+ * https://www.eclipse.org/legal/epl-2.0/
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ *
+ * Contributors:
+ * SAP SE - initial API and implementation
+ *******************************************************************************/
+package org.eclipse.jface.viewers.internal;
+
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
+import org.eclipse.jface.viewers.FocusCellOwnerDrawHighlighter;
+import org.eclipse.jface.viewers.StructuredViewer;
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.graphics.GC;
+import org.eclipse.swt.widgets.Control;
+import org.eclipse.swt.widgets.Event;
+import org.eclipse.swt.widgets.Listener;
+import org.eclipse.swt.widgets.Scrollable;
+
+/**
+ * EraseItem event listener that takes care of coloring the selection color of
+ * column viewers. The coloring is only applied if no other erase item listener
+ * is registered for the viewer. If other erase item listeners are registers,
+ * most probably a other customer coloring is applied and should not be
+ * overwritten.
+ *
+ * @see FocusCellOwnerDrawHighlighter
+ */
+public class ColumnViewerSelectionColorListener implements Listener {
+
+ /**
+ * Registers an erase item event listener that takes care of coloring the
+ * selection color of the given viewer.
+ *
+ * @param viewer The viewer that should be colored
+ */
+ public static void addListenerToViewer(StructuredViewer viewer) {
+ viewer.getControl().addListener(SWT.EraseItem, new ColumnViewerSelectionColorListener());
+ }
+
+ @Override
+ public void handleEvent(Event event) {
+ if ((event.detail & SWT.SELECTED) == 0) {
+ return; /* item not selected */
+ }
+
+ if (event.widget instanceof Control control && !control.isEnabled()) {
+ return; /* item is disabled, no coloring required */
+ }
+
+ Listener[] eraseItemListeners = event.widget.getListeners(SWT.EraseItem);
+ if (eraseItemListeners.length != 1) {
+ return; /* other eraseItemListener exists, do not apply coloring */
+ }
+
+ GC gc = event.gc;
+ ColorRegistry colorRegistry = JFaceResources.getColorRegistry();
+ if (event.widget instanceof Control control && control.isFocusControl()) {
+ gc.setBackground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND")); //$NON-NLS-1$
+ gc.setForeground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND")); //$NON-NLS-1$
+ } else {
+ gc.setBackground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS")); //$NON-NLS-1$
+ gc.setForeground(colorRegistry.get("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS")); //$NON-NLS-1$
+ }
+
+ int width = event.width;
+ if (event.widget instanceof Scrollable scrollable) {
+ width = scrollable.getClientArea().width;
+ }
+
+ gc.fillRectangle(0, event.y, width, event.height);
+
+ event.detail &= ~SWT.SELECTED;
+ }
+
+}
diff --git a/bundles/org.eclipse.ui.themes/css/dark/e4-dark_preferencestyle.css b/bundles/org.eclipse.ui.themes/css/dark/e4-dark_preferencestyle.css
index b9adbc6bbb2..4fda7024e51 100644
--- a/bundles/org.eclipse.ui.themes/css/dark/e4-dark_preferencestyle.css
+++ b/bundles/org.eclipse.ui.themes/css/dark/e4-dark_preferencestyle.css
@@ -65,6 +65,8 @@ IEclipsePreferences#org-eclipse-ui-workbench:org-eclipse-ui-themes { /* pseudo a
'org.eclipse.ui.workbench.FORM_HEADING_ERROR_COLOR=255,110,110'
'org.eclipse.ui.workbench.FORM_HEADING_WARNING_COLOR=255,200,0'
'org.eclipse.ui.workbench.FORM_HEADING_INFO_COLOR=170,170,170'
+ 'org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS=240,240,240'
+ 'org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS=95,95,95'
'ERROR_COLOR=247,68,117'
'HYPERLINK_COLOR=111,197,238'
'INCOMING_COLOR=31,179,235'
diff --git a/bundles/org.eclipse.ui.themes/css/e4-dark_win.css b/bundles/org.eclipse.ui.themes/css/e4-dark_win.css
index 0b0afa540d0..a61c5268db1 100644
--- a/bundles/org.eclipse.ui.themes/css/e4-dark_win.css
+++ b/bundles/org.eclipse.ui.themes/css/e4-dark_win.css
@@ -135,3 +135,9 @@ ImageBasedFrame,
color: #DDDDDD;
}
+IEclipsePreferences#org-eclipse-ui-workbench:org-eclipse-ui-themes-win { /* pseudo attribute added to allow contributions without replacing this node, see Bug 466075 */
+ preferences:
+ 'org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND=255,255,255'
+ 'org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND=0,100,255'
+}
+
diff --git a/bundles/org.eclipse.ui.workbench/Eclipse UI/org/eclipse/ui/internal/themes/ColumnViewerSelectionColorFactory.java b/bundles/org.eclipse.ui.workbench/Eclipse UI/org/eclipse/ui/internal/themes/ColumnViewerSelectionColorFactory.java
new file mode 100644
index 00000000000..a2e852d3458
--- /dev/null
+++ b/bundles/org.eclipse.ui.workbench/Eclipse UI/org/eclipse/ui/internal/themes/ColumnViewerSelectionColorFactory.java
@@ -0,0 +1,63 @@
+/*******************************************************************************
+ * Copyright (c) 2024 SAP SE.
+ *
+ * This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License 2.0
+ * which accompanies this distribution, and is available at
+ * https://www.eclipse.org/legal/epl-2.0/
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ *
+ * Contributors:
+ * SAP SE - initial API and implementation
+ *******************************************************************************/
+package org.eclipse.ui.internal.themes;
+
+import java.util.Hashtable;
+import org.eclipse.core.runtime.IConfigurationElement;
+import org.eclipse.core.runtime.IExecutableExtension;
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.graphics.RGB;
+import org.eclipse.swt.widgets.Display;
+import org.eclipse.ui.themes.IColorFactory;
+
+/**
+ * The factory to provide the default colors for viewer selections. The default
+ * color is based on the system color for the title, which is the actual
+ * highlight color. The color can be overwritten via the color preferences.
+ *
+ * @since 3.5
+ *
+ */
+public class ColumnViewerSelectionColorFactory implements IColorFactory, IExecutableExtension {
+
+ private String color = null;
+
+ @Override
+ public RGB createColor() {
+ if ("SELECTED_CELL_BACKGROUND".equals(color)) { //$NON-NLS-1$
+ return Display.getDefault().getSystemColor(SWT.COLOR_TITLE_BACKGROUND).getRGB();
+
+ } else if ("SELECTED_CELL_FOREGROUND".equals(color)) { //$NON-NLS-1$
+ // Selection foreground should be white if not changed by the user
+ return Display.getDefault().getSystemColor(SWT.COLOR_WHITE).getRGB();
+
+ } else if ("SELECTED_CELL_BACKGROUND_NO_FOCUS".equals(color)) { //$NON-NLS-1$
+ return Display.getDefault().getSystemColor(SWT.COLOR_TITLE_INACTIVE_BACKGROUND).getRGB();
+
+ } else if ("SELECTED_CELL_FOREGROUND_NO_FOCUS".equals(color)) { //$NON-NLS-1$
+ return Display.getDefault().getSystemColor(SWT.COLOR_TITLE_INACTIVE_FOREGROUND).getRGB();
+
+ } else {
+ return null;
+ }
+ }
+
+ @Override
+ public void setInitializationData(IConfigurationElement config, String propertyName, Object data) {
+ if (data instanceof Hashtable table) {
+ this.color = (String) table.get("color"); //$NON-NLS-1$
+ }
+ }
+
+}
\ No newline at end of file
diff --git a/bundles/org.eclipse.ui/plugin.properties b/bundles/org.eclipse.ui/plugin.properties
index 525c2caece2..af2b0558d89 100644
--- a/bundles/org.eclipse.ui/plugin.properties
+++ b/bundles/org.eclipse.ui/plugin.properties
@@ -374,6 +374,15 @@ Color.showKeysForegroundDesc=Color for foreground of the control to visualize pr
Color.showKeysBackground=Keys background color
Color.showKeysBackgroundDesc=Color for background of the control to visualize pressed keys
+Color.selectedCellForeground=Selection foreground color for cell viewer
+Color.selectedCellForegroundDesc=The selection foreground color of a cell viewer
+Color.selectedCellBackground=Selection background color for cell viewer
+Color.selectedCellBackgroundDesc=The selection background color of a cell viewer
+Color.selectedCellForegroundNoFocus=Selection foreground color for cell viewer without focus
+Color.selectedCellForegroundNoFocusDesc=The selection foreground color of a cell viewer without focus
+Color.selectedCellBackgroundNoFocus=Selection background color for cell viewer without focus
+Color.selectedCellBackgroundNoFocusDesc=The selection background color of a cell viewer without focus
+
ThemeName.SystemDefault = Reduced Palette
HighContrast.ThemeDescription = A theme that takes all of its values from the system settings.
diff --git a/bundles/org.eclipse.ui/plugin.xml b/bundles/org.eclipse.ui/plugin.xml
index f4523fefbd8..89153764ace 100644
--- a/bundles/org.eclipse.ui/plugin.xml
+++ b/bundles/org.eclipse.ui/plugin.xml
@@ -2054,6 +2054,74 @@
%Color.showKeysBackgroundDesc
+
+
+ %Color.selectedCellForegroundDesc
+
+
+
+
+
+
+
+
+ %Color.selectedCellBackgroundDesc
+
+
+
+
+
+
+
+
+ %Color.selectedCellForegroundNoFocusDesc
+
+
+
+
+
+
+
+
+ %Color.selectedCellBackgroundNoFocusDesc
+
+
+
+
+
+
diff --git a/tests/org.eclipse.jface.text.tests/META-INF/MANIFEST.MF b/tests/org.eclipse.jface.text.tests/META-INF/MANIFEST.MF
index fca4c84b39c..eca971f2fee 100644
--- a/tests/org.eclipse.jface.text.tests/META-INF/MANIFEST.MF
+++ b/tests/org.eclipse.jface.text.tests/META-INF/MANIFEST.MF
@@ -22,7 +22,8 @@ Require-Bundle:
org.eclipse.text.tests;bundle-version="[3.5.0,4.0.0)",
org.eclipse.core.runtime;bundle-version="[3.29.0,4.0.0)",
org.eclipse.ui.workbench.texteditor,
- org.eclipse.test.performance;bundle-version="3.13.0"
+ org.eclipse.test.performance;bundle-version="3.13.0",
+ org.mockito.mockito-core;bundle-version="5.12.0"
Bundle-RequiredExecutionEnvironment: JavaSE-17
Eclipse-BundleShape: dir
Automatic-Module-Name: org.eclipse.jface.text.tests
diff --git a/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/TableOwnerDrawSupportTest.java b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/TableOwnerDrawSupportTest.java
new file mode 100644
index 00000000000..558c5c9aaeb
--- /dev/null
+++ b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/TableOwnerDrawSupportTest.java
@@ -0,0 +1,93 @@
+package org.eclipse.jface.internal.text;
+
+import static org.mockito.Mockito.doReturn;
+
+import org.junit.Before;
+import org.junit.Test;
+import org.mockito.Mockito;
+
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.graphics.Color;
+import org.eclipse.swt.graphics.GC;
+import org.eclipse.swt.graphics.RGB;
+import org.eclipse.swt.widgets.Display;
+import org.eclipse.swt.widgets.Event;
+import org.eclipse.swt.widgets.Shell;
+import org.eclipse.swt.widgets.Table;
+import org.eclipse.swt.widgets.TableItem;
+
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
+
+public class TableOwnerDrawSupportTest {
+
+ private static final int EVENT_Y= 2;
+
+ private static final int EVENT_HEIGHT= 3;
+
+ private Table table;
+
+ private Event event;
+
+ private RGB selectedRowBackgroundColor= new RGB(0, 0, 0);
+
+ private RGB selectedRowForegroundColor= new RGB(1, 1, 1);
+
+ private RGB selectedRowBackgroundColorNoFocus= new RGB(2, 2, 2);
+
+ private RGB selectedRowForegroundColorNoFocus= new RGB(3, 3, 3);
+
+ private GC gc;
+
+ @Before
+ public void setup() {
+ Shell shell= new Shell(Display.getDefault());
+ Table originalTable= new Table(shell, SWT.SINGLE);
+ //Spy in order to overwrite isFocusControl()
+ table= Mockito.spy(originalTable);
+
+ TableItem tableItem= Mockito.mock(TableItem.class);
+ Mockito.when(tableItem.getParent()).thenReturn(table);
+
+ gc= Mockito.mock(GC.class);
+ event= new Event();
+ event.type= SWT.PaintItem;
+ event.gc= gc;
+ event.detail= SWT.SELECTED;
+ event.item= tableItem;
+ event.y= EVENT_Y;
+ event.height= EVENT_HEIGHT;
+
+ Display.getDefault();
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND", new RGB(0, 0, 0));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND", new RGB(1, 1, 1));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS", new RGB(2, 2, 2));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS", new RGB(3, 3, 3));
+
+ TableOwnerDrawSupport.install(table);
+ }
+
+ @Test
+ public void testPaintSelectionFocus() {
+ doReturn(true).when(table).isFocusControl();
+
+ table.notifyListeners(SWT.PaintItem, event);
+
+ Mockito.verify(gc).setBackground(new Color(selectedRowBackgroundColor));
+ Mockito.verify(gc).setForeground(new Color(selectedRowForegroundColor));
+ Mockito.verify(gc).fillRectangle(0, EVENT_Y, table.getBounds().width, EVENT_HEIGHT);
+ }
+
+ @Test
+ public void testPaintSelectionNoFocus() {
+ doReturn(false).when(table).isFocusControl();
+
+ table.notifyListeners(SWT.PaintItem, event);
+
+ Mockito.verify(gc).setBackground(new Color(selectedRowBackgroundColorNoFocus));
+ Mockito.verify(gc).setForeground(new Color(selectedRowForegroundColorNoFocus));
+ Mockito.verify(gc).fillRectangle(0, EVENT_Y, table.getBounds().width, EVENT_HEIGHT);
+ }
+
+}
diff --git a/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupportTest.java b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupportTest.java
new file mode 100644
index 00000000000..b89a9833bfc
--- /dev/null
+++ b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/internal/text/contentassist/CompletionTableDrawSupportTest.java
@@ -0,0 +1,92 @@
+package org.eclipse.jface.internal.text.contentassist;
+
+import static org.hamcrest.CoreMatchers.instanceOf;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.collection.IsArrayWithSize.arrayWithSize;
+import static org.hamcrest.collection.IsArrayWithSize.emptyArray;
+import static org.junit.Assert.assertEquals;
+
+import org.junit.Test;
+import org.mockito.Mockito;
+
+import org.eclipse.swt.SWT;
+import org.eclipse.swt.custom.StyleRange;
+import org.eclipse.swt.graphics.Color;
+import org.eclipse.swt.graphics.GC;
+import org.eclipse.swt.graphics.RGB;
+import org.eclipse.swt.widgets.Event;
+import org.eclipse.swt.widgets.Shell;
+import org.eclipse.swt.widgets.Table;
+import org.eclipse.swt.widgets.TableItem;
+
+import org.eclipse.jface.internal.text.TableOwnerDrawSupport;
+import org.eclipse.jface.resource.ColorRegistry;
+import org.eclipse.jface.resource.JFaceResources;
+
+public class CompletionTableDrawSupportTest {
+
+ @Test
+ public void testInstall() {
+ Shell shell= new Shell();
+ Table table= new Table(shell, SWT.NONE);
+
+ assertThat(table.getListeners(SWT.MeasureItem), emptyArray());
+ assertThat(table.getListeners(SWT.EraseItem), emptyArray());
+ assertThat(table.getListeners(SWT.PaintItem), emptyArray());
+
+ CompletionTableDrawSupport.install(table);
+
+ assertThat(table.getListeners(SWT.MeasureItem), arrayWithSize(1));
+ assertThat(table.getListeners(SWT.MeasureItem)[0], instanceOf(TableOwnerDrawSupport.class));
+ assertThat(table.getListeners(SWT.EraseItem), arrayWithSize(1));
+ assertThat(table.getListeners(SWT.EraseItem)[0], instanceOf(TableOwnerDrawSupport.class));
+ assertThat(table.getListeners(SWT.PaintItem), arrayWithSize(1));
+ assertThat(table.getListeners(SWT.PaintItem)[0], instanceOf(TableOwnerDrawSupport.class));
+ }
+
+ @Test
+ public void testStoreStyleRanges() {
+ Shell shell= new Shell();
+ Table table= new Table(shell, SWT.NONE);
+ TableItem tableItem= new TableItem(table, SWT.NONE);
+ StyleRange[] ranges= new StyleRange[] {};
+
+ CompletionTableDrawSupport.storeStyleRanges(tableItem, 2, ranges);
+
+ assertEquals(ranges, tableItem.getData("styled_ranges2"));
+ }
+
+ @Test
+ public void testPaintNonFocusSelectionInFocusColors() {
+ int EVENT_Y= 2;
+ int EVENT_HEIGHT= 3;
+
+ Shell shell= new Shell();
+ Table table= new Table(shell, SWT.NONE);
+ TableItem tableItem= new TableItem(table, SWT.NONE);
+
+ ColorRegistry colorRegistry= JFaceResources.getColorRegistry();
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND", new RGB(0, 0, 0));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND", new RGB(1, 1, 1));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_BACKGROUND_NO_FOCUS", new RGB(2, 2, 2));
+ colorRegistry.put("org.eclipse.ui.workbench.SELECTED_CELL_FOREGROUND_NO_FOCUS", new RGB(3, 3, 3));
+
+ GC gc= Mockito.mock(GC.class);
+ Event event= new Event();
+ event.type= SWT.PaintItem;
+ event.gc= gc;
+ event.detail= SWT.SELECTED;
+ event.item= tableItem;
+ event.y= EVENT_Y;
+ event.height= EVENT_HEIGHT;
+
+ CompletionTableDrawSupport.install(table);
+
+ table.notifyListeners(SWT.PaintItem, event);
+
+ Mockito.verify(gc).setBackground(new Color(new RGB(0, 0, 0)));
+ Mockito.verify(gc).setForeground(new Color(new RGB(1, 1, 1)));
+ Mockito.verify(gc).fillRectangle(0, EVENT_Y, table.getBounds().width, EVENT_HEIGHT);
+ }
+
+}
diff --git a/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/text/tests/JFaceTextTestSuite.java b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/text/tests/JFaceTextTestSuite.java
index 1b190e48845..cc939d99d56 100644
--- a/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/text/tests/JFaceTextTestSuite.java
+++ b/tests/org.eclipse.jface.text.tests/src/org/eclipse/jface/text/tests/JFaceTextTestSuite.java
@@ -17,6 +17,9 @@
import org.junit.runners.Suite;
import org.junit.runners.Suite.SuiteClasses;
+import org.eclipse.jface.internal.text.TableOwnerDrawSupportTest;
+import org.eclipse.jface.internal.text.contentassist.CompletionTableDrawSupportTest;
+
import org.eclipse.jface.text.tests.codemining.CodeMiningProjectionViewerTest;
import org.eclipse.jface.text.tests.codemining.CodeMiningTest;
import org.eclipse.jface.text.tests.contentassist.AsyncContentAssistTest;
@@ -80,7 +83,10 @@
DefaultTextDoubleClickStrategyTest.class,
MultiSelectionTest.class,
FindReplaceDocumentAdapterContentProposalProviderTest.class,
- ProjectionViewerTest.class
+ ProjectionViewerTest.class,
+
+ TableOwnerDrawSupportTest.class,
+ CompletionTableDrawSupportTest.class
})
public class JFaceTextTestSuite {
// see @SuiteClasses