zoukankan      html  css  js  c++  java
  • Java小项目——画板

    一、项目设计

    1、界面设计JFrame

    上面为控制栏JPanel

    下面为绘制面板JPanel

    2、基本思路

    鼠标事件监听器:展示一个JPanel界面,添加鼠标事件监听器,记录鼠标点击事件的坐标;

    getGraphics()方法:可以得到java.awt.Graphics对象gr,gr作为“画笔”,调用draw方法并传入坐标,可以绘制相应图像。

    3、具体功能

    3.1、绘制方式:

    画直线:

    g.drawLine(x1,y1,x2,y2);
    

    画矩形:

    g.drawRect(Math.min(x1,x2),Math.min(y1,y2),Math.abs(x2-x1),Math.abs(y2-y1));
    

    画多边形:

    若为第一次单击,保存该坐标;非第一次单击,绘制两个单击之间的直线;双击后,绘制第一次单击和本次双击坐标之间的直线,再清空所有坐标。

    拖动绘制:

    拖动鼠标中,不断绘制两个时间段之间两点之间的直线。

    3.2、颜色选取

    可以选取对应颜色进行绘制

    3.3、重绘

    改变窗体大小时,会调用paint()函数,重绘。

    需要在这个函数中,写重绘已经绘制的图形的代码,用于保存代码。

    二、初步实现:界面实现+绘制直线

    1、主函数部分:SampleDraw类

    package Draw;
    
    import javax.swing.*;
    import java.awt.*;
    
    public class SampleDraw {
        public static void main(String args[]){
            SampleDraw draw = new SampleDraw();
            draw.showUI();
        }
    
        public void showUI(){
    //主界面JFrame jf
            JFrame jf = new JFrame();
            jf.setTitle("画图1.0");
            jf.setSize(800,800);
            jf.setLocationRelativeTo(null);
            jf.setDefaultCloseOperation(3);
            
            //控制台JPanel jp1+绘制界面JPanel jp2
            JPanel jp1 = new JPanel();
            JPanel jp2 = new JPanel();
            jp1.setBackground(java.awt.Color.white);
            jf.add(jp1,BorderLayout.NORTH);
            jf.add(jp2,BorderLayout.CENTER);
            
            //按钮:前四个为绘制功能;后两个为颜色选择
            JButton jb1 = new JButton("线条");
            JButton jb2 = new JButton("矩形");
            JButton jb3 = new JButton("多边形");
            JButton jb4 = new JButton("绘制");
            JButton jbc1 = new JButton("蓝色");
            JButton jbc2 = new JButton("绿色");
            jbc1.setForeground(Color.blue);
            jbc2.setForeground(Color.green);
            //jbc2.setBackground(Color.BLACK);
            jp1.add(jb1);
            jp1.add(jb2);
            jp1.add(jb3);
            jp1.add(jb4);
            jp1.add(jbc1);
            jp1.add(jbc2);
    
            //可见放在add之后,获取之前
            jf.setVisible(true);
    
            //创建画笔对象gr,添加至jpanel2
            Graphics gr = jp2.getGraphics();
            
            //鼠标监听器对象mouse
            //构造函数初始化:使得鼠标中的位置,获取画布类:鼠标在画布上画线、多边形
            addMouseListener mouse = new addMouseListener(gr);
            
            //在各个事件源(组件)上添加鼠标监听器
            jp2.addMouseListener(mouse);
            jp2.addMouseMotionListener(mouse);
            jb1.addActionListener(mouse);
            jb2.addActionListener(mouse);
            jb3.addActionListener(mouse);
            jb4.addActionListener(mouse);
            jbc1.addActionListener(mouse);
            jbc2.addActionListener(mouse);
        }
    }
    

    2、事件监听器部分:addMouseListener类,继承了多个动作监听器接口

    2.1 首先在actionPerformed获取目前状态:画什么,用什么颜色

    2.2 在按下去 mousePressed 、释放 mouseReleased 中分别获取当时坐标,用绘制函数绘制直线和矩形

    if(name.equals("线条")||name.equals("")){
        g.drawLine(x1,y1,x2,y2);
    }
    
    if(name.equals("矩形")){
        g.drawRect(Math.min(x1,x2),Math.min(y1,y2),Math.abs(x2-x1),Math.abs(y2-y1));
    }

    2.3 在点击mouseClicked中,用函数绘制多边形

    第一次按压记录x0;第二次之后,绘制上一次点击位置和本次点击位置的直线;双击之后,绘制上一次点击位置和本次点击位置+x0与本次双击点的两条直线

    g.drawLine(x01,y01,x02,y02); //第二次以后
    g.drawLine(x0,y0,x01,y01);   //双击后
    

    2.4 在拖拽mouseDragged中,绘制两次时间段之间的直线,时间段片小,则显示出来像曲线

    g.drawLine(x,y,xt,yt);
    package Draw;
    
    import javax.swing.*;
    import java.awt.*;
    import java.awt.event.ActionEvent;
    import java.awt.event.ActionListener;
    import java.awt.event.MouseEvent;
    import java.awt.event.MouseListener;
    import java.awt.event.MouseMotionListener;
    
    //MouseListener:绘制直线、矩形、多边形
    //ActionListener:动作监听,获取button的text,来判断目前处于何种工作状态
    //MouseMotionListener:拖动画线
    public class addMouseListener implements MouseListener ,ActionListener , MouseMotionListener{
        //变量如下,分别对应直线、矩形x1,x2;多边形x01,x02,x0;
        private int x1=0,y1=0,x2=0,y2=0;
        private int x=0,y=0,xt=0,yt=0;
        private int x01,y01,x02,y02,x0,y0;
        private Graphics g ;
        private String name = "线条";
        private Color jbcolor = Color.BLACK;
    
        //构造方法,将画笔传入监听器,在监听器中动作执行时,绘制相应图形
        public addMouseListener(Graphics gr){
            this.g = gr;
        }
    
        @Override
        public void mouseClicked(MouseEvent e) {
            //3-多边形绘制,最后双击,在第一个点和最后一个点的位置之间画线
            if(e.getClickCount()==2){
                System.out.println("双击!!");
                g.drawLine(x0,y0,x01,y01);
                x0=0;y0=0;x01=0;y01=0;x02=0;y02=0;
            }
            System.out.println("点击");
    
            if(name.equals("多边形")) {
                //当第一次按压,x01=0时候,则将第一次按压坐标存入x0,y0
                if(x01==0){
                    x0 = e.getX();
                    y0 = e.getY();
                }
                x01 = e.getX();
                y01 = e.getY();
                //第二次按压以后画线
                if (x02!=0){
                    g.drawLine(x01,y01,x02,y02);
    
                }
                //将本次x01按压存入x02,用于下一次绘制直线
                x02 = x01;
                y02 = y01;
            }
    
        }
    
        @Override
        public void mousePressed(MouseEvent e) {
            //1线条-2矩形-按压时为绘制起点x1,y1
            System.out.println("按压");
            x1 = e.getX();
            y1 = e.getY();
    
        }
    
        @Override
        public void mouseReleased(MouseEvent e) {
            //1线条-2矩形-释放时为绘制终点x2,y2
            System.out.println("释放");
            x=0;y=0;xt=0;yt=0;
            x2 = e.getX();
            y2 = e.getY();
            //
            if(name.equals("线条")||name.equals("")){
                g.drawLine(x1,y1,x2,y2);
            }
    
            if(name.equals("矩形")){
                g.drawRect(Math.min(x1,x2),Math.min(y1,y2),Math.abs(x2-x1),Math.abs(y2-y1));
            }
        }
    
        @Override
        public void mouseEntered(MouseEvent e) {
            System.out.println("进入");
        }
    
        @Override
        public void mouseExited(MouseEvent e) {
            System.out.println("离开");
        }
    
    
        @Override
        public void mouseDragged(MouseEvent e) {
            //拖动绘制
            if(name.equals("绘制")){
                x = e.getX();
                y = e.getY();
                if(xt!=0){
                    g.drawLine(x,y,xt,yt);
                }
                xt = x;
                yt = y;
            }
    
        }
    
        @Override
        public void mouseMoved(MouseEvent e) {
    
        }
    
        @Override
        public void actionPerformed(ActionEvent e) {
    
            if(e.getActionCommand().equals("蓝色")){
                //定义新的jb按钮对象,来接受e.getSource()的object对象,强制转型为JButton
                JButton jb = (JButton)e.getSource();
                jbcolor = jb.getForeground();
                g.setColor(jbcolor);
            }else if(e.getActionCommand().equals("绿色")){
                JButton jb = (JButton)e.getSource();
                jbcolor = jb.getForeground();
                g.setColor(jbcolor);
            }else{
                name = e.getActionCommand();
            }
    
            System.out.println("button:"+name);
    
        }
    }
    

       结果:如下   

    三、重绘    ❤️【本章节三个类组成最终代码】❤️

    1、需要重绘,则需要数组存储下绘制过的线条

    定义Shape类

    属性:包括初始坐标、终止坐标、绘制形状、颜色

    方法:重新画方法,调用该方法,将shape中的对象重新绘制出来

    package com.java7.DrawRedraw0710;
    
    import java.awt.*;
    
    public class Shape {
        private int x1,y1,x2,y2;
        private String name;
        private Color color;
    
        public Shape(int x1,int y1,int x2,int y2,String name,Color color){
            this.x1 = x1;
            this.y1 = y1;
            this.x2 = x2;
            this.y2 = y2;
            this.name = name;
            this.color = color;
        }
        //重绘中,调用Redraw,则绘制shape对象
        public void Redraw(Graphics g){
            System.out.println("Graphics name = "+name);
            g.setColor(color);
            switch (name){
                case "线条":
                    g.drawLine(x1,y1,x2,y2);
                    break;
                case "矩形":
                    g.drawRect(Math.min(x1,x2),Math.min(y1,y2),Math.abs(x2-x1),Math.abs(y2-y1));
                    break;
            }
        }
    }
    

    2、定义好shape类后,我们需要在在监听器类,绘制的时候,将坐标、颜色、绘制种类存入Shape,再将Shape存入ShapeArray[]数组。

    //1.把线条坐标、名称存入Shape
    Shape shape = new Shape(x1,y1,x2,y2,name,jbcolor);
    shapeArray[index++] = shape;
    //3.1.单击后,把多边形的初始坐标、名称存入Shape
    Shape shape = new Shape(x01,y01,x02,y02,"线条",jbcolor);
    shapeArray[index++] = shape;
    //3.2.双击后,把多边形的初次按下坐标+最后按压的坐标、名称存入Shape;清空所有坐标,用于下一次判断是否为第一次按压
    Shape shape = new Shape(x0,y0,x01,y01,"线条",jbcolor);
    shapeArray[index++] = shape;
    x0=0;y0=0;x01=0;y01=0;x02=0;y02=0;
    

    源码:addMouseListener类

    package com.java7.DrawRedraw0710;
    
    import javax.swing.*;
    import java.awt.*;
    import java.awt.event.*;
    
    public class addMouseListener implements MouseListener ,ActionListener , MouseMotionListener{
    
        private int x1=0,y1=0,x2=0,y2=0;
        private int x=0,y=0,xt=0,yt=0;
        private int x01,y01,x02,y02,x0,y0;
        private Graphics g ;
        private String name = "线条";
        private Color jbcolor=Color.BLACK;
    
        //重绘:
        private Shape shapeArray[];
        private int index = 0;
        //set函数,使用后,初始化画笔gr,shapeArray[]数组,用于将其传入addMouseListener
        public void setGr(Graphics gr){
            this.g = gr;
        }
        public void setSa(Shape shapeArray[]){
            this.shapeArray = shapeArray;
        }
    
        @Override
        public void mouseClicked(MouseEvent e) {
            if(e.getClickCount()==2){
                g.drawLine(x0,y0,x01,y01);
                //3.2.双击后,把多边形的初次按下坐标+最后按压的坐标、名称存入Shape
                Shape shape = new Shape(x0,y0,x01,y01,"线条",jbcolor);
                shapeArray[index++] = shape;
                x0=0;y0=0;x01=0;y01=0;x02=0;y02=0;
            }
        }
    
        public void mousePressed(MouseEvent e) {
            //System.out.println("按压");
            x1 = e.getX();
            y1 = e.getY();
    
        }
    
        public void mouseReleased(MouseEvent e) {
            x=0;y=0;xt=0;yt=0;
            x2 = e.getX();
            y2 = e.getY();
            System.out.println("name="+name);
            if(name.equals("线条")||name.equals("")){
                g.drawLine(x1,y1,x2,y2);
                //1.把线条坐标、名称存入Shape
                Shape shape = new Shape(x1,y1,x2,y2,name,jbcolor);
                shapeArray[index++] = shape;
            }
    
            if(name.equals("矩形")){
                g.drawRect(Math.min(x1,x2),Math.min(y1,y2),Math.abs(x2-x1),Math.abs(y2-y1));
                //2.把矩形坐标、名称存入Shape
                Shape shape = new Shape(x1,y1,x2,y2,name,jbcolor);
                shapeArray[index++] = shape;
            }
    
            if(name.equals("多边形")) {
                System.out.println("多边形");
                if(x01==0){//第一次按下,记录地刺次按下的位置,用于双击后,多边形闭合
                    x0 = e.getX();
                    y0 = e.getY();
                }
                x01 = e.getX();
                y01 = e.getY();
    
                if (x02!=0){//第二次按压
                    g.drawLine(x01,y01,x02,y02);
                    //3.1.单击后,把多边形的初始坐标、名称存入Shape
                    Shape shape = new Shape(x01,y01,x02,y02,"线条",jbcolor);
                    shapeArray[index++] = shape;
                }
                x02 = x01;
                y02 = y01;
            }
        }
    
        public void mouseEntered(MouseEvent e) {
            ///System.out.println("进入");
        }
    
        public void mouseExited(MouseEvent e) {
            //System.out.println("离开");
        }
    
    
        public void mouseDragged(MouseEvent e) {
            if(name.equals("绘制")){
                System.out.println("DRAGE —— 拖动绘制曲线:");
                x = e.getX();
                y = e.getY();
                if(xt!=0){
                    g.drawLine(x,y,xt,yt);
                    //把坐标、名称存入Shape
                    Shape shape = new Shape(x,y,xt,yt,"线条",jbcolor);
                    shapeArray[index++] = shape;
                }
                xt = x;
                yt = y;
            }
        }
    
        public void mouseMoved(MouseEvent e) {
        }
    
        public void actionPerformed(ActionEvent e) {
            //首先获取jb对象(JButton)e.getSource();再对象的颜色jb.getBackground()
            if("".equals(e.getActionCommand())){
                JButton jb = (JButton)e.getSource();
                jbcolor = jb.getBackground();
                g.setColor(jbcolor);
            }
            //获取动作:直线、矩形、多边形、拖动绘制线条
            else{
                name = e.getActionCommand();
                System.out.println("button:"+name);
            }
        }
    }
    

    3、 主函数,继承JPanel,在重写panit()函数时,写入:绘制shapeArray[]数组中所有的Shape对象。

    @Override
    public void paint(Graphics g) {
        //调用重绘
        super.paint(g);
        for(int i=0;i<shapeArray.length;i++){
            if(shapeArray[i]!=null){
                shapeArray[i].Redraw(g);
            }
            else
                break;
        }
        System.out.println("重绘");
    }
    

    源码:SampleDraw

    package com.java7.DrawRedraw0710;
    
    import javax.swing.*;
    import java.awt.*;
    
    public class SampleDraw extends JPanel{
        public static void main(String args[]){
            SampleDraw draw = new SampleDraw();
            draw.showUI();
        }
        private Shape shapeArray[]=new Shape[1000];
    
        @Override
        public void paint(Graphics g) {
            //调用重绘
            super.paint(g);
            for(int i=0;i<shapeArray.length;i++){
                if(shapeArray[i]!=null){
                    shapeArray[i].Redraw(g);
                }
                else
                    break;
            }
            System.out.println("重绘");
        }
    
        public void showUI(){
            //界面
            JFrame jf = new JFrame();
            jf.setTitle("画图1.0");
            jf.setSize(800,800);
            jf.setLocationRelativeTo(null);
            jf.setDefaultCloseOperation(3);
    
            //JPanel
            JPanel jp1 = new JPanel();
            jp1.setBackground(Color.white);
            jf.add(jp1,BorderLayout.NORTH);
            jf.add(this,BorderLayout.CENTER);
    
            //按钮
            JButton jb1 = new JButton("线条");
            JButton jb2 = new JButton("矩形");
            JButton jb3 = new JButton("多边形");
            JButton jb4 = new JButton("绘制");
    
            jp1.add(jb1);
            jp1.add(jb2);
            jp1.add(jb3);
            jp1.add(jb4);
    
            //鼠标监听器,初始化设置shapeArray
            addMouseListener mouse = new addMouseListener();
    
            //颜色数组,循环添加颜色按钮
            Color colorArray[] = {Color.GREEN,Color.BLUE,Color.CYAN};
            for(int i=0;i<colorArray.length;i++){
                JButton jb = new JButton();
                jb.setBackground(colorArray[i]);
                jb.setForeground(colorArray[i]);
                jp1.add(jb);
                jb.addActionListener(mouse);
            }
    
            //可见放在add之后,获取之前
            jf.setVisible(true);
    
            //画布,添加至jpanel2
            Graphics gr = this.getGraphics();
            System.out.println("graphics = " + gr);
    
            //添加鼠标监听器
            this.addMouseListener(mouse);
            this.addMouseMotionListener(mouse);
    
            jb1.addActionListener(mouse);
            jb2.addActionListener(mouse);
            jb3.addActionListener(mouse);
            jb4.addActionListener(mouse);
    
            mouse.setGr(gr);//使得鼠标中的位置,获取画布类:鼠标在画布上画线、多边形
            mouse.setSa(shapeArray);//获得shapeArray,用于paint中重绘
    
        }
    
    }
    

    结果:

  • 相关阅读:
    图片处理连环画特效
    卡片翻页算法
    android 自定义属性
    android 中捕获全局异常
    c++ 学习笔记
    图片怀旧特效处理
    Linux 网络配置
    指针参数传递
    python 读写文件
    PopupWindow 点击外面取消
  • 原文地址:https://www.cnblogs.com/iriswang/p/11084649.html
Copyright © 2011-2022 走看看