-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBrushWorkApp.cpp
275 lines (232 loc) · 8.18 KB
/
BrushWorkApp.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
//
// BrushWorkApp.cpp
// Originally created by the CSci-3081W TAs.
// Edited by Jacob Grafenstein, Stevie Frisbie, and Jonathon Meyer
//
#include "BrushWorkApp.h"
#include "ColorData.h"
#include "PixelBuffer.h"
#include "CalligraphyPen.h"
#include "Highlighter.h"
#include "Eraser.h"
#include "SprayCan.h"
#include "Pen.h"
#include "XPen.h"
#include "Tool.h"
#include <cmath>
#include <iostream>
#include <vector>
using std::cout;
using std::endl;
BrushWorkApp::BrushWorkApp(int argc, char* argv[], int width, int height, ColorData backgroundColor) :
BaseGfxApp(argc, argv, width, height, 50, 50, GLUT_RGB|GLUT_DOUBLE|GLUT_DEPTH, true, width+51, 50) {
m_curTool = 0;//we'll just set this to the first one on initialization
// Set the name of the window
setCaption("BrushWork");
// Initialize an array of Tools to be used to modify the canvas
initializeTools();
// Initialize Interface
initializeBuffers(backgroundColor, width, height);
initGlui();
initGraphics();
// Initialize local variables used for various Tools and to make application to canvas continuous
backColor = backgroundColor;
previousX = -1;
previousY = -1;
}
void BrushWorkApp::display() {
drawPixels(0, 0, m_width, m_height, m_displayBuffer->getData());
}
BrushWorkApp::~BrushWorkApp() {
int i;
int toolSize = (int) tools.size();
if (m_displayBuffer) {
delete m_displayBuffer;
}
for (i=0;i<toolSize;i++) {
delete tools[i];
std::cout << "Deleted all the tools" << std::endl;
}
tools.clear();
}
// This function first applies the current tool to the canvas, then connects the points to form a continuous line
void BrushWorkApp::mouseDragged(int x, int y)
{
float slope;
// XY is used to determine which slope is used, 0 is y/x, 1 is x/y
int xy;
(*tools[m_curTool]).paintMask(x,y,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
if ((previousX == -1) || (previousY == -1)) {
// Do Nothing
} else if ((previousX != -1) && (previousY != -1)) {
if ((previousX-x) == 0) {
xy = 1;
slope = 0;
}
else {
slope = (float)(-1.0*(previousY-y)/(previousX-x));
xy = 0;
if (slope > 1 || slope < -1)
{
slope = 1.0/slope;
xy = 1;
}
}
fillLine(slope,previousX,previousY,x,y,xy);
previousX = x;
previousY = y;
}
}
void BrushWorkApp::mouseMoved(int x, int y) {
}
void BrushWorkApp::leftMouseDown(int x, int y) {
// If the leftMouseDown is clicked without moving, the tool should be applied to the pixelBuffer once
(*tools[m_curTool]).paintMask(x,y,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
// Set the previous x and y values to fill the line
previousX = x;
previousY = y;
}
void BrushWorkApp::leftMouseUp(int x, int y) {
// If the mouse is lifted up, the previous x and y coordinates should not be remembered.
previousX = -1;
previousY = -1;
}
// The fillLine function connects two points by applying the mask to the pixelBuffer for all points in between
void BrushWorkApp::fillLine(float slope, int previousX, int previousY, int x, int y,int xy) {
int i,nextCoord,stepSize;
stepSize = (int) ( ((float) (*tools[m_curTool]).getMaskSize()) * 2.0/7.0);
// Use the y/x slope
if (xy==0) {
// Moving left on the canvas
for (i = previousX; i > x; i-=stepSize) {
nextCoord = getNextYValue(slope, previousX, i, previousY);
(*tools[m_curTool]).paintMask(i,nextCoord,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
}
// Moving right on the canvas
for (i = previousX; i < x; i+=stepSize) {
nextCoord = getNextYValue(slope, previousX, i, previousY);
(*tools[m_curTool]).paintMask(i,nextCoord,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
}
}
else if (xy==1) {
// Moving left on the canvas
for (i = previousY; i > y; i-=stepSize) {
nextCoord = getNextYValue(slope, previousY, i, previousX);
(*tools[m_curTool]).paintMask(nextCoord,i,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
}
// Moving right on the canvas
for (i = previousY; i < y; i+=stepSize) {
nextCoord = getNextYValue(slope, previousY, i, previousX);
(*tools[m_curTool]).paintMask(nextCoord,i,&m_displayBuffer,ColorData(m_curColorRed,m_curColorGreen,m_curColorBlue),backColor);
}
}
}
// Finds the next Y value given the slope of the line, the previous x and y values, and the new x value
int BrushWorkApp::getNextYValue(float slope, int previousX, int newX, int previousY) {
return (int)(-1.0*((slope*newX)-(slope*previousX)-previousY));
}
void BrushWorkApp::initializeBuffers(ColorData backgroundColor, int width, int height) {
m_displayBuffer = new PixelBuffer(width, height, backgroundColor);
}
// Uses a vector to store the Tools
void BrushWorkApp::initializeTools() {
tools.push_back(new Pen());
tools.push_back(new Eraser());
tools.push_back(new SprayCan());
tools.push_back(new CalligraphyPen());
tools.push_back(new Highlighter());
tools.push_back(new XPen());
}
void BrushWorkApp::initGlui() {
// Select first tool (this activates the first radio button in glui)
m_curTool = 0;
GLUI_Panel *toolPanel = new GLUI_Panel(m_glui, "Tool Type");
GLUI_RadioGroup *radio = new GLUI_RadioGroup(toolPanel, &m_curTool, UI_TOOLTYPE, s_gluicallback);
// Create interface buttons for different tools:
new GLUI_RadioButton(radio, "Pen");
new GLUI_RadioButton(radio, "Eraser");
new GLUI_RadioButton(radio, "Spray Can");
new GLUI_RadioButton(radio, "Caligraphy Pen");
new GLUI_RadioButton(radio, "Highlighter");
new GLUI_RadioButton(radio, "XPen");
GLUI_Panel *colPanel = new GLUI_Panel(m_glui, "Tool Color");
m_curColorRed = 0;
m_spinnerR = new GLUI_Spinner(colPanel, "Red:", &m_curColorRed, UI_COLOR_R, s_gluicallback);
m_spinnerR->set_float_limits(0, 1.0);
m_curColorGreen = 0;
m_spinnerG = new GLUI_Spinner(colPanel, "Green:", &m_curColorGreen, UI_COLOR_G, s_gluicallback);
m_spinnerG->set_float_limits(0, 1.0);
m_curColorBlue = 0;
m_spinnerB = new GLUI_Spinner(colPanel, "Blue:", &m_curColorBlue, UI_COLOR_B, s_gluicallback);
m_spinnerB->set_float_limits(0, 1.0);
new GLUI_Button(colPanel, "Red", UI_PRESET_RED, s_gluicallback);
new GLUI_Button(colPanel, "Orange", UI_PRESET_ORANGE, s_gluicallback);
new GLUI_Button(colPanel, "Yellow", UI_PRESET_YELLOW, s_gluicallback);
new GLUI_Button(colPanel, "Green", UI_PRESET_GREEN, s_gluicallback);
new GLUI_Button(colPanel, "Blue", UI_PRESET_BLUE, s_gluicallback);
new GLUI_Button(colPanel, "Purple", UI_PRESET_PURPLE, s_gluicallback);
new GLUI_Button(colPanel, "White", UI_PRESET_WHITE, s_gluicallback);
new GLUI_Button(colPanel, "Black", UI_PRESET_BLACK, s_gluicallback);
new GLUI_Button(m_glui, "Quit", UI_QUIT, (GLUI_Update_CB)exit);
}
void BrushWorkApp::initGraphics() {
// Initialize OpenGL for 2D graphics as used in the BrushWork app
glClearColor(0.0f, 0.0f, 0.0f, 1.0f );
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
gluOrtho2D(0, m_width, 0, m_height);
glViewport(0, 0, m_width, m_height);
}
void BrushWorkApp::gluiControl(int controlID) {
switch (controlID) {
case UI_PRESET_RED:
m_curColorRed = 1;
m_curColorGreen = 0;
m_curColorBlue = 0;
break;
case UI_PRESET_ORANGE:
m_curColorRed = 1;
m_curColorGreen = 0.5;
m_curColorBlue = 0;
break;
case UI_PRESET_YELLOW:
m_curColorRed = 1;
m_curColorGreen = 1;
m_curColorBlue = 0;
break;
case UI_PRESET_GREEN:
m_curColorRed = 0;
m_curColorGreen = 1;
m_curColorBlue = 0;
break;
case UI_PRESET_BLUE:
m_curColorRed = 0;
m_curColorGreen = 0;
m_curColorBlue = 1;
break;
case UI_PRESET_PURPLE:
m_curColorRed = 0.5;
m_curColorGreen = 0;
m_curColorBlue = 1;
break;
case UI_PRESET_WHITE:
m_curColorRed = 1;
m_curColorGreen = 1;
m_curColorBlue = 1;
break;
case UI_PRESET_BLACK:
m_curColorRed = 0;
m_curColorGreen = 0;
m_curColorBlue = 0;
break;
default:
break;
}
m_spinnerB->set_float_val(m_curColorBlue);
m_spinnerG->set_float_val(m_curColorGreen);
m_spinnerR->set_float_val(m_curColorRed);
}