Swing JLabel drehen für Kartenspiel

guitarflow

Mitglied
Hallo liebe Community,

möchte gerne ein Mau-Mau programmieren, um neben dem Studium noch ein bisschen Java zu üben.
Dachte auch an eine eventuelle p2p-Architektur und andere Features ... bin aber jetz erstmal beim Entwurf meiner GUI auf ein Problem gestossen und hoffe ihr könnt mir helfen.

Ich dachte mir, dass jeder Spieler eine "Hand" hat, in der die Karten so aufgefächert werden, wie man sie beim normalen Spielen auch auf der Hand hält. Also z.B. in einem festen Winkel von 120°.

Nun möchte ich meine JLabels, die die Karten repräsentieren, dementsprechend um einen festen Punkt so drehen, dass sie insgesamt diesen 120°-Winkel erreichen.

Also bei 3 Karten : 1. Karte = 40°, 2.Karte = 80°, 3. Karte =120°
bei 4 Karten : 1.Karte = 30°, 2.Karte = 60° ....

Natürlich sollte die Karte dann auch klickbar sein, damit der Spieler auswählen kann, welche Karte er als nächstes spielen will.


Habe festgestellt, dass das mit dem Drehen gar nicht so einfach ist. Was ich bis jetz gesehen habe, müsste ich mit BufferedImages und AffineTransform arbeiten, allerdings glaub ich nicht, dass das dann noch klickbar wäre, oder??

Hat jemand ne Idee, wie man das angehen könnte?

Danke schon mal vorab,


guitarflow
 

guitarflow

Mitglied
Hallo Marco,

super, danke für den Hinweis! Bin bei meiner Googlerei schon mal drüber gestolpert aber ich dachte aus irgendeinem Grund, dass das nicht das ist, was ich brauch :oops:

Jetz hab ich ein bisschen mit rumgespielt, Kartendrehen funktioniert einwandfrei!
Allerdings werden die Karten ganz schön pixelig, aber ich denk da kann ich noch mit "renderingHints" arbeiten.

Das Problem ist, dass das gedrehte JLabel trotzdem sozusagen Platz für ihr komplett umgebenes Rechteck nach dem Drehen belegt und ich so keine gedrehten Karten unmittelbar nebeneinander setzen kann.
Anbei mal ein Screenie. Denk dass man sofort versteht, was ich meine!
Habe versucht, für die oberste Karte einen gelben Hintergrund reinzumachen, um zu sehen, ob das was da überlappt zum Label-Hintergrund gehört. Wie man am Screenshot sieht, is da nix Gelbes zu sehen.
Als Container dient übrigens ein JLayeredPane....

Hoffe ihr könnt mir nochmal helfen.

mfg,

guitarflow
 

Anhänge

  • Bildschirmfoto 2010-06-19 um 21.00.07.jpg
    Bildschirmfoto 2010-06-19 um 21.00.07.jpg
    30,5 KB · Aufrufe: 49
Zuletzt bearbeitet:

Marco13

Top Contributor
Hm. Schwer zu sagen, es könnte natürlich auch sein, dass der JXTransformer da was falsch macht (KÖNNTE - das müßte man genauer prüfen). Spontan geraten: Du könntest mal schauen, ob ein
label.setOpaque(false);
auf allen Labels was hilft. Ansonsten noch der obligatirische Hinweis, dass es IMHO ein Krampf ist, für sowas JLabels zu verwenden. Man kann die Bilder auch einfach mit g.drawImage malen... Das Aklicken kann man dann zwar nicht mehr mit einem MouseListener machen, aber sooo viel schwieriger wäre das auch nicht...
 

guitarflow

Mitglied
setOpaque hat nix geholfen.

Naja, ich dachte mir dass es mit den JLabels am einfachsten is, die einzelnen Karten zu handhaben.

Wie würdest dus denn mit Graphics machen??
Kann ich mir jetz gar nicht vorstellen. Bzw. hab ich da ein kleines Verständnisproblem ....

Vielleicht kannst du mir mal ein Beispiel nennen?

Danke schonmal,

mfg,
guitarflow
 

Marco13

Top Contributor
Joa, kommt halt drauf an... Vielleicht habe ich auch die Tendenz, sowas zu weit unten anzupacken.. :oops: Aber bevor man da mit JLayeredPane anfängt und eine externe Library einbindet, um so ein gedrehtes Bildchen zu malen... ich würde mir da halt eine Klasse erstellen, die das macht.

Auch auf die Gefahr hin, dass du das, was ich jetzt in ein paar Minuten schnell zusammengeschustert habe, 1:1 und unverstanden übernimmst, und ich mir dann Vorwürfe anhören darf, wenn damit irgendwas nicht geht, was du gerne machen würdest (aber bisher noch nicht erwähnt hast) hier mal ein Beispiel, wie man ein Bild machen könnte, das man gedreht und verschoben zeichnen und anklicken kann...
Java:
import javax.swing.*;
import java.awt.*;
import java.awt.geom.*;
import java.awt.image.*;
import java.awt.event.*;
import javax.imageio.*;
import java.io.*;

class ImagePaint extends JPanel implements MouseListener
{
    public static void main(String arg[])
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            public void run()
            {
                JFrame f = new JFrame();
                f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

                ImagePaint imagePaint = new ImagePaint();
                f.getContentPane().add(imagePaint);

                f.setSize(800,600);
                f.setVisible(true);

            }
        });
    }

    private static class RotatedImage
    {
        private Point position = new Point(0,0);
        private BufferedImage image;
        private float angle = 0;

        private Point pickPoint = new Point(0,0);

        public RotatedImage(String fileName)
        {
            try
            {
                image = ImageIO.read(new File(fileName));
                System.out.println("image "+image);
            }
            catch (Exception e)
            {
                e.printStackTrace();
            }
        }
        public void setPosition(int x, int y)
        {
            this.position = new Point(x,y);
        }
        public void setAngle(float angle)
        {
            this.angle = angle;
        }

        private AffineTransform computeTransform()
        {
            AffineTransform at = new AffineTransform();
            int w = image.getWidth();
            int h = image.getHeight();
            at.translate(position.x, position.y);
            at.rotate(angle);
            at.translate(-w/2,-h/2);
            return at;
        }

        public boolean isHit(Point p)
        {
            AffineTransform at = computeTransform();
            AffineTransform ati = at;
            try
            {
                ati = at.createInverse();
            }
            catch (NoninvertibleTransformException e)
            {
                e.printStackTrace();
            }
            Point2D dst = ati.transform(new Point2D.Float(p.x, p.y), null);
            System.out.println("dst "+dst);
            pickPoint = new Point((int)dst.getX(),(int)dst.getY());
            return
                dst.getX() >= 0 && dst.getX() < image.getWidth() &&
                dst.getY() >= 0 && dst.getY() < image.getHeight();
        }

        public void paintOn(Graphics gr)
        {
            Graphics2D g = (Graphics2D)gr;
            AffineTransform oldAT = g.getTransform();
            g.transform(computeTransform());
            g.drawImage(image,0,0,null);
            g.setTransform(oldAT);

            g.setColor(Color.RED);
            g.fillOval(pickPoint.x-2, pickPoint.y-2,4,4);
        }
    }

    private RotatedImage rotatedImage;

    public ImagePaint()
    {
        addMouseListener(this);
        rotatedImage = new RotatedImage("SplineTest01.gif");

        rotatedImage.setPosition(350, 250);

        Thread t = new Thread(new Runnable()
        {
            public void run()
            {
                float a = 0;
                while (true)
                {
                    rotatedImage.setAngle(a);
                    a += 0.05f;
                    repaint();
                    try
                    {
                        Thread.sleep(50);
                    }
                    catch (Exception e)
                    {
                        e.printStackTrace();
                    }
                }

            }
        });
        t.start();
    }

    public void paintComponent(Graphics g)
    {
        super.paintComponent(g);
        rotatedImage.paintOn(g);
    }

    public void mouseClicked(MouseEvent e)
    {
        boolean hit = rotatedImage.isHit(e.getPoint());
        System.out.println("Image hit? "+hit);
        repaint();
    }

    public void mousePressed(MouseEvent e) {}
    public void mouseEntered(MouseEvent e) {}
    public void mouseExited(MouseEvent e) {}
    public void mouseReleased(MouseEvent e) {}
}
 

guitarflow

Mitglied
Boa, vielen herzlichen Dank für deinen Code!!! :toll:

Wie du schon geschrieben hast, is der jetz vielleicht für mich nicht so ganz leicht zu verstehen.

Hätt noch ein paar Fragen zum Code. Vielleicht kannst mir nochmal weiter helfen?
Also ich hab jetz mal der Einfachheit halber deine Animation rausgebaut.
Hab mich bis jetz auch noch nicht wirklich mit Threads beschäftigt. Deswegen lass ich das jetz erst mal aussen vor.

Hast ganz oben im Code mal die Zeilen :
Java:
 SwingUtilities.invokeLater(new Runnable(){
           public void run()
           {
Habs schon öfter gesehen, dass Eclipse oder NetBeans das per default so reinschreibt in die Main. Habs bei meinem eigenen Code eigentlich immer weggelassen. Was machen diese Zeilen denn??
Bzw. welchen Nachteil hats, wenn ichs nicht hinschreib??

Hab versucht, deine "isHit"-Funktion zu verstehen, allerdings will mir das nicht ganz einleuchten:
Java:
public boolean isHit(Point p){

        AffineTransform at = computeTransform();
        AffineTransform ati = at;

        try{
            ati = at.createInverse();

        }catch (NoninvertibleTransformException e){
            e.printStackTrace();
        }

        Point2D dst = ati.transform(new Point2D.Float(p.x, p.y), null);

        System.out.println("dst "+dst);
        pickPoint = new Point((int)dst.getX(),(int)dst.getY());

        return
               dst.getX() >= 0 && dst.getX() < image.getWidth() &&
               dst.getY() >= 0 && dst.getY() < image.getHeight();
    }

Is das at.createInverse() wie ein Spiegeln zu verstehen??
Und was ist der Unterschied zwischen nem Point und nem Point2D?
Hab diese Java Image API noch nicht so ganz durchschaut ....

Hoffe du kannst mir nochmal helfen. Hab auch schon alles an "Danke" verballert was geht :D


mfg,

guitarflow
 

Marco13

Top Contributor
Java:
 SwingUtilities.invokeLater(new Runnable(){
           public void run()
           {
Habs schon öfter gesehen, dass Eclipse oder NetBeans das per default so reinschreibt in die Main. Habs bei meinem eigenen Code eigentlich immer weggelassen. Was machen diese Zeilen denn??
Bzw. welchen Nachteil hats, wenn ichs nicht hinschreib??

ALLE Veränderungen an Swing-Components (und auch das Erstellen von Swing-Components) müssen vom Event-Dispatch-Thread gemacht werden. Mit den SwingUtilities legt man so eine Aufgabe in die Event-Warteschlange, die dann vom EDT abgearbeitet wird. Siehe auch Threads and Swing


Und was ist der Unterschied zwischen nem Point und nem Point2D?
Point hat zwei public ints x und y, die die Position des Punktes darstellen. Der ist praktisch geeignet um einen Punkt auf dem Bildschirm zu beschreiben. Point2D (bzw. Point2D.Float und Point2D.Double) sind für Punkte, bei denen die Koordinaten als float- oder double gepsiechert sind. Diese Punkte können einen "beliebigen" Punkt repräsentieren (also z.B. auch den Punkt (0.5, 1) der mit int's nicht darstellbar wäre).
Wenn man zufällig einen MouseEvent-Punkt hat, ist das ein "Point", den muss man, damit man vernünftig damit rechnen kann, in einen Point2D umwandeln. (Die AffinetTransform transformiert Punkte sinnvollerweise mit float- oder double-Koordinaten...)


Is das at.createInverse() wie ein Spiegeln zu verstehen??
Nein, das ist die... Inverse :D Wenn man eine AffineTransform hat, die z.B. eine Drehung um 10 Grad nach rechts und eine Verschiebung um 10 Pixel nach oben beschreibt, dann ist die Inverse dazu eben eine AffineTransform, die eine Verschiebung um 10 Pixel nach unten und eine Drehung um 10 Grad nach links beschreibt. Sozusagen "das Gegenteil". In diesem Fall wird das verwendet, um einen beliebigen Punkt "auf die Karte zurück zu rechnen". Wenn man eine Karte mit einer AffineTransform verändert, dann landet z.B. der obere linke Eckpunkt der Karte bei der Position (123, -45). Wenn man diesen Punkt mit der Inversen AffineTransform multipliziert, dann kommt da der Punkt (0,0) raus (eben gerade der obere, linke Punkt). In diesem Beispiel wird also "irgendein" Punkt (nämlich der, wo man hingeklickt hat) mit dieser Inversen transformiert. Wenn der Punkt, der dabei rauskommt, in dem Bereich (0,0)-(kartenBreite,kartenHöhe) liegt, dann hat man mit dem Klick die (gedrehte und verschobene) Karte getroffen. Die Mathematik dahinter ist ad hoc erstmal nicht so zugänglich, d.h. man verliert sich vielleicht wenn man planlos bei Koordinatentransformation ? Wikipedia lossurft, aber an sich ist es gar nicht sooo kompliziert, wenn man die AffineTransforms erstmal ansatzweise verstanden hat.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
P JComponent / JLabel drehen (Rotation) AWT, Swing, JavaFX & SWT 2
O JLabel um 90 Grad drehen ? AWT, Swing, JavaFX & SWT 9
G Problem mit der Anzeige von jLabel. Unlesbar wenn der Text geändert wird. AWT, Swing, JavaFX & SWT 28
C Swing Übergabe von String an JLabel AWT, Swing, JavaFX & SWT 9
H Swing Anpassen der Textgröße im JLabel funktioniert nur bedingt AWT, Swing, JavaFX & SWT 7
Mojo7310 Ändern von Text in JLabel mit einem Button AWT, Swing, JavaFX & SWT 11
raeuchertofu Text von JTextfield und JLabel ändern AWT, Swing, JavaFX & SWT 2
D JLabel übereinanderlegen AWT, Swing, JavaFX & SWT 5
E JLabel kopieren AWT, Swing, JavaFX & SWT 31
R Text vom Jlabel wird nicht angezeigt AWT, Swing, JavaFX & SWT 2
kodela HTML-tags für JLabel AWT, Swing, JavaFX & SWT 9
J Swing JLabel Verknüpfung aktualisieren AWT, Swing, JavaFX & SWT 3
L Swing JLabel wird beim ändern der Schriftart immer neu gezeichnet. AWT, Swing, JavaFX & SWT 2
F JLabel mit ImageIcon per Button bewegen AWT, Swing, JavaFX & SWT 4
H String teilen, damit bei JLabel keine Punkte am Ende angezeigt werden AWT, Swing, JavaFX & SWT 4
S Swing JLabel Text mit der Zeit ändern AWT, Swing, JavaFX & SWT 1
E JLabel AWT, Swing, JavaFX & SWT 1
K JLabel mit Bilder im nicht initialisierten JPanel hinzufügen AWT, Swing, JavaFX & SWT 5
I JLabel neue Zeile AWT, Swing, JavaFX & SWT 4
L JLabel ist verzerrt/größer als eingestellt AWT, Swing, JavaFX & SWT 15
B Observer Pattern JLabel ändern AWT, Swing, JavaFX & SWT 7
R Kann JLabel in ActionListener nicht aufrufen AWT, Swing, JavaFX & SWT 4
E Wie kann ich ein JLabel auf größe der schriftlänge setzten? AWT, Swing, JavaFX & SWT 2
xYurisha JLabel Text einer geöffneten Gui über eine andere Klasse ändern! AWT, Swing, JavaFX & SWT 3
D zwei JLabel stapeln in einem JPanel AWT, Swing, JavaFX & SWT 5
BreakdownBob Swing JLabel verschwindet hinter Image AWT, Swing, JavaFX & SWT 2
L Kalendar als JLabel einfügen AWT, Swing, JavaFX & SWT 3
stroggi Swing Mehrere transparentes Objekte (Grafiken) über einem Bild (JLabel) darstellen AWT, Swing, JavaFX & SWT 4
it_is_all jLabel.setText -> Char in Str: NullpointerException AWT, Swing, JavaFX & SWT 1
4 Swing JLabel wird ,,abgehackt" AWT, Swing, JavaFX & SWT 3
it_is_all JLabel.setIcon - funktioniert nicht mehr AWT, Swing, JavaFX & SWT 2
P Aktualisierung von jLabel in jFrames AWT, Swing, JavaFX & SWT 8
DaCrazyJavaExpert Swing JLabel Klickanimation AWT, Swing, JavaFX & SWT 4
A Swing JLabel als Parameter übergeben AWT, Swing, JavaFX & SWT 9
M JLabel ausrichten AWT, Swing, JavaFX & SWT 2
S JLabel setText() Problem AWT, Swing, JavaFX & SWT 6
Z jLabel von Methode aus aktualisieren AWT, Swing, JavaFX & SWT 7
T JLabel Textinhalt von Oben anzeigen AWT, Swing, JavaFX & SWT 1
N JLabel in JTabbedPane verschieben AWT, Swing, JavaFX & SWT 2
N JLabel ändern während Programm ausgeführt wird AWT, Swing, JavaFX & SWT 4
R Swing JLabel berührung an einem anderen Label prüfen AWT, Swing, JavaFX & SWT 3
Thallius Swing "..." beim JLabel verhindern? AWT, Swing, JavaFX & SWT 3
OlafHD JLabel im JFrame Zentrieren AWT, Swing, JavaFX & SWT 2
T Event Handling JLabel als eigener Button AWT, Swing, JavaFX & SWT 7
T JLabel wird nicht angezeigt AWT, Swing, JavaFX & SWT 2
KilledByCheese Swing 2D JLabel Array Maze zeichnen AWT, Swing, JavaFX & SWT 2
F Abgeleitetes JLabel AWT, Swing, JavaFX & SWT 16
D Swing keine JLabel-Aktualisierung bei "externem" Aufruf durch Helferklasse AWT, Swing, JavaFX & SWT 10
S jLabel HTML Formatierung AWT, Swing, JavaFX & SWT 2
J Statisches JLabel adden AWT, Swing, JavaFX & SWT 4
J JLabel Visible setzen in KeyListener AWT, Swing, JavaFX & SWT 13
T JLabel in die Mitte(JLabel.CENTER geht nicht) AWT, Swing, JavaFX & SWT 12
2 JLabel - setText ändert den Text nicht AWT, Swing, JavaFX & SWT 4
B Jlabel Text von anderen Klasse aus ändern AWT, Swing, JavaFX & SWT 9
stylegangsta JLabel durch Klick auf JButton einblenden AWT, Swing, JavaFX & SWT 16
stylegangsta JLabel anzuzeigenden Text zentrieren AWT, Swing, JavaFX & SWT 9
K Swing unterschied JTextField und JLabel AWT, Swing, JavaFX & SWT 7
F Swing JLabel in JFrame anpassen AWT, Swing, JavaFX & SWT 20
E JLabel aus Runnable ändern AWT, Swing, JavaFX & SWT 2
Z AWT JLabel setzt kein neuen Text AWT, Swing, JavaFX & SWT 6
D JLabel bei Aufruf neue Zeile AWT, Swing, JavaFX & SWT 5
W Swing JLabel jede Sekunde aktualisieren, ohne Timer zu benutzen AWT, Swing, JavaFX & SWT 4
F ListCellRenderer mit JLabel und MouseListener AWT, Swing, JavaFX & SWT 4
F Swing JLabel wird nicht sofort Angezeigt AWT, Swing, JavaFX & SWT 3
D Timer für Bildfolge in einem jLabel AWT, Swing, JavaFX & SWT 5
S JLabel-Text in Methode setzen? AWT, Swing, JavaFX & SWT 2
GreyFox JLabel in laufender Anwendung erzeugen AWT, Swing, JavaFX & SWT 6
H Position eines JLabel in einem JPanel AWT, Swing, JavaFX & SWT 2
Z Auf ein JLabel drauf malen? AWT, Swing, JavaFX & SWT 1
S JLabel in den Vordergrund rücken AWT, Swing, JavaFX & SWT 1
A JLabel hochzählen lassen. AWT, Swing, JavaFX & SWT 6
S JLabel mit ImageIcon, komischer Rahmen? AWT, Swing, JavaFX & SWT 2
I Swing JLabel Bild für 1sec anzeigen lassen AWT, Swing, JavaFX & SWT 13
S JLabel anordnen AWT, Swing, JavaFX & SWT 2
J ungewollt-automatische Größenänderung von JLabel AWT, Swing, JavaFX & SWT 5
H AWT Fenster- und JLabel-Größe automatisch anpassen AWT, Swing, JavaFX & SWT 2
L 2D-Grafik PNG Datei in JLabel Array Problem AWT, Swing, JavaFX & SWT 4
S Swing Exception in thread "AWT-EventQueue-0" bei Jlabel AWT, Swing, JavaFX & SWT 4
B Swing JLabel und LayoutManager AWT, Swing, JavaFX & SWT 20
M Bild aus JLabel resizen und speichern AWT, Swing, JavaFX & SWT 0
A Swing JLabel/JTextField Inhalt mit JSlider verändern AWT, Swing, JavaFX & SWT 12
O Statusleiste mit JLabel aktualisieren AWT, Swing, JavaFX & SWT 5
G jLabel aus anderem Fenster ändern AWT, Swing, JavaFX & SWT 7
T JLabel Nullpunkt verschieben AWT, Swing, JavaFX & SWT 6
F Swing JPanel mit JLabel spinnt? AWT, Swing, JavaFX & SWT 2
M Massenweise JLabel ? oder ? AWT, Swing, JavaFX & SWT 3
J JLabel während einer methode aktualisieren AWT, Swing, JavaFX & SWT 6
R AWT JLabel oder JButton aktualisieren AWT, Swing, JavaFX & SWT 1
N JLabel HTML mit custom Font AWT, Swing, JavaFX & SWT 0
V JLabel bzw. GUI aus externen Thread ansteuern AWT, Swing, JavaFX & SWT 3
G JLabel verdeckt AWT, Swing, JavaFX & SWT 12
A JLabel zeichendicke einstellen AWT, Swing, JavaFX & SWT 3
B Repaint auf JFrame, JLabel und ImageIcon AWT, Swing, JavaFX & SWT 4
S JLabel zeigt ImageIcon nicht AWT, Swing, JavaFX & SWT 13
S jLabel / jText beim starten ausblenden AWT, Swing, JavaFX & SWT 2
A Array von JLabel mit Grafiken anzeigen AWT, Swing, JavaFX & SWT 3
T Swing HTML Text aus JLabel ohne "HTML-Tags" in String einlesen AWT, Swing, JavaFX & SWT 5
A JLabel im Listener färben AWT, Swing, JavaFX & SWT 11
M JLabel über JButtons zeichnen AWT, Swing, JavaFX & SWT 4
T Kreis in Jlabel darstellen. AWT, Swing, JavaFX & SWT 13

Ähnliche Java Themen


Oben