【发布时间】:2015-07-08 02:16:40
【问题描述】:
我正在尝试创建一个绘图程序,它允许您移动一个形状并将该形状绘制到JFrame 上。但是,当我绘画时,它不会留下痕迹。我认为它不会留下痕迹,因为我正在覆盖绘制方法,并且当我想绘制一个新对象时,它会清除我之前绘制的对象。但是当我删除覆盖它的代码并运行程序时,形状会留下痕迹,但框架顶部的按钮会变得一团糟。
MovingBlockFrame.java:
package MovingBlock;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.Graphics2D;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import javax.swing.*;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
public class MovingBlockFrame extends JFrame implements KeyListener, ActionListener, ChangeListener{
FlowLayout flo = new FlowLayout(FlowLayout.LEADING, 10, 10);
JPanel buttonPanel = new JPanel();
JButton circle = new JButton("Circle");
JButton rectangle = new JButton("Rectangle");
JButton color = new JButton("Color");
JSlider speed = new JSlider(0, 100, 10);
Graphics2D test3;
int desiredWidth = 780;
int desiredHeight = 800;
MovingBlockCanvas background = new MovingBlockCanvas();
int paintSpeed = 10;
boolean diagonalKeysDown;
boolean diagonalKeysUp;
public MovingBlockFrame(){
setTitle("Painter");
buttonPanel.setBorder(BorderFactory.createLineBorder(Color.black));
setSize(desiredWidth, desiredHeight);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
KeyListener();
buttonPanel.setLayout(flo);
buttonPanel.add(circle);
buttonPanel.add(rectangle);
buttonPanel.add(color);
buttonPanel.add(speed);
background.desiredShape = "Circle";
background.addKeyListener(this);
add(background);
add(buttonPanel, BorderLayout.NORTH);
setVisible(true);
}
public static void main(String[] args){
MovingBlockFrame frame = new MovingBlockFrame();
frame.addKeyListener(frame);
}
public void keyPressed(KeyEvent input) {
int pressed = input.getKeyCode();
if(pressed == 83){
diagonalKeysDown = true;
background.xValue+= paintSpeed;
background.repaint();
}else if(pressed == 68 && !diagonalKeysDown){
background.yValue+=paintSpeed;
background.repaint();
}else if(pressed == 87 ){
diagonalKeysUp = true;
background.xValue-=paintSpeed;
background.repaint();
}else if(pressed == 65 ){
background.yValue-=paintSpeed;
background.repaint();
} if(diagonalKeysDown && pressed == 68){
background.yValue+=paintSpeed;
background.xValue+= paintSpeed;
background.repaint();
} if(diagonalKeysUp && pressed == 68){
background.yValue+=paintSpeed;
background.xValue-= paintSpeed;
background.repaint();
} if(diagonalKeysDown && pressed == 65){
background.yValue-=paintSpeed;
background.xValue+= paintSpeed;
background.repaint();
} if(diagonalKeysUp && pressed == 65){
background.yValue-=paintSpeed;
background.xValue-= paintSpeed;
background.repaint();
}
}
public void keyReleased(KeyEvent input) {
int released = input.getKeyCode();
if(released == 83){
diagonalKeysDown = false;
} if(released == 87){
diagonalKeysUp = false;
}
}
public void keyTyped(KeyEvent input) {
}
public void actionPerformed(ActionEvent event) {
String command = event.getActionCommand();
if(command.equals("Circle")){
background.desiredShape = "Circle";
background.repaint();
}else if (command.equals("Rectangle")){
background.desiredShape = "Rectangle";
background.repaint();
}else if(command.equals("Color")){
//ColorSliders test = new ColorSliders();
}
}
public void stateChanged(ChangeEvent event) {
JSlider source = (JSlider) event.getSource();
if(source.getValueIsAdjusting() != true){
paintSpeed = source.getValue();
}
}
public void KeyListener(){
buttonPanel.addKeyListener(this);
circle.addKeyListener(this);
rectangle.addKeyListener(this);
circle.addActionListener(this);
rectangle.addActionListener(this);
color.addKeyListener(this);
color.addActionListener(this);
speed.addKeyListener(this);
speed.addChangeListener(this);
}
}
MovingBlockCanvas.java:
package MovingBlock;
import java.awt.Graphics;
import javax.swing.JPanel;
import javax.swing.JPanel;
public class MovingBlockCanvas extends JPanel {
int xValue = 50;
int yValue = 50;
String desiredShape;
public MovingBlockCanvas(){
}
public void paint(Graphics render){
//super.paint(render);
render.setColor(ColorPanel.getColor2());
if(desiredShape == "Rectangle"){
render.fillRect(yValue, xValue, 50, 50);
}
if(desiredShape == "Circle"){
render.fillOval(yValue, xValue, 50, 50);
}
}
}
问题出在paint方法中,当我重写该方法时,我不会为形状留下痕迹。另外,如果您有任何编码技巧,我很想听听。提前谢谢你。
【问题讨论】:
-
我想到了至少两个选项,要么绘制到屏幕外
BufferedImage,然后将此图像绘制到屏幕上,然后它不会被清除或以某种形式存储一系列点List并通过在点之间画线来连接点 -
你知道我在哪里可以阅读 BuffereImages,因为目前我对它们知之甚少。
-
Working with Images。
BufferedImage允许您通过多种不同方式访问它的“内容”,但在您的情况下最有用的是通过它的Graphics上下文,它允许您像在屏幕上一样绘制图像跨度> -
谢谢你,我会阅读缓冲图像。