You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
314 lines
9.6 KiB
314 lines
9.6 KiB
3 years ago
|
package sig.engine;
|
||
|
import java.awt.Graphics;
|
||
|
import java.awt.GraphicsConfiguration;
|
||
|
import java.awt.GraphicsEnvironment;
|
||
|
import java.awt.Image;
|
||
|
import java.awt.Toolkit;
|
||
|
import java.awt.image.ColorModel;
|
||
|
import java.awt.image.MemoryImageSource;
|
||
|
import java.util.ArrayList;
|
||
|
import java.util.List;
|
||
|
|
||
|
import javax.swing.JFrame;
|
||
|
import javax.swing.JPanel;
|
||
|
|
||
|
import sig.JavaProjectTemplate;
|
||
|
|
||
|
public class Panel extends JPanel implements Runnable {
|
||
|
JFrame window;
|
||
|
public int pixel[];
|
||
|
public int width=1280;
|
||
|
public int height=720;
|
||
|
final int CIRCLE_PRECISION=32;
|
||
|
final int OUTLINE_COL=Color.BRIGHT_WHITE.getColor();
|
||
|
private Thread thread;
|
||
|
private Image imageBuffer;
|
||
|
private MemoryImageSource mImageProducer;
|
||
|
private ColorModel cm;
|
||
|
int scanLine=0;
|
||
|
int nextScanLine=0;
|
||
|
double x_offset=0;
|
||
|
double y_offset=0;
|
||
|
int frameCount=0;
|
||
|
long lastSecond=0;
|
||
|
int lastFrameCount=0;
|
||
|
|
||
|
public Panel(JFrame f) {
|
||
|
super(true);
|
||
|
this.window=f;
|
||
|
thread = new Thread(this, "MyPanel Thread");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get Best Color model available for current screen.
|
||
|
* @return color model
|
||
|
*/
|
||
|
protected static ColorModel getCompatibleColorModel(){
|
||
|
GraphicsConfiguration gfx_config = GraphicsEnvironment.
|
||
|
getLocalGraphicsEnvironment().getDefaultScreenDevice().
|
||
|
getDefaultConfiguration();
|
||
|
return gfx_config.getColorModel();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Call it after been visible and after resizes.
|
||
|
*/
|
||
|
public void init(){
|
||
|
cm = getCompatibleColorModel();
|
||
|
int screenSize = width * height;
|
||
|
if(pixel == null || pixel.length < screenSize){
|
||
|
pixel = new int[screenSize];
|
||
|
}
|
||
|
if(thread.isInterrupted() || !thread.isAlive()){
|
||
|
thread.start();
|
||
|
}
|
||
|
mImageProducer = new MemoryImageSource(width, height, cm, pixel,0, width);
|
||
|
mImageProducer.setAnimated(true);
|
||
|
mImageProducer.setFullBufferUpdates(true);
|
||
|
imageBuffer = Toolkit.getDefaultToolkit().createImage(mImageProducer);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void paintComponent(Graphics g) {
|
||
|
super.paintComponent(g);
|
||
|
// perform draws on pixels
|
||
|
render();
|
||
|
// ask ImageProducer to update image
|
||
|
mImageProducer.newPixels();
|
||
|
// draw it on panel
|
||
|
g.drawImage(this.imageBuffer, 0, 0, this);
|
||
|
|
||
|
|
||
|
if (window!=null&&System.currentTimeMillis()-lastSecond>=1000) {
|
||
|
window.setTitle(JavaProjectTemplate.PROGRAM_NAME+" - FPS: "+(frameCount-lastFrameCount));
|
||
|
lastFrameCount=frameCount;
|
||
|
lastSecond=System.currentTimeMillis();
|
||
|
}
|
||
|
frameCount++;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Overrides ImageObserver.imageUpdate.
|
||
|
* Always return true, assuming that imageBuffer is ready to go when called
|
||
|
*/
|
||
|
@Override
|
||
|
public boolean imageUpdate(Image image, int a, int b, int c, int d, int e) {
|
||
|
return true;
|
||
|
}
|
||
|
/**
|
||
|
* Do your draws in here !!
|
||
|
* pixel is your canvas!
|
||
|
*/
|
||
|
public /* abstract */ void render(){
|
||
|
int[] p = pixel; // this avoid crash when resizing
|
||
|
//a=h/w
|
||
|
|
||
|
for (int x=0;x<width;x++) {
|
||
|
for (int y=0;y<height;y++) {
|
||
|
p[y*width+x]=(0<<16)+(0<<8)+0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
x_offset+=1;
|
||
|
y_offset=50;
|
||
|
|
||
|
FillPolygon(p,Color.WHITE,50,50,new Point[] {
|
||
|
new Point(135,2),
|
||
|
new Point(166,96),
|
||
|
new Point(265,97),
|
||
|
new Point(185,156),
|
||
|
new Point(215,251),
|
||
|
new Point(134,192),
|
||
|
new Point(54,251),
|
||
|
new Point(84,156),
|
||
|
new Point(4,97),
|
||
|
new Point(103,96),
|
||
|
});
|
||
|
FillPolygon(p,Color.BRIGHT_CYAN,x_offset,y_offset,new Point[] {
|
||
|
new Point(28,29),
|
||
|
new Point(78,103),
|
||
|
new Point(120,31),
|
||
|
new Point(123,221),
|
||
|
new Point(30,218),
|
||
|
});
|
||
|
//FillRect(p,Color.BRIGHT_RED,200,200,600,64);
|
||
|
final Color testAlpha = new Color(150,0,0,128);
|
||
|
FillCircle(p,testAlpha,150,150,100);
|
||
|
FillOval(p,Color.BRIGHT_GREEN,300,150,100,50);
|
||
|
}
|
||
|
|
||
|
public void FillRect(int[] p,Color col,double x,double y,double w,double h) {
|
||
|
for (int xx=0;xx<w;xx++) {
|
||
|
for (int yy=0;yy<h;yy++) {
|
||
|
int index = ((int)y+yy)*width+(int)x+xx;
|
||
|
p[index]=col.getColor();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void FillCircle(int[] p,Color col,double center_x,double center_y,double r) {
|
||
|
int counter=0;
|
||
|
Point[] points = new Point[CIRCLE_PRECISION];
|
||
|
for (double theta=0;theta<Math.PI*2;theta+=((Math.PI*2)/CIRCLE_PRECISION)) {
|
||
|
//System.out.println("Loop "+counter+++". Theta:"+theta);
|
||
|
//System.out.println("X:"+(Math.sin(theta)*r+center_x)+" Y:"+(Math.cos(theta)*r+center_y));
|
||
|
points[counter++]=new Point((int)(Math.round(Math.sin(theta)*r+center_x)),(int)(Math.round(Math.cos(theta)*r+center_y)));
|
||
|
}
|
||
|
FillPolygon(p,col,0,0,points);
|
||
|
}
|
||
|
|
||
|
|
||
|
public void FillOval(int[] p,Color col,double center_x,double center_y,double w,double h) {
|
||
|
int counter=0;
|
||
|
Point[] points = new Point[CIRCLE_PRECISION];
|
||
|
double r = Math.max(w,h);
|
||
|
double ratio = Math.min(w,h)/r;
|
||
|
for (double theta=0;theta<Math.PI*2;theta+=((Math.PI*2)/CIRCLE_PRECISION)) {
|
||
|
//System.out.println("Loop "+counter+++". Theta:"+theta);
|
||
|
//System.out.println("X:"+(Math.sin(theta)*r+center_x)+" Y:"+(Math.cos(theta)*r+center_y));
|
||
|
Point newP = new Point((int)(Math.round(Math.sin(theta)*r)),(int)(Math.round(Math.cos(theta)*r)));
|
||
|
if (w<h) {
|
||
|
newP.x=(int)Math.round(newP.x*ratio);
|
||
|
} else {
|
||
|
newP.y=(int)Math.round(newP.y*ratio);
|
||
|
}
|
||
|
newP.x+=center_x;
|
||
|
newP.y+=center_y;
|
||
|
points[counter++]=newP;
|
||
|
}
|
||
|
FillPolygon(p,col,0,0,points);
|
||
|
}
|
||
|
|
||
|
public void FillPolygon(int[] p,Color col,double x_offset,double y_offset,Point...points) {
|
||
|
Edge[] edges = new Edge[points.length];
|
||
|
List<Edge> edges_sorted = new ArrayList<Edge>();
|
||
|
for (int i=0;i<points.length;i++) {
|
||
|
edges[i] = new Edge(points[i],points[(i+1)%points.length]);
|
||
|
if (!Double.isInfinite(edges[i].inverse_slope)) {
|
||
|
if (edges_sorted.size()==0) {
|
||
|
edges_sorted.add(edges[i]);
|
||
|
} else {
|
||
|
boolean inserted=false;
|
||
|
for (int j=0;j<edges_sorted.size();j++) {
|
||
|
Edge e2 = edges_sorted.get(j);
|
||
|
if (e2.min_y>=edges[i].min_y) {
|
||
|
edges_sorted.add(j,edges[i]);
|
||
|
inserted=true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!inserted) {
|
||
|
edges_sorted.add(edges[i]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
//System.out.println(edges_sorted);
|
||
|
List<Edge> active_edges = new ArrayList<Edge>();
|
||
|
scanLine = edges_sorted.get(0).min_y-1;
|
||
|
nextScanLine = scanLine+1;
|
||
|
do {
|
||
|
for (int i=0;i<active_edges.size();i+=2) {
|
||
|
Edge e1 = active_edges.get(i);
|
||
|
Edge e2 = active_edges.get(i+1);
|
||
|
//System.out.println("Drawing from "+((int)Math.round(e1.x_of_min_y))+" to "+e2.x_of_min_y+" on line "+scanLine);
|
||
|
for (int x=(int)Math.round(e1.x_of_min_y);x<=e2.x_of_min_y;x++) {
|
||
|
int index = (scanLine+(int)y_offset)*width+x+(int)x_offset;
|
||
|
if (index<p.length&&index>=0) {
|
||
|
Draw(p,index,col.getColor());
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
List<Edge> new_active_edges = new ArrayList<Edge>();
|
||
|
for (int i=0;i<active_edges.size();i++) {
|
||
|
Edge e = active_edges.get(i);
|
||
|
if (e.max_y==scanLine+1) {
|
||
|
active_edges.remove(i--);
|
||
|
} else {
|
||
|
e.x_of_min_y+=e.inverse_slope;
|
||
|
}
|
||
|
}
|
||
|
scanLine++;
|
||
|
for (int i=0;i<active_edges.size();i++) {
|
||
|
Edge e = active_edges.get(i);
|
||
|
boolean inserted=false;
|
||
|
for (int j=0;j<new_active_edges.size();j++) {
|
||
|
Edge e2 = new_active_edges.get(j);
|
||
|
if (e2.x_of_min_y>e.x_of_min_y) {
|
||
|
new_active_edges.add(j,e);
|
||
|
inserted=true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!inserted) {
|
||
|
new_active_edges.add(e);
|
||
|
}
|
||
|
}
|
||
|
active_edges=new_active_edges;
|
||
|
GetNextScanLineEdges(edges_sorted, active_edges);
|
||
|
}
|
||
|
while (active_edges.size()>0);
|
||
|
}
|
||
|
|
||
|
private void GetNextScanLineEdges(List<Edge> edges_sorted, List<Edge> active_edges) {
|
||
|
if (scanLine==nextScanLine) {
|
||
|
for (int i=0;i<edges_sorted.size();i++) {
|
||
|
Edge e = edges_sorted.get(i);
|
||
|
if (e.min_y==scanLine) {
|
||
|
e = edges_sorted.remove(i--);
|
||
|
boolean inserted=false;
|
||
|
for (int j=0;j<active_edges.size();j++) {
|
||
|
if (e.x_of_min_y<active_edges.get(j).x_of_min_y) {
|
||
|
active_edges.add(j,e);
|
||
|
inserted=true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!inserted) {
|
||
|
active_edges.add(e);
|
||
|
}
|
||
|
|
||
|
} else
|
||
|
if (e.min_y>scanLine) {
|
||
|
nextScanLine=e.min_y;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Draw(int[] canvas,int index, int col) {
|
||
|
int alpha = col>>>24;
|
||
|
if (alpha==0) {
|
||
|
return;}
|
||
|
else
|
||
|
if (alpha==255) {
|
||
|
canvas[index]=col;
|
||
|
} else {
|
||
|
float ratio=alpha/255f;
|
||
|
int prev_col=canvas[index];
|
||
|
int prev_r=(prev_col&0xFF);
|
||
|
int prev_g=(prev_col&0xFF00)>>>8;
|
||
|
int prev_b=(prev_col&0xFF0000)>>>16;
|
||
|
int r=(col&0xFF);
|
||
|
int g=(col&0xFF00)>>>8;
|
||
|
int b=(col&0xFF0000)>>>16;
|
||
|
|
||
|
int new_r=(int)(ratio*r+(1-ratio)*prev_r);
|
||
|
int new_g=(int)(ratio*g+(1-ratio)*prev_g);
|
||
|
int new_b=(int)(ratio*b+(1-ratio)*prev_b);
|
||
|
|
||
|
canvas[index]=new_r+(new_g<<8)+(new_b<<16)+(col&0xFF000000);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void run() {
|
||
|
while (true) {
|
||
|
// request a JPanel re-drawing
|
||
|
repaint();
|
||
|
//System.out.println("Repaint "+frameCount++);
|
||
|
//try {Thread.sleep(1);} catch (InterruptedException e) {}
|
||
|
}
|
||
|
}
|
||
|
}
|