4783068: Components with HTML text should gray out the text when disabled

Views fixed to use different colors when container is disabled

Reviewed-by: gsm, rupashka
This commit is contained in:
Peter Zhelezniakov 2009-03-23 16:41:47 +03:00
parent 4ce28ac8a0
commit 1a1db06a37
4 changed files with 162 additions and 56 deletions

View File

@ -30,6 +30,7 @@ import javax.swing.event.*;
import java.util.BitSet; import java.util.BitSet;
import java.util.Locale; import java.util.Locale;
import javax.swing.UIManager;
import sun.swing.SwingUtilities2; import sun.swing.SwingUtilities2;
/** /**
@ -382,11 +383,10 @@ public class GlyphView extends View implements TabableView, Cloneable {
Color bg = getBackground(); Color bg = getBackground();
Color fg = getForeground(); Color fg = getForeground();
if (c instanceof JTextComponent) { if (c != null && ! c.isEnabled()) {
JTextComponent tc = (JTextComponent) c; fg = (c instanceof JTextComponent ?
if (!tc.isEnabled()) { ((JTextComponent)c).getDisabledTextColor() :
fg = tc.getDisabledTextColor(); UIManager.getColor("textInactiveText"));
}
} }
if (bg != null) { if (bg != null) {
g.setColor(bg); g.setColor(bg);

View File

@ -25,9 +25,7 @@
package javax.swing.text.html; package javax.swing.text.html;
import java.awt.*; import java.awt.*;
import java.awt.event.*;
import java.awt.image.ImageObserver; import java.awt.image.ImageObserver;
import java.io.*;
import java.net.*; import java.net.*;
import java.util.Dictionary; import java.util.Dictionary;
import javax.swing.*; import javax.swing.*;
@ -97,6 +95,7 @@ public class ImageView extends View {
private AttributeSet attr; private AttributeSet attr;
private Image image; private Image image;
private Image disabledImage;
private int width; private int width;
private int height; private int height;
/** Bitmask containing some of the above bitmask values. Because the /** Bitmask containing some of the above bitmask values. Because the
@ -193,6 +192,17 @@ public class ImageView extends View {
return image; return image;
} }
private Image getImage(boolean enabled) {
Image img = getImage();
if (! enabled) {
if (disabledImage == null) {
disabledImage = GrayFilter.createDisabledImage(img);
}
img = disabledImage;
}
return img;
}
/** /**
* Sets how the image is loaded. If <code>newValue</code> is true, * Sets how the image is loaded. If <code>newValue</code> is true,
* the image we be loaded when first asked for, otherwise it will * the image we be loaded when first asked for, otherwise it will
@ -338,8 +348,6 @@ public class ImageView extends View {
Rectangle rect = (a instanceof Rectangle) ? (Rectangle)a : Rectangle rect = (a instanceof Rectangle) ? (Rectangle)a :
a.getBounds(); a.getBounds();
Image image = getImage();
Rectangle clip = g.getClipBounds(); Rectangle clip = g.getClipBounds();
fBounds.setBounds(rect); fBounds.setBounds(rect);
@ -350,29 +358,29 @@ public class ImageView extends View {
rect.width - leftInset - rightInset, rect.width - leftInset - rightInset,
rect.height - topInset - bottomInset); rect.height - topInset - bottomInset);
} }
if (image != null) {
if (!hasPixels(image)) {
// No pixels yet, use the default
Icon icon = (image == null) ? getNoImageIcon() :
getLoadingImageIcon();
Container host = getContainer();
Image img = getImage(host == null || host.isEnabled());
if (img != null) {
if (! hasPixels(img)) {
// No pixels yet, use the default
Icon icon = getLoadingImageIcon();
if (icon != null) { if (icon != null) {
icon.paintIcon(getContainer(), g, rect.x + leftInset, icon.paintIcon(host, g,
rect.y + topInset); rect.x + leftInset, rect.y + topInset);
} }
} }
else { else {
// Draw the image // Draw the image
g.drawImage(image, rect.x + leftInset, rect.y + topInset, g.drawImage(img, rect.x + leftInset, rect.y + topInset,
width, height, imageObserver); width, height, imageObserver);
} }
} }
else { else {
Icon icon = getNoImageIcon(); Icon icon = getNoImageIcon();
if (icon != null) { if (icon != null) {
icon.paintIcon(getContainer(), g, rect.x + leftInset, icon.paintIcon(host, g,
rect.y + topInset); rect.x + leftInset, rect.y + topInset);
} }
View view = getAltView(); View view = getAltView();
// Paint the view representing the alt text, if its non-null // Paint the view representing the alt text, if its non-null
@ -855,7 +863,9 @@ public class ImageView extends View {
// it will pick up the new height/width, if necessary. // it will pick up the new height/width, if necessary.
public boolean imageUpdate(Image img, int flags, int x, int y, public boolean imageUpdate(Image img, int flags, int x, int y,
int newWidth, int newHeight ) { int newWidth, int newHeight ) {
if (image == null || image != img || getParent() == null) { if (img != image && img != disabledImage ||
image == null || getParent() == null) {
return false; return false;
} }
@ -873,6 +883,8 @@ public class ImageView extends View {
if ((state & HEIGHT_FLAG) != HEIGHT_FLAG) { if ((state & HEIGHT_FLAG) != HEIGHT_FLAG) {
height = DEFAULT_HEIGHT; height = DEFAULT_HEIGHT;
} }
} else {
disabledImage = null;
} }
if ((state & LOADING_FLAG) == LOADING_FLAG) { if ((state & LOADING_FLAG) == LOADING_FLAG) {
// No need to resize or repaint, still in the process // No need to resize or repaint, still in the process
@ -885,38 +897,37 @@ public class ImageView extends View {
return false; return false;
} }
// Resize image if necessary: if (image == img) {
short changed = 0; // Resize image if necessary:
if ((flags & ImageObserver.HEIGHT) != 0 && !getElement(). short changed = 0;
getAttributes().isDefined(HTML.Attribute.HEIGHT)) { if ((flags & ImageObserver.HEIGHT) != 0 && !getElement().
changed |= 1; getAttributes().isDefined(HTML.Attribute.HEIGHT)) {
} changed |= 1;
if ((flags & ImageObserver.WIDTH) != 0 && !getElement(). }
getAttributes().isDefined(HTML.Attribute.WIDTH)) { if ((flags & ImageObserver.WIDTH) != 0 && !getElement().
changed |= 2; getAttributes().isDefined(HTML.Attribute.WIDTH)) {
} changed |= 2;
}
synchronized(ImageView.this) { synchronized(ImageView.this) {
if (image != img) { if ((changed & 1) == 1 && (state & WIDTH_FLAG) == 0) {
return false; width = newWidth;
}
if ((changed & 2) == 2 && (state & HEIGHT_FLAG) == 0) {
height = newHeight;
}
if ((state & LOADING_FLAG) == LOADING_FLAG) {
// No need to resize or repaint, still in the process of
// loading.
return true;
}
} }
if ((changed & 1) == 1 && (state & WIDTH_FLAG) == 0) { if (changed != 0) {
width = newWidth; // May need to resize myself, asynchronously:
} safePreferenceChanged();
if ((changed & 2) == 2 && (state & HEIGHT_FLAG) == 0) {
height = newHeight;
}
if ((state & LOADING_FLAG) == LOADING_FLAG) {
// No need to resize or repaint, still in the process of
// loading.
return true; return true;
} }
} }
if (changed != 0) {
// May need to resize myself, asynchronously:
safePreferenceChanged();
return true;
}
// Repaint when done or when new pixels arrive: // Repaint when done or when new pixels arrive:
if ((flags & (FRAMEBITS|ALLBITS)) != 0) { if ((flags & (FRAMEBITS|ALLBITS)) != 0) {

View File

@ -31,6 +31,7 @@ import java.io.*;
import java.net.*; import java.net.*;
import javax.swing.Icon; import javax.swing.Icon;
import javax.swing.ImageIcon; import javax.swing.ImageIcon;
import javax.swing.UIManager;
import javax.swing.border.*; import javax.swing.border.*;
import javax.swing.event.ChangeListener; import javax.swing.event.ChangeListener;
import javax.swing.text.*; import javax.swing.text.*;
@ -2161,6 +2162,7 @@ public class StyleSheet extends StyleContext {
*/ */
public void paint(Graphics g, float x, float y, float w, float h, View v, int item) { public void paint(Graphics g, float x, float y, float w, float h, View v, int item) {
View cv = v.getView(item); View cv = v.getView(item);
Container host = v.getContainer();
Object name = cv.getElement().getAttributes().getAttribute Object name = cv.getElement().getAttributes().getAttribute
(StyleConstants.NameAttribute); (StyleConstants.NameAttribute);
// Only draw something if the View is a list item. This won't // Only draw something if the View is a list item. This won't
@ -2171,7 +2173,7 @@ public class StyleSheet extends StyleContext {
} }
// deside on what side draw bullets, etc. // deside on what side draw bullets, etc.
isLeftToRight = isLeftToRight =
cv.getContainer().getComponentOrientation().isLeftToRight(); host.getComponentOrientation().isLeftToRight();
// How the list indicator is aligned is not specified, it is // How the list indicator is aligned is not specified, it is
// left up to the UA. IE and NS differ on this behavior. // left up to the UA. IE and NS differ on this behavior.
@ -2200,15 +2202,15 @@ public class StyleSheet extends StyleContext {
} }
// set the color of a decoration // set the color of a decoration
if (ss != null) { Color c = (host.isEnabled()
g.setColor(ss.getForeground(cv.getAttributes())); ? (ss != null
} else { ? ss.getForeground(cv.getAttributes())
g.setColor(Color.black); : host.getForeground())
} : UIManager.getColor("textInactiveText"));
g.setColor(c);
if (img != null) { if (img != null) {
drawIcon(g, (int) x, (int) y, (int) w, (int) h, align, drawIcon(g, (int) x, (int) y, (int) w, (int) h, align, host);
v.getContainer());
return; return;
} }
CSS.Value childtype = getChildType(cv); CSS.Value childtype = getChildType(cv);

View File

@ -0,0 +1,93 @@
/*
* Copyright 2007 Sun Microsystems, Inc. All Rights Reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
* CA 95054 USA or visit www.sun.com if you need additional information or
* have any questions.
*/
/* @test
@bug 4783068
@summary Disabled components should render grayed-out HTML
@author Peter Zhelezniakov
@run main Test4783068
*/
import java.awt.*;
import java.awt.image.BufferedImage;
import javax.swing.*;
import javax.swing.plaf.metal.MetalLookAndFeel;
public class Test4783068 {
final static Color TEST_COLOR = Color.WHITE;
final static String html = "<html>" +
"This is a <font color='red'>colored</font> <b>text</b>" +
"<p>with a <a href='http://ru.sun.com'>link</a>" +
"<ul><li>an unordered<li>list</ul>" +
"<ol><li>and an ordered<li>list</ol>" +
"</html>";
void test() {
try {
UIManager.setLookAndFeel(new MetalLookAndFeel());
} catch (UnsupportedLookAndFeelException e) {
throw new Error("Cannot set Metal LAF");
}
// Render text using background color
UIManager.put("textInactiveText", TEST_COLOR);
test(new JLabel(html));
test(new JButton(html));
JEditorPane pane = new JEditorPane("text/html", html);
pane.setDisabledTextColor(TEST_COLOR);
test(pane);
}
void test(JComponent c) {
c.setEnabled(false);
c.setOpaque(true);
c.setBackground(TEST_COLOR);
c.setBorder(null);
Dimension size = c.getPreferredSize();
c.setBounds(0, 0, size.width, size.height);
BufferedImage image = new BufferedImage(size.width, size.height, BufferedImage.TYPE_INT_ARGB);
c.paint(image.getGraphics());
int rgb = TEST_COLOR.getRGB();
for (int i = 0; i < size.height; i++) {
for (int j = 0; j < size.width; j++) {
if (image.getRGB(j, i) != rgb) {
throw new RuntimeException(
String.format("Color mismatch at [%d, %d]", j, i));
}
}
}
}
public static void main(String[] args) throws Exception {
SwingUtilities.invokeAndWait(new Runnable() {
@Override public void run() {
new Test4783068().test();
}
});
}
}