JTextField: Blinkender Unterstrich

XPlayer

Mitglied
Hallo,

ich arbeite gerade an einer kleinen Java-Anwendung, die eine Buchstabenauswahl in einem JTextField nur über das Mausrad ermöglicht (zur nächsten Position wird mittels eines Buttons gesprungen). Nun soll die aktuelle Position im JTextField, also wo gerade der Buchstabe ausgewählt wird, durch einen blinkenden Unterstrich (wie in der Konsole) gekennzeichnet werden.
Hat jemand zumindest einen Ansatz, wie so etwas in Java umsetzbar wäre.

Danke schonmal im Voraus.

Gruß XPlayer
 
Zuletzt bearbeitet:

Ebenius

Top Contributor
Eine eigene Implementierung von Caret hilft hier. Und da man die ganz sicher nicht selber machen will, passt man am besten DefaultCaret an:

Java:
final JTextField tf = new JTextField();
final DefaultCaret caret = new DefaultCaret() {

  private static final long serialVersionUID = 1L;

  @Override
  public void paint(Graphics g) {
    if (isVisible()) {
      try {
        final JTextComponent component = getComponent();
        TextUI mapper = component.getUI();
        final int dot = getDot();
        final Bias dotBias = getDotBias();
        Rectangle r = mapper.modelToView(component, dot, dotBias);

        if ((r == null) || ((r.width == 0) && (r.height == 0))) {
          return;
        }
        if (width > 0
              && height > 0
              && !this.contains(r.x, r.y, r.width, r.height)) {
          // We seem to have gotten out of sync and no longer
          // contain the right location, adjust accordingly.
          Rectangle clip = g.getClipBounds();

          if (clip != null && !clip.contains(this)) {
            // Clip doesn't contain the old location, force it
            // to be repainted lest we leave a caret around.
            repaint();
          }
          // This will potentially cause a repaint of something
          // we're already repainting, but without changing the
          // semantics of damage we can't really get around this.
          damage(r);
        }
        g.setColor(component.getCaretColor());
        int paintWidth =
              component.getFontMetrics(component.getFont())
                    .charWidth('M');
        g.fillRect(r.x, r.y, paintWidth, r.height);
      } catch (BadLocationException e) {
        // can't render I guess
        // System.err.println("Can't render cursor");
      }
    }
  }
};
caret.setBlinkRate(300);
tf.setCaret(caret);
Ebenius
 

XPlayer

Mitglied
Hi,

thx für deine Lösung.

Allerdings findet das Blinken über dem gesamten Buchstaben statt und nicht nur als Linie unter ihm.

Vermutlich muss an dieser Anweisung noch etwas an den Parametern verändert werden:
Java:
g.fillRect(r.x, r.y, paintWidth, r.height);

Desweiteren habe ich das Problem, das beim Scrollen der Maus die Animation unterbrochen wird, sprich das schwarze Rechteck ist immer da.

Ich hoffe du kannst mir nochmals weiterhelfen.

Gruß
XPlayer
 
Zuletzt bearbeitet:

Ebenius

Top Contributor
Ich wusste's ja immer schon; Textcursor sind einfach kompliziert. So sollte's gehen:

Java:
final DefaultCaret caret = new DefaultCaret() {

  private static final long serialVersionUID = 1L;

  private Timer flasher;
  private FlashHandler flashHandler;
  private boolean hide;

  private volatile int caretHeight = 3; // adjust this value
  private volatile int caretWidth = 0;

  class FlashHandler implements ActionListener, java.io.Serializable {

    private static final long serialVersionUID = 1L;

    @SuppressWarnings("boxing")
    public void actionPerformed(ActionEvent e) {
      hide = !hide;
      System.out.printf("Hide %b%n", hide);
      repaint();
    }
  }

  @Override
  public void install(JTextComponent c) {
    super.install(c);
    flashHandler = new FlashHandler();
    if (flasher != null) {
      flasher.addActionListener(flashHandler);
    }
  }

  @Override
  public void deinstall(JTextComponent c) {
    super.deinstall(c);
    if (flasher != null) {
      flasher.stop();
      flasher.removeActionListener(flashHandler);
    }
    flashHandler = null;
  }

  @Override
  public int getBlinkRate() {
    return (flasher == null) ? 0 : flasher.getDelay();
  }

  @Override
  public void setBlinkRate(int rate) {
    if (rate != 0) {
      if (flasher == null) {
        flasher = new Timer(rate, flashHandler);
      }
      flasher.setDelay(rate);
    } else {
      if (flasher != null) {
        flasher.stop();
        flasher.removeActionListener(flashHandler);
        flasher = null;
      }
    }
  }

  @Override
  public void paint(Graphics g) {
    if (isVisible()) {
      try {
        final JTextComponent component = getComponent();
        TextUI mapper = component.getUI();
        final int dot = getDot();
        final Bias dotBias = getDotBias();
        Rectangle r = mapper.modelToView(component, dot, dotBias);

        if ((r == null) || ((r.width == 0) && (r.height == 0))) {
          return;
        }
        g.setFont(getComponent().getFont());
        caretWidth = g.getFontMetrics().charWidth('M');
        System.out.println("Caret set");
        if (width > 0
              && height > 0
              && !this.contains(r.x, r.y, r.width, r.height)) {
          // We seem to have gotten out of sync and no longer
          // contain the right location, adjust accordingly.
          Rectangle clip = g.getClipBounds();

          if (clip != null && !clip.contains(this)) {
            // Clip doesn't contain the old location, force it
            // to be repainted lest we leave a caret around.
            repaint();
          }

          // This will potentially cause a repaint of something
          // we're already repainting, but without changing the
          // semantics of damage we can't really get around this.
          damage(r);
        }
        if (!hide) {
          g.setColor(component.getCaretColor());
          final int h = Math.min(caretHeight, r.height);
          g.fillRect(r.x, r.y + height - h, caretWidth, h);
        }
      } catch (BadLocationException e) {
        // can't render I guess
        // System.err.println("Can't render cursor");
      }
    }
  }

  @Override
  protected synchronized void damage(Rectangle r) {
    if (r != null) {
      int damageWidth = caretWidth;
      x = r.x;
      y = r.y;
      width = damageWidth;
      height = r.height + caretHeight;
      repaint();
      System.out.println("Damage");
    }
  }

  @Override
  public void setVisible(boolean visible) {
    if (visible) {
      final JTextComponent component = getComponent();
      final Font font = component.getFont();
      caretWidth = component.getFontMetrics(font).charWidth('M');
    }
    super.setVisible(visible);
    if (flasher != null) {
      if (visible) {
        flasher.start();
      } else {
        flasher.stop();
      }
    }
  }
};

Ebenius
 

XPlayer

Mitglied
Hallo Ebenius,

vielen Dank nochmals für deine Hilfe.

Der Code tut im Gegensatz zum ersten leider irgendwie gar nicht, sprich ich erhalte gar keinen Unterstrich, die Ausgaben auf der Konsole erfolgen dagegen. Die Parameter caretHeight (= 3) und caretWidth (= 0) sind dabei unverändert.

Gruß XPlayer
 

Ebenius

Top Contributor
Hm. Könnte an fehlender Höhe liegen oder an was anderem. Ich teste auch nur unter Linux. Probier mal meinen gesamten Test:
Java:
/* (@)JCustomCaretTest.java */

/* Copyright 2011 Sebastian Haufe

 * Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       [url]http://www.apache.org/licenses/LICENSE-2.0[/url]

 * Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License. */

package com.ebenius;

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.*;
import javax.swing.plaf.TextUI;
import javax.swing.text.*;
import javax.swing.text.Position.Bias;

/**
 * TODO: Javadoc me!
 * 
 * @version $Revision$ as of $Date$
 * @author Sebastian Haufe
 * @since Playground-3.8
 */
public class JCustomCaretTest {

  /** Creates the GUI. Call on EDT, only! */
  static void createAndShowGui() {
    final JPanel contentPane = new JPanel(new BorderLayout(6, 6));

    final JTextField tf = new JTextField();
    final DefaultCaret caret = new DefaultCaret() {

      private static final long serialVersionUID = 1L;

      private Timer flasher;
      private FlashHandler flashHandler;
      private boolean hide;

      private volatile int caretHeight = 3; // adjust this value
      private volatile int caretWidth = 0;

      class FlashHandler implements ActionListener, java.io.Serializable {

        private static final long serialVersionUID = 1L;

        public void actionPerformed(ActionEvent e) {
          hide = !hide;
          repaint();
        }
      }

      @Override
      public void install(JTextComponent c) {
        super.install(c);
        flashHandler = new FlashHandler();
        if (flasher != null) {
          flasher.addActionListener(flashHandler);
        }
      }

      @Override
      public void deinstall(JTextComponent c) {
        super.deinstall(c);
        if (flasher != null) {
          flasher.stop();
          flasher.removeActionListener(flashHandler);
        }
        flashHandler = null;
      }

      @Override
      public int getBlinkRate() {
        return (flasher == null) ? 0 : flasher.getDelay();
      }

      @Override
      public void setBlinkRate(int rate) {
        if (rate != 0) {
          if (flasher == null) {
            flasher = new Timer(rate, flashHandler);
          }
          flasher.setDelay(rate);
        } else {
          if (flasher != null) {
            flasher.stop();
            flasher.removeActionListener(flashHandler);
            flasher = null;
          }
        }
      }

      @Override
      public void paint(Graphics g) {
        if (isVisible()) {
          try {
            final JTextComponent component = getComponent();
            TextUI mapper = component.getUI();
            final int dot = getDot();
            final Bias dotBias = getDotBias();
            Rectangle r = mapper.modelToView(component, dot, dotBias);

            if ((r == null) || ((r.width == 0) && (r.height == 0))) {
              return;
            }
            g.setFont(getComponent().getFont());
            caretWidth = g.getFontMetrics().charWidth('M');
            if (width > 0
                  && height > 0
                  && !this.contains(r.x, r.y, r.width, r.height)) {
              // We seem to have gotten out of sync and no longer
              // contain the right location, adjust accordingly.
              Rectangle clip = g.getClipBounds();

              if (clip != null && !clip.contains(this)) {
                // Clip doesn't contain the old location, force it
                // to be repainted lest we leave a caret around.
                repaint();
              }

              // This will potentially cause a repaint of something
              // we're already repainting, but without changing the
              // semantics of damage we can't really get around this.
              damage(r);
            }
            if (!hide) {
              g.setColor(component.getCaretColor());
              final int h = Math.min(caretHeight, r.height);
              g.fillRect(r.x, r.y + height - h, caretWidth, h);
            }
          } catch (BadLocationException e) {
            // can't render I guess
            // System.err.println("Can't render cursor");
          }
        }
      }

      @Override
      protected synchronized void damage(Rectangle r) {
        if (r != null) {
          int damageWidth = caretWidth;
          x = r.x;
          y = r.y;
          width = damageWidth;
          height = r.height + caretHeight;
          repaint();
        }
      }

      @Override
      public void setVisible(boolean visible) {
        if (visible) {
          final JTextComponent component = getComponent();
          final Font font = component.getFont();
          caretWidth = component.getFontMetrics(font).charWidth('M');
        }
        super.setVisible(visible);
        if (flasher != null) {
          if (visible) {
            flasher.start();
          } else {
            flasher.stop();
          }
        }
      }
    };
    caret.setBlinkRate(300);
    tf.setCaret(caret);

    contentPane.add(tf);
    final JFrame f = new JFrame("Test Frame: JCustomCaretTest"); //$NON-NLS-1$
    f.setContentPane(contentPane);
    f.setSize(400, 400);
    f.setLocationRelativeTo(null);
    f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    f.setVisible(true);
  }

  /** @param args ignored */
  public static void main(String[] args) {
    SwingUtilities.invokeLater(new Runnable() {

      public void run() {
        createAndShowGui();
      }
    });
  }
}
Wenns nicht funktioniert, musst du mal einen Debugger anschmeißen. Besonders interessant ist dabei, wann/ob der flasher bei Dir überhaupt gestartet und gestoppt wird.

Ebenius
 

XPlayer

Mitglied
Hallo Ebenius,

inzwischen habe ich herausgefunden, an was es gelegen hat. Du hast im Test-Code das JTextField als BorderLayout.Center festgelegt, es nimmt also den vorhandenen Platz vollständig ein. In meinem Code steht das JTextField im FlowLayout. Da es dort eine viel geringere Höhe besitzt, ist der blickende Strich vermutlich außerdem des sichtbaren Bereichs.

Ich habe nun folgende Zeile
Java:
g.fillRect(r.x, r.y + height - h, caretWidth, h);
durch diese Zeile ersetzt
Java:
g.fillRect(r.x, r.y + height - 3, caretWidth, 2);
und
Java:
private volatile int caretHeight = 1;
dann ist der blinkende Strich sichtbar.

Die Variable h ist damit unnnötig geworden.
Vielleicht hast du aber auch noch eine bessere Lösung...

Übrigens gibt es da noch ein kleines Problem, wenn der Text markiert ist. Es bleibt dann nämlich ein kleiner Punkt übrig (unten rechts), der weiter blinkt.

Aber danke nochmals für die Hilfe!

Gruß
XPlayer
 
Zuletzt bearbeitet:
Ähnliche Java Themen
  Titel Forum Antworten Datum
P JTextField wird nur sehr klein angezeigt und verändert die Größe nicht AWT, Swing, JavaFX & SWT 3
P JTextfield Text durch Keylistener verändern AWT, Swing, JavaFX & SWT 4
raeuchertofu Text von JTextfield und JLabel ändern AWT, Swing, JavaFX & SWT 2
I JTextfield senden über Enter AWT, Swing, JavaFX & SWT 1
H Swing Wie Fokus von JTextField entfernen? AWT, Swing, JavaFX & SWT 9
M mehrere jTextField untereinander AWT, Swing, JavaFX & SWT 1
G JTextField Änderungen überprüfen AWT, Swing, JavaFX & SWT 4
B JTextField in Panel positionieren AWT, Swing, JavaFX & SWT 1
A Swing JTextField an Button übergeben für Popup-Fenster funktioniert nicht AWT, Swing, JavaFX & SWT 3
R Swing JTextField nicht-kopierbarer Text AWT, Swing, JavaFX & SWT 1
R Swing JTextField - Schriftgröße je nach Länge und Größe ändern AWT, Swing, JavaFX & SWT 0
A Swing Datei lässt sich zwar speichern, aber nicht laden (Inhalt im JTextField anzeigen) AWT, Swing, JavaFX & SWT 18
A Fenster genau unterhalb von JTextField anzeigen AWT, Swing, JavaFX & SWT 1
xYurisha Gui Focus von JTextField weglenken AWT, Swing, JavaFX & SWT 6
G Eingabe in JTextField in Array speichern und in JList ausgeben AWT, Swing, JavaFX & SWT 7
G JTextField Eingabe in DB speichern AWT, Swing, JavaFX & SWT 68
C JTextField Inhalt mit Maus selektieren wenn disabled AWT, Swing, JavaFX & SWT 2
R jTextField Eingabe überprüfen AWT, Swing, JavaFX & SWT 2
ralfb1105 Swing Wert in JTextField wird falsch ausgegeben AWT, Swing, JavaFX & SWT 2
D Inhalt von JTextField in MySQL speichern. Wie ? AWT, Swing, JavaFX & SWT 1
D Platzierung von JTextfield in JPanel AWT, Swing, JavaFX & SWT 3
J 100% CPU Last mit JTextField AWT, Swing, JavaFX & SWT 2
U Swing JTextField aus dem Editmodus nehmen AWT, Swing, JavaFX & SWT 1
Q-bert JTextField per Mausklick einfügen AWT, Swing, JavaFX & SWT 3
L JTextField erscheint erst nach Veränderung der Framegröße AWT, Swing, JavaFX & SWT 5
BobbGott durchsichtiges JTextField mit Hintergrund AWT, Swing, JavaFX & SWT 12
L Swing JTextField nicht sichtbar AWT, Swing, JavaFX & SWT 1
Soloeco Swing JTextField auslesen Klassen übergreifend AWT, Swing, JavaFX & SWT 16
N Swing Abfrage durch JTextField nach Button betätigung AWT, Swing, JavaFX & SWT 3
J JTextfield automatisch aus zweiter Klasse aktualisieren AWT, Swing, JavaFX & SWT 5
T ActionListener nimmt JTextField nicht mehr an. AWT, Swing, JavaFX & SWT 2
N Swing Jtextfield und Ausgabe von Wärungsbeträgen AWT, Swing, JavaFX & SWT 3
beatles Swing JTextField - Game Status aus Array anzeigen AWT, Swing, JavaFX & SWT 3
blazingblade komischerweise probleme mit jtextfield.gettext() AWT, Swing, JavaFX & SWT 9
KaffeeFan mehrere JTextField durchlaufen AWT, Swing, JavaFX & SWT 4
T JTextField Array im JPanel wird nicht komplett angezeigt AWT, Swing, JavaFX & SWT 7
K Swing unterschied JTextField und JLabel AWT, Swing, JavaFX & SWT 7
Tausendsassa Text changed event handler jTextField AWT, Swing, JavaFX & SWT 3
B jTextfield Übergabe des Strings AWT, Swing, JavaFX & SWT 16
P JTextField unsichtbar machen AWT, Swing, JavaFX & SWT 2
G .txt-Datei einlesen /formatieren in JTextfield ?? AWT, Swing, JavaFX & SWT 8
C Swing JTextField Funktion zuweisen - Löschen von Buchstaben/Sonderzeichen AWT, Swing, JavaFX & SWT 6
S JTextField parameter unveränderlich AWT, Swing, JavaFX & SWT 9
T bekomme keinen Focus auf JTextField AWT, Swing, JavaFX & SWT 3
N Cursor im JTextField AWT, Swing, JavaFX & SWT 6
T Swing NullPointerException beim auslesen von jTextField - Anfänger! AWT, Swing, JavaFX & SWT 3
A JTextField und JButtons AWT, Swing, JavaFX & SWT 2
T jTextField mit String befüllen AWT, Swing, JavaFX & SWT 1
O JTextfield wird über'drawed'.. AWT, Swing, JavaFX & SWT 6
M Swing JtextField Panel in ander Panel aufrufen AWT, Swing, JavaFX & SWT 0
S Bekomme bein Buttonklick das Jtextfield nicht ausgegeben AWT, Swing, JavaFX & SWT 8
Seikuassi Swing JTextField nimmt kein Einfügen (Strg+V) an AWT, Swing, JavaFX & SWT 4
Z Fenster leer, wenn ich ein JTextField erzeuge AWT, Swing, JavaFX & SWT 3
M Swing JTextField getText Mehtode übergeben in eine Klasse zu einer Rechnung AWT, Swing, JavaFX & SWT 1
B Swing JTextField mit Bildschirmauflösung skalieren lassen. AWT, Swing, JavaFX & SWT 4
C Geloest: JTextfield: Titel automatisch entfernen, sobald Text eingegeben wird AWT, Swing, JavaFX & SWT 1
T Internes Event-handling in TextField/JTextField AWT, Swing, JavaFX & SWT 2
T AWT JTextfield verschwindet AWT, Swing, JavaFX & SWT 10
dat_vin KeyEvent jTextField AWT, Swing, JavaFX & SWT 12
C JTextfield ohne Enter feuern AWT, Swing, JavaFX & SWT 8
W Swing JTextField nur für Integer AWT, Swing, JavaFX & SWT 2
H Swing JList/JTable mit JButton, JTextField, Image, JComboBox und JLable AWT, Swing, JavaFX & SWT 2
O JTextfield und JComboBox Wert in SQL Database speichern AWT, Swing, JavaFX & SWT 4
F Swing JTextField (unbeschriebbar machen) AWT, Swing, JavaFX & SWT 12
W JTextfield - Wert lässt sich nicht setzen AWT, Swing, JavaFX & SWT 3
TheJavaKid JTextField: Reine Domain herausfiltern, sonst Inhalt löschen AWT, Swing, JavaFX & SWT 1
R Swing User tippt in unsichtbares JTextField AWT, Swing, JavaFX & SWT 3
F Swing JTextField in JList (Funktion beibehalten) AWT, Swing, JavaFX & SWT 2
A Swing JLabel/JTextField Inhalt mit JSlider verändern AWT, Swing, JavaFX & SWT 12
U Swing JTextField zur Laufzeit auf Eingaben prüfen AWT, Swing, JavaFX & SWT 0
X Swing JTextField AWT, Swing, JavaFX & SWT 4
M Swing JTextField nicht ausfüllbar AWT, Swing, JavaFX & SWT 8
C Problem mit JTextField AWT, Swing, JavaFX & SWT 4
S Swing WYSIWYG JTextField AWT, Swing, JavaFX & SWT 2
vodkaz JTextField && Strings AWT, Swing, JavaFX & SWT 2
R Swing JScrollBar mit JTextField koppeln AWT, Swing, JavaFX & SWT 3
A Swing 2 JSlider verknüpfen und mit einem JTextField vergleichen, dann Ausgabe AWT, Swing, JavaFX & SWT 5
X JTextField Ausgabe klappt nicht AWT, Swing, JavaFX & SWT 4
S Swing Verzeichnis/Datei-Pfad in JTextField schreiben AWT, Swing, JavaFX & SWT 2
R Swing JTextField getText() liefert leeren String AWT, Swing, JavaFX & SWT 6
J Mausfarbe ändern JTextField AWT, Swing, JavaFX & SWT 6
S Anordnung von GUI-Komponenten mit LayouManger (Problem beim anzeigen von JTextField) AWT, Swing, JavaFX & SWT 5
A editable JTextField & setText AWT, Swing, JavaFX & SWT 2
O Swing JTextField Sync View -> Model AWT, Swing, JavaFX & SWT 3
M JTextField gibt beim auslesen null aus AWT, Swing, JavaFX & SWT 9
K JTextField mit Standardtext AWT, Swing, JavaFX & SWT 3
F Swing JTextfield Beispieltext soll nach Klick verschwinden AWT, Swing, JavaFX & SWT 3
TheJavaKid DocumentListener soll Text in JTextField ändern AWT, Swing, JavaFX & SWT 4
I Swing JTextField ausblenden -? AWT, Swing, JavaFX & SWT 6
0 Swing JTextField - Position ? AWT, Swing, JavaFX & SWT 2
L Button über/auf JTextField AWT, Swing, JavaFX & SWT 5
vandread Swing Probleme mit jTextField im zusammenspiel mit einem Einblendeffekt (inkl. KSKB) AWT, Swing, JavaFX & SWT 6
M Swing jTextfield und PlainDocument AWT, Swing, JavaFX & SWT 18
D JTextField über ein Game AWT, Swing, JavaFX & SWT 11
A Swing JTextField durch einen JButton leeren AWT, Swing, JavaFX & SWT 15
Kaniee Swing Woher bekommt JTextField die Eingaben? AWT, Swing, JavaFX & SWT 2
M JTextField gesperrt nach erster Benutzung... AWT, Swing, JavaFX & SWT 4
C JTextField KeyListener AWT, Swing, JavaFX & SWT 6
R JTextField mit abgerundeten Ecken - Geht über Grenze hinaus AWT, Swing, JavaFX & SWT 4
R JTextField Eingabefeld innerhalb verschieben AWT, Swing, JavaFX & SWT 2

Ähnliche Java Themen

Neue Themen


Oben