GK/QT/scan-line/mywindow.cpp

340 lines
9.8 KiB
C++
Raw Permalink Normal View History

2022-12-17 15:45:10 +01:00
// Dolaczamy plik naglowkowy naszej klasy MyWindow
#include "mywindow.h"
// Dolaczamy plik naglowkowy zawierajacy definicje GUI
// Plik ten jest generowany automatycznie
// z pliku XML "mywindow.ui"
#include "ui_mywindow.h"
#include <cmath>
#include <QColor>
#include <QDebug>
#include <QColorDialog>
#include <iostream>
#include <float.h> // for float,double macros
#include <string.h>
#define PI 3.1415
bool Intersects(Line a, Line b, Point& p){
return false;
}
int Shape::FindPoint(int x, int y, float radius){
Point src_point(x, y);
float min_dist = FLT_MAX;
int min_id = -1;
for(int i = 0; i < (int)points.size(); i++){
float dist = (src_point - (Point)(points[i])).length();
if(dist < min_dist && dist < radius){
min_id = i;
min_dist = dist;
}
}
return min_id;
}
2022-12-17 15:45:10 +01:00
// Definicja konstruktora, wywolujemy najpierw
// konstruktor klasy nadrzednej, nastepnie tworzymy
// obiekt klasy Ui_MyWindow reprezentujacy GUI
MyWindow::MyWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MyWindow)
{
// Wywolujemy funkcje tworzaca elementy GUI
// Jej definicja znajduje sie w pliku "ui_mywindow.h"
ui->setupUi(this);
// Pobieramy wymiary i wspolrzedne lewego gornego naroznika ramki
// i ustawiamy wartosci odpowiednich pol
// Uwaga: ramke "rysujFrame" wykorzystujemy tylko do
// wygodnego ustaiwenia tych wymiarow. Rysunek bedziemy wyswietlac
// bezposrednio w glownym oknie aplikacji.
szer = ui->rysujFrame->width();
wys = ui->rysujFrame->height();
poczX = ui->rysujFrame->x();
poczY = ui->rysujFrame->y();
shapes.push_back(Shape());
2022-12-17 15:45:10 +01:00
// Tworzymy obiekt klasy QImage, o odpowiedniej szerokosci
// i wysokosci. Ustawiamy format bitmapy na 32 bitowe RGB
// (0xffRRGGBB).
img = new QImage(szer,wys,QImage::Format_RGB32);
img_tmp = new QImage(szer,wys,QImage::Format_RGB32);
active_img = img;
}
// Definicja destruktora
MyWindow::~MyWindow()
{
delete ui;
}
// Funkcja (slot) wywolywana po nacisnieciu przycisku "Wyjscie" (exitButton)
// Uwaga: polaczenie tej funkcji z sygnalem "clicked"
// emitowanym przez przycisk jest realizowane
// za pomoca funkcji QMetaObject::connectSlotsByName(MyWindow)
// znajdujacej sie w automatycznie generowanym pliku "ui_mywindow.h"
// Nie musimy wiec sami wywolywac funkcji "connect"
void MyWindow::on_exitButton_clicked()
{
// qApp to globalny wskaznik do obiektu reprezentujacego aplikacje
// quit() to funkcja (slot) powodujaca zakonczenie aplikacji z kodem 0 (brak bledu)
qApp->quit();
}
// Funkcja "odmalowujaca" komponent
void MyWindow::paintEvent(QPaintEvent*)
{
// Obiekt klasy QPainter pozwala nam rysowac na komponentach
QPainter p(this);
// Rysuje obrazek "img" w punkcie (poczX,poczY)
// (tu bedzie lewy gorny naroznik)
p.drawImage(poczX,poczY,*active_img);
}
// Funkcja (slot) wywolywana po nacisnieciu przycisku "Czysc" (cleanButton)
void MyWindow::on_cleanButton_clicked()
{
// Funkcja czysci (zamalowuje na bialo) obszar rysowania
// definicja znajduje sie ponizej
czysc();
// Funkcja "update()" powoduje ponowne "namalowanie" calego komponentu
// Wywoluje funkcje "paintEvent"
update();
}
// Funkcja powoduje wyczyszczenie (zamalowanie na bialo)
// obszaru rysowania
void MyWindow::czysc()
{
shapes.clear();
2022-12-18 19:10:44 +01:00
shapes.push_back(Shape());
DrawShapes();
2022-12-17 15:45:10 +01:00
}
// Funkcja (slot) wywolywana po nacisnieciu przycisku myszy (w glownym oknie)
void MyWindow::mousePressEvent(QMouseEvent *event)
{
// Pobieramy wspolrzedne punktu klikniecia
int x = event->x();
int y = event->y();
// Sa to wspolrzedne wzgledem glownego okna,
// Musimy odjac od nich wpolrzedne lewego gornego naroznika rysunku
x -= poczX;
y -= poczY;
if(x >= szer || y >= wys || x < 0 || y < 0){
return;
}
int id = -1;
int pt_id, sh_id;
bool res;
2022-12-17 15:45:10 +01:00
switch(event->button()){
case Qt::LeftButton:
res = FindPoint(x, y, pt_id, sh_id, 20);
if(!res){
2022-12-17 15:45:10 +01:00
mode = Add;
shapes[shapes.size()-1].points.push_back(Point(x, y));
2022-12-17 15:45:10 +01:00
}else{
if(pt_id == 0 && shapes[shapes.size()-1].points.size() >= 3){
mode = Add;
shapes.push_back(Shape());
}else{
mode = Move;
tmp_point_id = pt_id;
tmp_shape_id = sh_id;
shapes[sh_id].points[pt_id].x = x;
shapes[sh_id].points[pt_id].y = y;
}
2022-12-17 15:45:10 +01:00
}
break;
case Qt::RightButton:
mode = Remove;
res = FindPoint(x, y, pt_id, sh_id, 20);
if(res){
// bezier_points.erase(bezier_points.begin() + id);
2022-12-17 15:45:10 +01:00
}
break;
default:
break;
}
DrawShapes();
2022-12-17 15:45:10 +01:00
update();
}
void MyWindow::mouseMoveEvent(QMouseEvent *event)
{
// Pobieramy wspolrzedne punktu klikniecia
int x = event->x();
int y = event->y();
// Sa to wspolrzedne wzgledem glownego okna,
// Musimy odjac od nich wpolrzedne lewego gornego naroznika rysunku
x -= poczX;
y -= poczY;
if(x >= szer || y >= wys || x < 0 || y < 0){
return;
}
switch(mode){
case Move:
// bezier_points[tmp_point_id].x = x;
// bezier_points[tmp_point_id].y = y;
shapes[tmp_shape_id].points[tmp_point_id].x = x;
shapes[tmp_shape_id].points[tmp_point_id].y = y;
2022-12-17 15:45:10 +01:00
break;
default:
break;
}
DrawShapes();
2022-12-17 15:45:10 +01:00
// Odswiezamy komponent
update();
}
bool MyWindow::FindPoint(int x, int y, int& pt_id, int& shape_id, float radius){
for(int i = 0; i < (int)shapes.size(); i++){
int id;
if((id = shapes[i].FindPoint(x, y, radius)) != -1){
shape_id = i;
pt_id = id;
return true;
2022-12-17 15:45:10 +01:00
}
}
return false;
2022-12-17 15:45:10 +01:00
}
void MyWindow::DrawPixel(QImage* img, int x, int y, QColor color){
if(x >= szer || y >= wys || x < 0 || y < 0)
return;
unsigned char* ptr = img->bits();
ptr[szer*4*y + 4*x] = color.blue();
ptr[szer*4*y + 4*x + 1] = color.green();
ptr[szer*4*y + 4*x + 2] = color.red();
ptr[szer*4*y + 4*x + 3] = color.alpha();
}
void MyWindow::DrawLine(int x1, int y1, int x2, int y2, QImage *img){
if(x1 > x2){
std::swap(x1, x2);
std::swap(y1, y2);
}
float diff = x2 - x1;
float a = diff != 0 ? (y2 - y1) / diff : FLT_MAX;
QColor color(255, 255, 255, 255);
2023-04-14 22:15:14 +02:00
if(abs(a) < 1.0f){
2022-12-17 15:45:10 +01:00
for(int x = x1; x <= x2; x++){
int x_form = x - x1;
int y = a * x_form + y1;
DrawPixel(img, x, y, color);
}
}
else{
if(y1 > y2){
std::swap(x1, x2);
std::swap(y1, y2);
}
float diff = x2 - x1;
float a = diff != 0 ? (y2 - y1) / diff : FLT_MAX;
for(int y = y1; y <= y2; y++){
int y_form = y - y1;
int x = ((float)(y_form) / a) + x1;
DrawPixel(img, x, y, color);
}
}
}
void MyWindow::DrawSquare(int x, int y, int size, QColor color){
size /= 2;
for(int x1 = x - size; x1 < x + size; x1++){
for(int y1 = y - size; y1 < y + size; y1++){
if(x1 >= szer || y1 >= wys || x1 < 0 || y1 < 0)
continue;
DrawPixel(img, x1, y1, color);
}
}
}
void MyWindow::DrawShapes(){
2022-12-17 15:45:10 +01:00
ClearImage(img);
for(int i = 0; i < (int)shapes.size(); i++){
for(int j = 0; j < (int)shapes[i].points.size() - 1; j++){
Point p1 = shapes[i].points[j];
Point p2 = shapes[i].points[j+1];
DrawLine(p1.x, p1.y, p2.x, p2.y, img);
}
if(i != (int)shapes.size() - 1 && (int)shapes[i].points.size() >= 2){
Point p1 = shapes[i].points[(int)shapes[i].points.size()-1];
Point p2 = shapes[i].points[0];
DrawLine(p1.x, p1.y, p2.x, p2.y, img);
2022-12-17 15:45:10 +01:00
}
2022-12-18 19:10:44 +01:00
std::vector<Line> lines;
int points_n = (int)shapes[i].points.size();
for(int j = 0; j < points_n; j++){
2022-12-18 19:10:44 +01:00
Point p1 = shapes[i].points[j];
Point p2 = j == points_n - 1 ? shapes[i].points[0] : shapes[i].points[j+1];
2022-12-18 19:10:44 +01:00
lines.push_back(Line(p1, p2));
}
for(int y = 0; y < wys; y++){
std::vector<int> xs = GetIntersectionsWithY(y, lines);
for(int l = 1; l < xs.size(); l+=2){
DrawLine(xs[l-1], y, xs[l], y, img);
}
}
}
}
std::vector<int> MyWindow::GetIntersectionsWithY(int y, std::vector<Line> lines){
std::vector<int> res;
for(int i = 0; i < lines.size(); i++){
Line l = lines[i];
if(l.A.x < l.B.x)
std::swap(l.A, l.B);
if((l.A.y <= y && l.B.y > y) || (l.A.y > y && l.B.y <= y)){
2022-12-18 19:10:44 +01:00
float d1 = (y - l.A.y);
float d2 = (l.B.y - l.A.y);
float dx = (l.B.x - l.A.x);
float ratio = d2/d1;
int x = l.A.x + dx / ratio;
res.push_back(x);
}
2022-12-17 15:45:10 +01:00
}
std::sort(res.begin(), res.end(), std::less<int>());
2022-12-18 19:10:44 +01:00
return res;
2022-12-17 15:45:10 +01:00
}
void MyWindow::ClearImage(QImage *img){
unsigned char* empty_val = (unsigned char*)malloc(4);
empty_val[0] = 0;
empty_val[1] = 0;
empty_val[2] = 0;
empty_val[3] = 255;
unsigned char* ptr = img->bits();
for(int i = 0; i < img->width(); i++){
for(int j = 0; j < img->height(); j++){
memcpy(ptr + 4 * (i + j * img->width()), empty_val, 4);
}
}
}