/*
winnie - an experimental window system
Copyright (C) 2013 Eleni Maria Stea
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
Author: Eleni Maria Stea
*/
#include
#include
#include
#include
#include "geom.h"
#include "gfx.h"
// used by the polygon rasterizer
#define MAX_SCANLINES 2048
static int32_t left_fx[MAX_SCANLINES], right_fx[MAX_SCANLINES];
void clear_screen(int r, int g, int b)
{
Rect screen_rect = get_screen_size();
fill_rect(screen_rect, r, g, b);
}
void fill_rect(const Rect &rect, int r, int g, int b)
{
Rect drect = rect;
Rect screen_rect = get_screen_size();
Rect clipping_rect = get_clipping_rect();
if(drect.x < clipping_rect.x) {
drect.width -= clipping_rect.x - drect.x;
drect.x = clipping_rect.x;
}
if(drect.y < clipping_rect.y) {
drect.height -= clipping_rect.y - drect.y;
drect.y = clipping_rect.y;
}
if(drect.x + drect.width >= clipping_rect.x + clipping_rect.width) {
drect.width = clipping_rect.width + clipping_rect.x - drect.x;
}
if(drect.y + drect.height >= clipping_rect.y + clipping_rect.height) {
drect.height = clipping_rect.height + clipping_rect.y - drect.y;
}
unsigned char *fb = get_framebuffer() + (drect.x + screen_rect.width * drect.y) * 4;
for(int i=0; i= irect.x + irect.width) {
width -= xend - (irect.x + irect.width);
}
int yend = dest_y + height;
if(yend >= irect.y + irect.height) {
height -= yend - (irect.y + irect.height);
}
if(width <= 0 || height <= 0) {
return;
}
unsigned char *sptr = src_img + ((src_rect.y + src_yoffs) * src_rect.width + src_rect.x + src_xoffs) * 4;
unsigned char *dptr = dest_img + (dest_y * dest_rect.width + dest_x) * 4;
for(int i=0; i= irect.x + irect.width) {
width -= xend - (irect.x + irect.width);
}
int yend = dest_y + height;
if(yend >= irect.y + irect.height) {
height -= yend - (irect.y + irect.height);
}
if(width <= 0 || height <= 0) {
return;
}
unsigned char *sptr = src_img + ((src_rect.y + src_yoffs) * src_rect.width + src_rect.x + src_xoffs) * 4;
unsigned char *dptr = dest_img + (dest_y * dest_rect.width + dest_x) * 4;
for(int i=0; ipixels + (y0 * pixmap->width + x0) * 4;
int dx = x1 - x0;
int dy = y1 - y0;
int x_inc = 4;
int y_inc = pixmap->width * 4;
if(dx < 0) {
x_inc = -x_inc;
dx = -dx;
}
if(dy < 0) {
y_inc = -y_inc;
dy = -dy;
}
if(dx > dy) {
int error = (dy << 1) - dx;
for(int i=0; i<=dx; i++) {
ptr[red_offs] = r;
ptr[green_offs] = g;
ptr[blue_offs] = b;
if(error >= 0) {
error -= dx << 1;
ptr += y_inc;
}
error += dy << 1;
ptr += x_inc;
}
} else {
int error = (dx << 1) - dy;
for(int i=0; i<=dy; i++) {
ptr[red_offs] = r;
ptr[green_offs] = g;
ptr[blue_offs] = b;
if(error >= 0) {
error -= dy << 1;
ptr += x_inc;
}
error += dx << 1;
ptr += y_inc;
}
}
}
void draw_polygon(Pixmap *pixmap, int *vpos, int *vtex, int num_verts, int r, int g, int b)
{
int roffs = 0, goffs = 1, boffs = 2;
if(pixmap == get_framebuffer_pixmap()) {
get_rgb_order(&roffs, &goffs, &boffs);
}
int ystart = pixmap->height, yend = 0;
for(int i=0; i= 0) {
for(int j=y0; j<=y1; j++) {
right_fx[j] = fx;
fx += fslope;
}
} else {
for(int j=y0; j>=y1; j--) {
left_fx[j] = fx;
fx += fslope;
}
}
ystart = std::min(ystart, std::min(y0, y1));
yend = std::max(yend, std::max(y0, y1));
}
if(ystart < 0) ystart = 0;
if(yend > pixmap->height) yend = pixmap->height;
for(int i=ystart; i> 8;
int dx = (right_fx[i] - left_fx[i]) >> 8;
// accept polygons of either order
if(dx < 0) {
x = (right_fx[i] + (1 << 7)) >> 8;
dx = -dx;
}
if(x < 0) {
dx += x;
x = 0;
}
if(x + dx >= pixmap->width) {
dx = pixmap->width - x;
}
unsigned char *pixptr = pixmap->pixels + (i * pixmap->width + x) * 4;
for(int j=0; j<=dx; j++) {
pixptr[roffs] = r;
pixptr[goffs] = g;
pixptr[boffs] = b;
pixptr += 4;
}
}
}