Skip to content

Commit d6722c7

Browse files
committed
Enable full window display, window resizing, add debug HTML, code refactoring for clarity
1 parent ac6a950 commit d6722c7

4 files changed

+329
-49
lines changed

.gitignore

-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,3 @@
22
*.js
33
*.wasm
44
*.vscode
5-
*_debug.html

hello_triangle.cpp

+93-45
Original file line numberDiff line numberDiff line change
@@ -6,48 +6,51 @@
66
//
77
// Build:
88
// emcc hello_triangle.cpp -s USE_SDL=2 -s FULL_ES2=1 -o hello_triangle.js
9+
// emrun hello_triangle.html
910
//
10-
// Run (open in browser):
11-
// index.html
11+
// Debug (provides stdout console):
12+
// emcc hello_triangle.cpp -s USE_SDL=2 -s FULL_ES2=1 -o hello_triangle_debug.js
13+
// emrun hello_triangle_debug.html
1214
//
1315
// Result:
14-
// A colorful triangle. Left mouse pans, wheel zooms in/out.
16+
// A colorful triangle. Left mouse pans, mouse wheel zooms in/out. Window is resizable.
1517
//
1618
#include <exception>
1719
#include <algorithm>
1820

21+
#define GL_GLEXT_PROTOTYPES 1
22+
1923
#ifdef __EMSCRIPTEN__
2024
#include <emscripten.h>
21-
#include <emscripten/html5.h>
2225
#include <SDL.h>
23-
#define GL_GLEXT_PROTOTYPES 1
2426
#include <SDL_opengles2.h>
2527
#else
2628
#include <SDL2/SDL.h>
27-
#define GL_GLEXT_PROTOTYPES 1
2829
#include <SDL2/SDL_opengles2.h>
2930
#endif
3031

31-
SDL_Window* wnd = nullptr;
32+
SDL_Window* window = nullptr;
3233
Uint32 windowID = 0;
33-
int wndWidth = 640, wndHeight = 480;
34+
int windowWidth = 640, windowHeight = 480;
3435
bool mouseDown = false;
3536

36-
// Uniforms - frame invariant shader vars
37-
GLint uniformPan, uniformZoom;
38-
GLfloat pan[2] = {0.0f, 0.0f}, zoom = 1.0f;
37+
// Shader vars
38+
GLint shaderPan, shaderZoom, shaderAspect;
39+
GLfloat pan[2] = {0.0f, 0.0f}, zoom = 1.0f, aspect = 1.0f;
3940

4041
// Vertex shader
4142
const GLchar* vertexSource =
4243
"uniform vec2 pan; \n"
4344
"uniform float zoom; \n"
45+
"uniform float aspect; \n"
4446
"attribute vec4 position; \n"
4547
"varying vec3 color; \n"
4648
"void main() \n"
4749
"{ \n"
4850
" gl_Position = vec4(position.xyz, 1.0); \n"
4951
" gl_Position.xy += pan; \n"
5052
" gl_Position.xy *= zoom; \n"
53+
" gl_Position.y *= aspect; \n"
5154
" color = gl_Position.xyz + vec3(0.5); \n"
5255
"} \n";
5356

@@ -65,23 +68,70 @@ float clamp (float val, float lo, float hi)
6568
return std::max(lo, std::min(val, hi));
6669
}
6770

71+
void updateShader()
72+
{
73+
glUniform2fv(shaderPan, 1, pan);
74+
glUniform1f(shaderZoom, zoom);
75+
glUniform1f(shaderAspect, aspect);
76+
}
77+
78+
void resizeEvent(int width, int height)
79+
{
80+
windowWidth = width;
81+
windowHeight = height;
82+
83+
// Update viewport and aspect ratio
84+
glViewport(0, 0, windowWidth, windowHeight);
85+
aspect = windowWidth / (float)windowHeight;
86+
updateShader();
87+
}
88+
89+
void panEvent(int x, int y)
90+
{
91+
// Make display follow cursor by normalizing cursor to range -2,2, scaled by inverse zoom
92+
pan[0] = ((x / (float) windowWidth) - 0.5f) * 2.0f / zoom;
93+
pan[1] = ((1.0f - (y / (float) windowHeight)) - 0.5f) * 2.0f / zoom / aspect;
94+
updateShader();
95+
}
96+
97+
void zoomEvent(bool wheelDown)
98+
{
99+
// Zoom by scaling up/down in 0.1 increments
100+
zoom += (wheelDown ? -0.1f : 0.1f);
101+
zoom = clamp(zoom, 0.1f, 10.0f);
102+
updateShader();
103+
}
104+
68105
void handleEvents()
69106
{
70107
// Handle events
71108
SDL_Event event;
72109
while (SDL_PollEvent(&event))
73110
{
111+
// Debugging
112+
printf ("pan=%f,%f zoom=%f aspect=%f window=%dx%d\n", pan[0], pan[1], zoom, aspect, windowWidth, windowHeight);
113+
74114
switch (event.type)
75115
{
116+
case SDL_QUIT:
117+
std::terminate();
118+
break;
119+
120+
case SDL_WINDOWEVENT:
121+
{
122+
if (event.window.windowID == windowID
123+
&& event.window.event == SDL_WINDOWEVENT_SIZE_CHANGED)
124+
{
125+
resizeEvent(event.window.data1, event.window.data2);
126+
}
127+
break;
128+
}
129+
76130
case SDL_MOUSEMOTION:
77131
{
78132
SDL_MouseMotionEvent *m = (SDL_MouseMotionEvent*)&event;
79133
if (mouseDown)
80-
{
81-
// Normalize cursor position to -2.0 to 2.0 in x and y
82-
pan[0] = ((m->x / (float) wndWidth) - 0.5f) * 2.0f / zoom;
83-
pan[1] = ((1.0f - (m->y / (float) wndHeight)) - 0.5f) * 2.0f / zoom;
84-
}
134+
panEvent(m->x, m->y);
85135
break;
86136
}
87137

@@ -92,7 +142,7 @@ void handleEvents()
92142
{
93143
mouseDown = true;
94144

95-
// Push a motion event to update display at current mouse
145+
// Push a motion event to update display at current mouse position
96146
SDL_Event push_event;
97147
push_event.type = SDL_MOUSEMOTION;
98148
push_event.motion.x = m->x;
@@ -113,8 +163,8 @@ void handleEvents()
113163
case SDL_MOUSEWHEEL:
114164
{
115165
SDL_MouseWheelEvent *m = (SDL_MouseWheelEvent*)&event;
116-
zoom += ((m->y < 0) ? -0.1f : 0.1f);
117-
zoom = clamp(zoom, 0.0f, 10.0f);
166+
bool wheelDown = m->y < 0;
167+
zoomEvent(wheelDown);
118168
break;
119169
}
120170
}
@@ -123,67 +173,64 @@ void handleEvents()
123173

124174
void redraw()
125175
{
126-
// Update uniforms
127-
glUniform2fv(uniformPan, 1, pan);
128-
glUniform1f(uniformZoom, zoom);
129-
130-
// Clear the screen to black
131-
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
176+
// Clear screen
132177
glClear(GL_COLOR_BUFFER_BIT);
133178

134179
// Draw the vertex buffer
135180
glDrawArrays(GL_TRIANGLES, 0, 3);
136181

137182
// Swap front/back framebuffers
138-
SDL_GL_SwapWindow(wnd);
183+
SDL_GL_SwapWindow(window);
139184
}
140185

141-
void main_loop()
186+
void mainLoop()
142187
{
143188
handleEvents();
144189
redraw();
145190
}
146191

147192
int main(int argc, char** argv)
148193
{
149-
#ifdef __EMSCRIPTEN__
150-
emscripten_get_canvas_element_size("#canvas", &wndWidth, &wndHeight);
151-
#endif
152-
153-
// Create SDL2 window with GL context
154-
wnd = SDL_CreateWindow("hello_triangle", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,
155-
wndWidth, wndHeight, SDL_WINDOW_OPENGL | SDL_WINDOW_SHOWN);
156-
windowID = SDL_GetWindowID(wnd);
194+
// Create SDL window
195+
window = SDL_CreateWindow("hello_triangle", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,
196+
windowWidth, windowHeight, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE| SDL_WINDOW_SHOWN);
197+
windowID = SDL_GetWindowID(window);
157198

199+
// Create OpenGLES 2 context on window
158200
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 2);
159201
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 0);
160202
SDL_GL_SetSwapInterval(1);
161203
SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);
162204
SDL_GL_SetAttribute(SDL_GL_DEPTH_SIZE, 24);
163-
SDL_GLContext glc = SDL_GL_CreateContext(wnd);
205+
SDL_GLContext glc = SDL_GL_CreateContext(window);
206+
207+
// Set clear color to black
208+
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
164209

165-
// Create and compile the vertex shader
210+
// Create and compile vertex shader
166211
GLuint vertexShader = glCreateShader(GL_VERTEX_SHADER);
167212
glShaderSource(vertexShader, 1, &vertexSource, NULL);
168213
glCompileShader(vertexShader);
169214

170-
// Create and compile the fragment shader
215+
// Create and compile fragment shader
171216
GLuint fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
172217
glShaderSource(fragmentShader, 1, &fragmentSource, NULL);
173218
glCompileShader(fragmentShader);
174219

175-
// Link the vertex and fragment shader into a shader program and use it
220+
// Link vertex and fragment shader into shader program and use it
176221
GLuint shaderProgram = glCreateProgram();
177222
glAttachShader(shaderProgram, vertexShader);
178223
glAttachShader(shaderProgram, fragmentShader);
179224
glLinkProgram(shaderProgram);
180225
glUseProgram(shaderProgram);
181226

182-
// Get uniform locations for updating later
183-
uniformPan = glGetUniformLocation(shaderProgram, "pan");
184-
uniformZoom = glGetUniformLocation(shaderProgram, "zoom");
227+
// Get shader variables and initalize them
228+
shaderPan = glGetUniformLocation(shaderProgram, "pan");
229+
shaderZoom = glGetUniformLocation(shaderProgram, "zoom");
230+
shaderAspect = glGetUniformLocation(shaderProgram, "aspect");
231+
updateShader();
185232

186-
// Create a vertex buffer object and copy the vertex data to it
233+
// Create vertex buffer object and copy vertex data into it
187234
GLuint vbo;
188235
glGenBuffers(1, &vbo);
189236
glBindBuffer(GL_ARRAY_BUFFER, vbo);
@@ -200,11 +247,12 @@ int main(int argc, char** argv)
200247
glEnableVertexAttribArray(posAttrib);
201248
glVertexAttribPointer(posAttrib, 3, GL_FLOAT, GL_FALSE, 0, 0);
202249

250+
// Start the main loop
203251
#ifdef __EMSCRIPTEN__
204-
emscripten_set_main_loop(main_loop, 0, true);
252+
emscripten_set_main_loop(mainLoop, 0, true);
205253
#else
206254
while(true)
207-
main_loop();
255+
mainLoop();
208256
#endif
209257

210258
return 0;

hello_triangle.html

+16-3
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,25 @@
11
<!DOCTYPE html>
22
<html lang="en">
33
<head>
4-
<meta charset="utf-8">
4+
<meta charset="utf-8">
55
<meta http-equiv="X-UA-Compatible" content="chrome=1, IE=edge">
66
<title>Hello Triangle</title>
7+
<style>
8+
.fullwindow {
9+
position: absolute;
10+
top: 0px;
11+
left: 0px;
12+
margin: 0px;
13+
border: 0;
14+
width: 100%;
15+
height: 100%;
16+
overflow: hidden;
17+
display: block;
18+
}
19+
</style>
720
</head>
8-
<body style="display: block; margin-left: auto; margin-right: auto; width: 50%; background-color: #FFFFFF;">
9-
<canvas id="canvas" width="800" height="600" />
21+
<body style="background-color: #FFFFFF;">
22+
<canvas class = "fullwindow" id="canvas" oncontextmenu="event.preventDefault()"/>
1023
<script type="text/javascript">
1124
var Module = {};
1225
Module.canvas = document.getElementById('canvas');

0 commit comments

Comments
 (0)