【问题标题】:Display moving image in Java Swing在 Java Swing 中显示动态图像
【发布时间】:2014-12-18 14:13:47
【问题描述】:

我正在尝试在 Java Swing 中显示高宽度图像(比如 2000x100,如心率条)。我只需要显示一个宽度为 500 的窗口,它会稍微向左移动。我当前的代码(有点复杂,它也有不必要的动画)做到了,但我需要添加的功能是:图像的结尾应该与图像的开头连接。所以它总是一遍又一遍地重复显示。

总之,我需要加入image的两端!我该怎么做?

import java.awt.AlphaComposite;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.LinearGradientPaint;
import java.awt.Rectangle;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.geom.Point2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;

import javax.imageio.ImageIO;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.Timer;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;

public class SlidingAnimation {

    public static void main(String[] args) {
        new SlidingAnimation();
    }

    public SlidingAnimation() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                    ex.printStackTrace();
                }

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                JPanel j = new JPanel();
                j.add(new AnimatedBar(true));
                frame.add(j);
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }
}

class AnimatedBar extends JPanel {

    private BufferedImage img;

    private Timer timer;
    private long startTime = -1;
    private int playTime = 4000;
    private int window = 500;
    private int moveX=0;
    public static boolean keepRunning=true;

    private float progress;

    public AnimatedBar(boolean x) {
        try {
            if(x)
                img = ImageIO.read(new File("strip2.jpg"));
            else
                img=null;
        } catch (IOException ex) {
            ex.printStackTrace();
        }
        if(x){
            timer = new Timer(40, new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    if (startTime == -1) {
                        startTime = System.currentTimeMillis();
                    } else {
                        long currentTime = System.currentTimeMillis();
                        long diff = currentTime - startTime;

                        if (diff >= playTime) {
                            diff = 0;
                            startTime = -1;
                        }
                        progress = diff / (float) playTime;
                    }

                    repaint();
                }
            });
            timer.start();
        }
    }

    @Override
    public Dimension getPreferredSize() {
        return img == null ? new Dimension(50, 50) : new Dimension(img.getWidth()/3, img.getHeight());
    }

    protected BufferedImage generateImage() {

        BufferedImage buffer = new BufferedImage(img.getWidth(), img.getHeight(), BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2d = buffer.createGraphics();
        g2d.setBackground(new Color(0, 0, 0, 0));
        g2d.clearRect(0, 0, buffer.getWidth(), buffer.getHeight());
        //            g2d.drawImage(img, 500, 0, this);
        g2d.drawImage(img,0,0,500,100,(moveX++),0,window+moveX,100,this);


        float startAt = progress- 0.05f;
        float endAt = progress + 0.05f;

        if (endAt <= 0.1f) {
            startAt = 0;
            endAt = Math.max(0.1f, progress);
        } else if (endAt >= 1f) {
            endAt = 1f;
            startAt = progress;
        }

        LinearGradientPaint lgp = new LinearGradientPaint(
                new Point2D.Float(0, 0),
                new Point2D.Float(img.getWidth(), 0),
                new float[]{startAt, endAt},
                new Color[]{new Color(0, 0, 0, 0), Color.RED});

        g2d.setPaint(lgp);

        g2d.setComposite(AlphaComposite.DstOut.derive(1f));
        g2d.fill(new Rectangle(0, 0, img.getWidth(), img.getHeight()));
        g2d.dispose();

        return buffer;

    }

    public void setImg(BufferedImage img) {
        this.img = img;
    }

    @Override
    protected void paintComponent(Graphics g) {
        if(keepRunning==false){
            img=null;
        }
        else{
            try {
                img = ImageIO.read(new File("strip2.jpg"));
            } catch (IOException e) {
            }
        }
            super.paintComponent(g);
            if (img != null) {
                Graphics2D g2d = (Graphics2D) g.create();
                int y = (getHeight() - img.getHeight()) / 2;
                int x = (getWidth() - img.getWidth()/3) / 2;
                g2d.drawImage(generateImage(), x, y, this);

                g2d.dispose();
        }
    }

}

【问题讨论】:

标签: java image swing


【解决方案1】:

要连接末端,请按照this answer 中的方式绘制两次图像。第一次绘制将是图像的结尾。第二次绘制将是图像的起点,偏移到终点的宽度。

例如

import java.awt.*;
import java.awt.event.*;
import java.awt.image.BufferedImage;
import javax.swing.*;

import java.net.URL;
import javax.imageio.ImageIO;

public class HeartBeat {

    public static void main(String[] args) throws Exception {
        URL url = new URL("http://i.stack.imgur.com/i8UJD.jpg");
        final BufferedImage bi = ImageIO.read(url);
        Runnable r = new Runnable() {

            @Override
            public void run() {
                final BufferedImage canvas = new BufferedImage(
                        bi.getWidth(), bi.getHeight(),
                        BufferedImage.TYPE_INT_RGB);
                final JLabel animationLabel = new JLabel(new ImageIcon(canvas));
                ActionListener animator = new ActionListener() {

                    int x = 0;

                    @Override
                    public void actionPerformed(ActionEvent e) {
                        Graphics2D g = canvas.createGraphics();

                        // paint last part of image in left of canvas
                        g.drawImage(bi, x, 0, null);
                        // paint first part of image immediately to the right
                        g.drawImage(bi, x + bi.getWidth(), 0, null);

                        // reset x to prevent hitting integer overflow
                        if (x%bi.getWidth()==0) x = 0;

                        g.dispose();
                        animationLabel.repaint();
                        x--;
                    }
                };
                Timer timer = new Timer(40, animator);
                timer.start();
                JOptionPane.showMessageDialog(null, animationLabel);
                timer.stop();
            }
        };
        // Swing GUIs should be created and updated on the EDT
        // http://docs.oracle.com/javase/tutorial/uiswing/concurrency
        SwingUtilities.invokeLater(r);
    }
}

【讨论】:

  • 您能否添加一两条评论来获得偏移量,解释那里到底发生了什么?我认为这对于试图了解图像如何换行的人很有用。
  • 作为一个简短的说明,根据这个显示器的用途,你可能想在它修改为 0 时重置 x...否则我认为你会遇到负整数溢出大概三年……可能没关系,但也许吧。
  • 这个不错!但是你可以添加一个带有值的选项(也许是一个计时器?)来指定以离散方式逐步移动而不是连续移动?比如,当 'step=0' 时它可能像现在一样,而当 'step=10' 时,它每 10 个像素移动一次。也许你可以添加一些 cmets 来描述逻辑可能很棒:)
  • “还有一些描述逻辑的cmets” 是的,我加了。真的没有那么多逻辑可以解释。 “但是你能添加一个选项吗..” 我将把它留给最终用户作为练习。无论如何,似乎 “我需要加入图像的两端!我该怎么做?” 已经得到了回答(使用两种不同的方法)。
  • @AndrewThompson:我面临一个愚蠢的问题!如何将动态图像添加到外部 jpanel?像 jp1.add(HeartBeat)
【解决方案2】:

图像的结尾应与图像的开头连接。所以它总是重复显示

查看Marquee Panel。您可以将JLabelImageIcon 添加到MarqueePanel

MarqueePanel 提供了多种自定义滚动的方法。

编辑:

基本代码是:

MarqueePanel panel = new MarqueePanel();
panel.setWrap(true);
panel.setWrapAmount(0);
panel.setPreferredWidth(250);

JLabel label = new JLabel( new ImageIcon( "heartbeat.jpg" ) );
panel.add( label );

frame.add( panel );

如果您希望图像在显示时完全显示在左侧,您可以更改startScrolling() 方法并使用scrollOffset = 0;

【讨论】:

  • 谢谢。但可以看出,当文本结束并消失时,它又开始了。像一个圆圈,我需要文本 start ASA 它结束,所以结尾粘在开头!
  • 嗯,'wrap' 选项可能会有所帮助。我希望应该连接在一起的两端之间没有空间。
  • @TinaJasmin,如果您不希望有间隙,您还必须在使用 wrap 时将 wrap amount 设置为 0。
  • Is it possible to first fill the display window, - 在我所做的编辑中已经回答了。
  • 好的,得到我的答案! setScrollWhenFocused :)
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-06-08
  • 1970-01-01
  • 2015-05-16
  • 2017-05-20
  • 1970-01-01
  • 2010-10-09
  • 2010-09-07
相关资源
最近更新 更多