/* * Copyright 1995-2003 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. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun in the LICENSE file that accompanied this code. * * 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. */ package sun.awt.motif; import java.awt.*; import java.awt.peer.*; import java.awt.event.TextEvent; import java.awt.event.MouseEvent; import java.awt.event.MouseWheelEvent; import java.awt.datatransfer.*; import java.io.BufferedReader; import java.io.StringReader; import java.io.IOException; import java.util.Vector; import java.awt.im.InputMethodRequests; public class MTextAreaPeer extends MComponentPeer implements TextAreaPeer { native void pCreate(MComponentPeer parent); private boolean firstChangeSkipped; /** * Initialize JNI field and method IDs */ private static native void initIDs(); static { initIDs(); } void create(MComponentPeer parent) { firstChangeSkipped = false; pCreate(parent); } void initialize() { int start, end; TextArea txt = (TextArea)target; String text; if ((text = txt.getText()) != null) { setText(text); } start = txt.getSelectionStart(); end = txt.getSelectionEnd(); if (end > start) { select(start, end); } else { setCaretPosition(start); } super.pSetScrollbarBackground(getParent_NoClientCode(target).getBackground()); if (!target.isBackgroundSet()) { // This is a way to set the background color of the TextArea // without calling setBackground - go through native C code setTargetBackground(SystemColor.text); } if (!target.isForegroundSet()) { target.setForeground(SystemColor.textText); } setEditable(txt.isEditable()); // oldSelectionStart = -1; // accessibility support // oldSelectionEnd = -1; // accessibility support super.initialize(); } public MTextAreaPeer(TextArea target) { super(target); } public void setEditable(boolean editable) { pSetEditable(editable); /* 4136955 - Calling setBackground() here works around an Xt * bug by forcing Xt to flush an internal widget cache */ setBackground(target.getBackground()); } public void setBackground(Color c) { setTextBackground(c); } public void setForeground(Color c) { pSetInnerForeground(c); } native int getExtraWidth(); native int getExtraHeight(); public native void setTextBackground(Color c); public native void pSetEditable(boolean e); public native void select(int selStart, int selEnd); public native int getSelectionStart(); public native int getSelectionEnd(); public native void setText(String txt); public native String getText(); public native void insert(String txt, int pos); public native void replaceRange(String txt, int start, int end); public native void setFont(Font f); public native void setCaretPosition(int pos); public native int getCaretPosition(); public native void pSetCursor(Cursor c); native void pShow2(); native void pMakeCursorVisible(); public Dimension getMinimumSize() { return getMinimumSize(10, 60); } public Dimension getPreferredSize(int rows, int cols) { return getMinimumSize(rows, cols); } public Dimension getMinimumSize(int rows, int cols) { FontMetrics fm = getFontMetrics(target.getFont()); /* Calculate proper size for text area plus scrollbars. * - Motif allows NO leading in its text areas ... * - extra width and height counts everything outside the * usable text space. * (bug 4103248, 4120310): * - Motif uses maxAscent + maxDescent, not ascent + descent. */ int colWidth = fm.charWidth('0'); int rowHeight = fm.getMaxAscent() + fm.getMaxDescent(); return new Dimension(cols * colWidth + getExtraWidth(), rows * rowHeight + getExtraHeight()); } public boolean isFocusable() { return true; } // Called from native widget when paste key is pressed and we // already own the selection (prevents Motif from hanging while // waiting for the selection) // public void pasteFromClipboard() { Clipboard clipboard = target.getToolkit().getSystemClipboard(); Transferable content = clipboard.getContents(this); if (content != null) { try { String data = (String)(content.getTransferData(DataFlavor.stringFlavor)); // fix for 4401853: to clear TextArea selection if null is pasted data = (data == null ? "" : data); replaceRange(data, getSelectionStart(), getSelectionEnd()); } catch (Exception e) { } } } /* * Print the native component by rendering the Motif look ourselves. * ToDo(aim): needs to query native motif for more accurate size and * color information, the top/left text offsets, and selected text. */ static final int MARGIN = 2; static final int BORDER = 1; static final int SCROLLBAR = 16; int fontHeight; int fontAscent; int fontLeading; int topLine = 0; int numLines = 0; int textLength = 0; Vector lines; int selStart = 0; int selEnd = 0; int movedRight = 0; // the following vars are assigned in print() method transient boolean hscrollbar; transient boolean vscrollbar; public void print(Graphics g) { TextArea area = (TextArea)target; Dimension d = area.size(); Color bg = area.getBackground(); Color fg = area.getForeground(); FontMetrics fm = getFontMetrics(area.getFont()); int vmin, vmax, vval, vvis; int hmin, hmax, hval, hvis; int max = 0; /* Doesn't work right yet. selStart = area.getSelectionStart(); selEnd = area.getSelectionEnd(); */ // Figure out number of lines and max line length String text = area.getText(); textLength = text.length(); BufferedReader is = new BufferedReader(new StringReader(text)); String line; int pos = 0; lines = new Vector(); int sv = ((TextArea)target).getScrollbarVisibility(); vscrollbar = (sv == TextArea.SCROLLBARS_BOTH || sv == TextArea.SCROLLBARS_VERTICAL_ONLY); hscrollbar = (sv == TextArea.SCROLLBARS_BOTH || sv == TextArea.SCROLLBARS_HORIZONTAL_ONLY); boolean wrap = !hscrollbar; int w = d.width - (vscrollbar ? SCROLLBAR : 0); int h = d.height - (hscrollbar ? SCROLLBAR : 0); try { numLines = 0; while((line = is.readLine()) != null) { int len = fm.stringWidth(line); if (len > w && wrap) { // need to do line wrapping int start = 0; int end = 0; int string_length = line.length(); while (true) { int line_width = 0; end = start + 1; // at least one character per line while (end < string_length) { char c = line.charAt(end); int cw = fm.charWidth(c); if (line_width + cw + 10 > w) // +10? break; line_width += cw; end++; } // form a line from start to end (not including end) String substr = line.substring(start, end); // System.out.println("wrap line: " + substr); TextLine tline = new TextLine(); tline.text = substr; tline.pos = pos + start; lines.addElement(tline); start = end; max = Math.max(max, len); numLines ++; if (end == string_length) { // we have processed the whole string pos += line.length() + 1; // +1 for the ending \n ? break; } } } else { TextLine tline = new TextLine(); tline.text = line; tline.pos = pos; lines.addElement(tline); pos += line.length() + 1; max = Math.max(max, len); numLines++; } } is.close(); } catch (IOException e) { } fontHeight = fm.getHeight(); fontAscent = fm.getAscent(); fontLeading = fm.getLeading(); hmin = vmin = 0; vvis = linesInWindow(true); vmax = Math.max(numLines - vvis, 0); vval = 0; hvis = w - (2 * MARGIN); hmax = Math.max(max - hvis, 0); hval = 0; g.setColor(bg); g.fillRect(BORDER, BORDER, w, h); if (vscrollbar) { int sbh = d.height - (hscrollbar ? SCROLLBAR : 0); g.fillRect(d.width - SCROLLBAR - 3, 1, SCROLLBAR - 3, sbh - 1); Graphics ng = g.create(); try { ng.translate(d.width - (SCROLLBAR - 2), 0); drawScrollbar(ng, bg, SCROLLBAR - 2, sbh, vmin, vmax, vval, vvis, false); } finally { ng.dispose(); } } if (hscrollbar) { int sbw = d.width - (vscrollbar ? SCROLLBAR : 0); g.fillRect(1, d.height - SCROLLBAR - 3, sbw - 1, SCROLLBAR - 3); Graphics ng = g.create(); try { ng.translate(0, d.height - (SCROLLBAR - 2)); drawScrollbar(ng, bg, SCROLLBAR - 2, sbw, hmin, hmax, hval, hvis, true); } finally { ng.dispose(); } } draw3DRect(g, bg, 0, 0, w - 1, h - 1, false); if (text != null) { int l = linesInWindow(true); h = d.height - ((2 * MARGIN) + SCROLLBAR); int e = Math.min(numLines - 1, (topLine + l) - 1); paintLines(g, bg, fg, topLine, e); } target.print(g); } int linesInWindow(boolean horizScrollbar) { Dimension d = target.size(); int htotal = d.height - ((2 * MARGIN) + (horizScrollbar? SCROLLBAR : 0)); return htotal / fontHeight; } void paintLines(Graphics g, Color bg, Color fg, int s, int e) { Dimension d = target.size(); int w = d.width - ((2 * BORDER) + (vscrollbar ? SCROLLBAR : 0)); int h = d.height - ((2 * BORDER) + (hscrollbar ? SCROLLBAR : 0)); int lm = linesInWindow(true) + topLine; s = Math.max(topLine, s); e = Math.min(e, lm - 1); Graphics ng = g.create(); try { ng.clipRect(BORDER + MARGIN, MARGIN + BORDER, w - (2*MARGIN), h - (2*MARGIN)); ng.setFont(target.getFont()); for (int i = s ; i <= e; i++) { paintLine(ng, bg, fg, i); } } finally { ng.dispose(); } } void paintLine(Graphics g, Color bg, Color fg, int lnr) { Dimension d = target.size(); int l = linesInWindow(true); if((lnr < topLine) || (lnr >= l + topLine)) { return; } int w = d.width - ((2 * BORDER) + (hscrollbar ? SCROLLBAR : 0)); int y = MARGIN + fontLeading + ((lnr - topLine) * fontHeight); String text = ((TextLine)lines.elementAt(lnr)).text; int len = text.length(); if (lnr > numLines - 1) { g.setColor(bg); g.fillRect(BORDER, y - fontLeading, w, fontHeight); return; } int s = 0; int e = (lnr < numLines - 1) ? len : textLength; int xs = pos2x(selStart) - movedRight; int xe = pos2x(selEnd) - movedRight; Color highlight = bg.brighter(); if ((selStart < s) && (selEnd > e)) { g.setColor(highlight); g.fillRect(BORDER, y - fontLeading, w, fontHeight); } else { g.setColor(bg); g.fillRect(BORDER, y - fontLeading, w, fontHeight); if ((selStart >= s) && (selStart <= e)) { g.setColor(highlight); if (selEnd > e) { g.fillRect(xs, y - fontLeading, (w + BORDER) - xs, fontHeight); } else if (selStart == selEnd) { //g.fillRect(xs, y - fontLeading, 1, fontHeight); } else { g.fillRect(xs, y - fontLeading, xe - xs, fontHeight); } } else if ((selEnd >= s) && (selEnd <= e)) { g.setColor(highlight); g.fillRect(BORDER, y - fontLeading, xe - BORDER, fontHeight); } } g.setColor(fg); g.drawString(text, MARGIN - movedRight, y + fontAscent); } int pos2x(int pos) { FontMetrics fm = getFontMetrics(target.getFont()); int widths[] = fm.getWidths(); TextLine tl1 = (TextLine)lines.elementAt(0); TextLine tl2; int l = 0; for (int i = 0; i < lines.size() - 1; i++) { tl2 = (TextLine)lines.elementAt(i+1); if (pos >= tl1.pos && pos < tl2.pos) { l = i; break; } tl1 = tl2; } int x = MARGIN; for (int i = 0 ; i < (pos - tl1.pos - 1) ; i++) { x += widths[tl1.text.charAt(i)]; } return x; } /** * DEPRECATED */ public void insertText(String txt, int pos) { insert(txt, pos); } /** * DEPRECATED */ public void replaceText(String txt, int start, int end) { replaceRange(txt, start, end); } /** * DEPRECATED */ public Dimension minimumSize() { return getMinimumSize(); } /** * DEPRECATED */ public Dimension preferredSize(int rows, int cols) { return getPreferredSize(rows, cols); } /** * DEPRECATED */ public Dimension minimumSize(int rows, int cols) { return getMinimumSize(rows, cols); } /* * Post a new TextEvent when the value of a text component changes. */ public void valueChanged() { postEvent(new TextEvent(target, TextEvent.TEXT_VALUE_CHANGED)); } void pShow(){ pShow2(); notifyTextComponentChange(true); } void pHide(){ notifyTextComponentChange(false); super.pHide(); } void pDispose(){ notifyTextComponentChange(false); super.pDispose(); } public boolean handlesWheelScrolling() {return true;} public void handleEvent(AWTEvent e) { if (e.getID() == MouseEvent.MOUSE_WHEEL) { MouseWheelEvent mwe = (MouseWheelEvent)e; nativeHandleMouseWheel(mwe.getScrollType(), mwe.getScrollAmount(), mwe.getWheelRotation()); } else { super.handleEvent(e); } } public InputMethodRequests getInputMethodRequests() { return null; } native void nativeHandleMouseWheel(int scrollType, int scrollAmount, int wheelRotation); // // Accessibility support // // stub functions: to be fully implemented in a future release public int getIndexAtPoint(int x, int y) { return -1; } public Rectangle getCharacterBounds(int i) { return null; } public long filterEvents(long mask) { return 0; } /* To be fully implemented in a future release int oldSelectionStart; int oldSelectionEnd; public native int getIndexAtPoint(int x, int y); public native Rectangle getCharacterBounds(int i); public native long filterEvents(long mask); /** * Handle a change in the text selection endpoints * (Note: could be simply a change in the caret location) * public void selectionValuesChanged(int start, int end) { return; // Need to write implementation of this. } */ } class TextLine { String text; int pos; }