6
6
//
7
7
// Build:
8
8
// emcc hello_triangle.cpp -s USE_SDL=2 -s FULL_ES2=1 -o hello_triangle.js
9
+ // emrun hello_triangle.html
9
10
//
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
12
14
//
13
15
// 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 .
15
17
//
16
18
#include < exception>
17
19
#include < algorithm>
18
20
21
+ #define GL_GLEXT_PROTOTYPES 1
22
+
19
23
#ifdef __EMSCRIPTEN__
20
24
#include < emscripten.h>
21
- #include < emscripten/html5.h>
22
25
#include < SDL.h>
23
- #define GL_GLEXT_PROTOTYPES 1
24
26
#include < SDL_opengles2.h>
25
27
#else
26
28
#include < SDL2/SDL.h>
27
- #define GL_GLEXT_PROTOTYPES 1
28
29
#include < SDL2/SDL_opengles2.h>
29
30
#endif
30
31
31
- SDL_Window* wnd = nullptr ;
32
+ SDL_Window* window = nullptr ;
32
33
Uint32 windowID = 0 ;
33
- int wndWidth = 640 , wndHeight = 480 ;
34
+ int windowWidth = 640 , windowHeight = 480 ;
34
35
bool mouseDown = false ;
35
36
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 ;
39
40
40
41
// Vertex shader
41
42
const GLchar* vertexSource =
42
43
" uniform vec2 pan; \n "
43
44
" uniform float zoom; \n "
45
+ " uniform float aspect; \n "
44
46
" attribute vec4 position; \n "
45
47
" varying vec3 color; \n "
46
48
" void main() \n "
47
49
" { \n "
48
50
" gl_Position = vec4(position.xyz, 1.0); \n "
49
51
" gl_Position.xy += pan; \n "
50
52
" gl_Position.xy *= zoom; \n "
53
+ " gl_Position.y *= aspect; \n "
51
54
" color = gl_Position.xyz + vec3(0.5); \n "
52
55
" } \n " ;
53
56
@@ -65,23 +68,70 @@ float clamp (float val, float lo, float hi)
65
68
return std::max (lo, std::min (val, hi));
66
69
}
67
70
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
+
68
105
void handleEvents ()
69
106
{
70
107
// Handle events
71
108
SDL_Event event;
72
109
while (SDL_PollEvent (&event))
73
110
{
111
+ // Debugging
112
+ printf (" pan=%f,%f zoom=%f aspect=%f window=%dx%d\n " , pan[0 ], pan[1 ], zoom, aspect, windowWidth, windowHeight);
113
+
74
114
switch (event.type )
75
115
{
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
+
76
130
case SDL_MOUSEMOTION:
77
131
{
78
132
SDL_MouseMotionEvent *m = (SDL_MouseMotionEvent*)&event;
79
133
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 );
85
135
break ;
86
136
}
87
137
@@ -92,7 +142,7 @@ void handleEvents()
92
142
{
93
143
mouseDown = true ;
94
144
95
- // Push a motion event to update display at current mouse
145
+ // Push a motion event to update display at current mouse position
96
146
SDL_Event push_event;
97
147
push_event.type = SDL_MOUSEMOTION;
98
148
push_event.motion .x = m->x ;
@@ -113,8 +163,8 @@ void handleEvents()
113
163
case SDL_MOUSEWHEEL:
114
164
{
115
165
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 );
118
168
break ;
119
169
}
120
170
}
@@ -123,67 +173,64 @@ void handleEvents()
123
173
124
174
void redraw ()
125
175
{
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
132
177
glClear (GL_COLOR_BUFFER_BIT);
133
178
134
179
// Draw the vertex buffer
135
180
glDrawArrays (GL_TRIANGLES, 0 , 3 );
136
181
137
182
// Swap front/back framebuffers
138
- SDL_GL_SwapWindow (wnd );
183
+ SDL_GL_SwapWindow (window );
139
184
}
140
185
141
- void main_loop ()
186
+ void mainLoop ()
142
187
{
143
188
handleEvents ();
144
189
redraw ();
145
190
}
146
191
147
192
int main (int argc, char ** argv)
148
193
{
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);
157
198
199
+ // Create OpenGLES 2 context on window
158
200
SDL_GL_SetAttribute (SDL_GL_CONTEXT_MAJOR_VERSION, 2 );
159
201
SDL_GL_SetAttribute (SDL_GL_CONTEXT_MINOR_VERSION, 0 );
160
202
SDL_GL_SetSwapInterval (1 );
161
203
SDL_GL_SetAttribute (SDL_GL_DOUBLEBUFFER, 1 );
162
204
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 );
164
209
165
- // Create and compile the vertex shader
210
+ // Create and compile vertex shader
166
211
GLuint vertexShader = glCreateShader (GL_VERTEX_SHADER);
167
212
glShaderSource (vertexShader, 1 , &vertexSource, NULL );
168
213
glCompileShader (vertexShader);
169
214
170
- // Create and compile the fragment shader
215
+ // Create and compile fragment shader
171
216
GLuint fragmentShader = glCreateShader (GL_FRAGMENT_SHADER);
172
217
glShaderSource (fragmentShader, 1 , &fragmentSource, NULL );
173
218
glCompileShader (fragmentShader);
174
219
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
176
221
GLuint shaderProgram = glCreateProgram ();
177
222
glAttachShader (shaderProgram, vertexShader);
178
223
glAttachShader (shaderProgram, fragmentShader);
179
224
glLinkProgram (shaderProgram);
180
225
glUseProgram (shaderProgram);
181
226
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 ();
185
232
186
- // Create a vertex buffer object and copy the vertex data to it
233
+ // Create vertex buffer object and copy vertex data into it
187
234
GLuint vbo;
188
235
glGenBuffers (1 , &vbo);
189
236
glBindBuffer (GL_ARRAY_BUFFER, vbo);
@@ -200,11 +247,12 @@ int main(int argc, char** argv)
200
247
glEnableVertexAttribArray (posAttrib);
201
248
glVertexAttribPointer (posAttrib, 3 , GL_FLOAT, GL_FALSE, 0 , 0 );
202
249
250
+ // Start the main loop
203
251
#ifdef __EMSCRIPTEN__
204
- emscripten_set_main_loop (main_loop , 0 , true );
252
+ emscripten_set_main_loop (mainLoop , 0 , true );
205
253
#else
206
254
while (true )
207
- main_loop ();
255
+ mainLoop ();
208
256
#endif
209
257
210
258
return 0 ;
0 commit comments