CMSC23700 Common Code Library
Support code for CS23700 programming projects
Loading...
Searching...
No Matches
cs237-window.hpp
Go to the documentation of this file.
1
8/*
9 * COPYRIGHT (c) 2023 John Reppy (http://cs.uchicago.edu/~jhr)
10 * All rights reserved.
11 */
12
13#ifndef _CS237_WINDOW_HPP_
14#define _CS237_WINDOW_HPP_
15
16#ifndef _CS237_HPP_
17#error "cs237-window.hpp should not be included directly"
18#endif
19
20#include <optional>
21
22namespace cs237 {
23
25//
27 int wid;
28 int ht;
29 std::string title;
30 bool resizable;
31 bool depth;
32 bool stencil;
33
41 CreateWindowInfo (int w, int h, std::string const &t, bool r, bool d, bool s)
42 : wid(w), ht(h), title(t), resizable(r), depth(d), stencil(s)
43 { }
44
48 CreateWindowInfo (int w, int h)
49 : wid(w), ht(h), title(""), resizable(false), depth(false), stencil(false)
50 { }
51
53 bool needsDepthBuf () const { return this->depth || this->stencil; }
54};
55
57//
58class Window {
59public:
60
62 virtual ~Window ();
63
65 Application *app () { return this->_app; }
66
68 vk::Device device () const { return this->_app->_device; }
69
71 vk::Queue graphicsQ () const { return this->_app->_queues.graphics; }
72
74 vk::Queue presentationQ () const { return this->_app->_queues.present; }
75
78 void refresh ()
79 {
80 if (this->_isVis) {
81 this->draw();
82 }
83 }
84
86 void hide ()
87 {
88 glfwHideWindow (this->_win);
89 this->_isVis = false;
90 }
91
93 void show ()
94 {
95 glfwShowWindow (this->_win);
96 this->_isVis = true;
97 }
98
101 virtual void draw () = 0;
102
110 virtual void reshape (int wid, int ht);
111
113 virtual void iconify (bool iconified);
114
117 {
118 return glfwWindowShouldClose (this->_win);
119 }
120
124 virtual void key (int key, int scancode, int action, int mods);
125 virtual void cursorPos (double xpos, double ypos);
126 virtual void cursorEnter (bool entered);
127 virtual void mouseButton (int button, int action, int mods);
128 virtual void scroll (double xoffset, double yoffset);
129 //}
130
133 void enableKeyEvent (bool enable);
134 void setCursorMode (int mode);
135 void enableCursorPosEvent (bool enable);
136 void enableCursorEnterEvent (bool enable);
137 void enableMouseButtonEvent (bool enable);
138 void enableScrollEvent (bool enable);
139 //}
140
141protected:
144 vk::SurfaceCapabilitiesKHR capabilities;
145 std::vector<vk::SurfaceFormatKHR> formats;
146 std::vector<vk::PresentModeKHR> presentModes;
147
149 vk::SurfaceFormatKHR chooseSurfaceFormat ();
152 vk::PresentModeKHR choosePresentMode ();
154 vk::Extent2D chooseExtent (GLFWwindow *win);
155 };
156
159 bool depth;
160 bool stencil;
161 vk::Format format;
162 vk::Image image;
163 vk::DeviceMemory imageMem;
164 vk::ImageView view;
165 };
166
168 struct SwapChain {
169 vk::Device device;
170 vk::SwapchainKHR chain;
171 vk::Format imageFormat;
172 vk::Extent2D extent;
174 // the following vectors hold the state for each of the buffers in the
175 // swap chain.
176 std::vector<vk::Image> images;
177 std::vector<vk::ImageView> views;
178 std::optional<DepthStencilBuffer> dsBuf;
179 std::vector<vk::Framebuffer> fBufs;
180
181 SwapChain (vk::Device dev)
182 : device(dev), dsBuf(std::nullopt)
183 { }
184
186 int size () const { return this->images.size(); }
187
189 void initFramebuffers (vk::RenderPass renderPass);
190
192 bool hasDepthBuffer () const
193 {
194 return this->dsBuf.has_value() && this->dsBuf->depth;
195 }
196
198 bool hasStencilBuffer () const
199 {
200 return this->dsBuf.has_value() && this->dsBuf->stencil;
201 }
202
204 void cleanup ();
205 };
206
208 struct SyncObjs {
210 vk::Semaphore imageAvailable;
212 vk::Semaphore renderFinished;
214 vk::Fence inFlight;
216
218 explicit SyncObjs (Window *w)
219 : win(w),
220 imageAvailable(nullptr),
221 renderFinished(nullptr),
222 inFlight(nullptr)
223 {
224 this->allocate();
225 }
226 SyncObjs () = delete;
227 SyncObjs (SyncObjs &) = delete;
228 SyncObjs (SyncObjs const &) = delete;
229 SyncObjs (SyncObjs &&) = delete;
230
233
235 void allocate ();
236
239 vk::ResultValue<uint32_t> acquireNextImage ();
240
242 void reset ();
243
247 void submitCommands (vk::Queue q, vk::CommandBuffer const &cmdBuf);
248
253 vk::Result present (vk::Queue q, uint32_t imageIndex);
254 };
255
257 GLFWwindow *_win;
258 int _wid, _ht;
259 bool _isVis;
265 // Vulkan state for rendering
266 vk::SurfaceKHR _surf;
268
273
276
281 void _createSwapChain (bool depth, bool stencil);
282
287
293 std::vector<vk::AttachmentDescription> &descs,
294 std::vector<vk::AttachmentReference> &refs);
295
299 uint32_t _graphicsQIdx () const { return this->_app->_qIdxs.graphics; }
300
304 uint32_t _presentationQIdx () const { return this->_app->_qIdxs.present; }
305
311 vk::Viewport _getViewport (bool oglView = false)
312 {
313 if (oglView) {
314 // to get the OpenGL-style viewport, we set the origin at Y = height
315 // and use a negative height
316 return vk::Viewport(
317 0.0f,
318 float(this->_swap.extent.width),
319 float(this->_swap.extent.width),
320 -float(this->_swap.extent.height),
321 0.0f, /* min depth */
322 1.0f);
323 } else {
324 return vk::Viewport(
325 0.0f, 0.0f,
326 float(this->_swap.extent.width),
327 float(this->_swap.extent.height),
328 0.0f, /* min depth */
329 1.0f); /* max depth */
330 }
331 }
332
335 vk::Rect2D _getScissorsRect ()
336 {
337 return vk::Rect2D(
338 {0, 0},
339 {this->_swap.extent.width, this->_swap.extent.height});
340 }
341
352 void _setViewportCmd (vk::CommandBuffer cmdBuf, bool oglView = false)
353 {
354 if (oglView) {
355 /* NOTE: we negate the height and set the Y origin to ht because Vulkan's
356 * viewport coordinates are from top-left down, instead of from
357 * bottom-left up. See
358 * https://www.saschawillems.de/blog/2019/03/29/flipping-the-vulkan-viewport
359 */
360 this->_setViewportCmd(cmdBuf,
361 0, this->_swap.extent.height,
362 this->_swap.extent.width, -(int32_t)this->_swap.extent.height);
363 } else {
364 this->_setViewportCmd(cmdBuf,
365 0, 0,
366 this->_swap.extent.width, this->_swap.extent.height);
367 }
368 }
369
380 void _setViewportCmd (vk::CommandBuffer cmdBuf,
381 int32_t x, int32_t y,
382 int32_t wid, int32_t ht);
383
384public:
385
387 int width () const { return this->_swap.extent.width; }
388
390 int height () const { return this->_swap.extent.height; }
391
392};
393
394} // namespace cs237
395
396#endif // !_CS237_WINDOW_HPP_
the base class for applications
Definition cs237-application.hpp:25
vk::Device _device
the logical device that we are using to render
Definition cs237-application.hpp:380
Queues< uint32_t > _qIdxs
the queue family indices
Definition cs237-application.hpp:381
Queues< vk::Queue > _queues
the device queues that we are using
Definition cs237-application.hpp:382
abstract base class for simple GLFW windows used to view buffers, etc.
Definition cs237-window.hpp:58
SwapChain _swap
buffer-swapping information
Definition cs237-window.hpp:267
virtual void cursorPos(double xpos, double ypos)
void enableKeyEvent(bool enable)
vk::Queue presentationQ() const
the presentation queue
Definition cs237-window.hpp:74
void enableScrollEvent(bool enable)
vk::Device device() const
return the logical device for this window
Definition cs237-window.hpp:68
bool _keyEnabled
true when the Key callback is enabled
Definition cs237-window.hpp:260
vk::Rect2D _getScissorsRect()
get the scissors rectangle for this window
Definition cs237-window.hpp:335
virtual void key(int key, int scancode, int action, int mods)
virtual void mouseButton(int button, int action, int mods)
vk::SurfaceKHR _surf
the Vulkan surface to render to
Definition cs237-window.hpp:266
void _setViewportCmd(vk::CommandBuffer cmdBuf, bool oglView=false)
add a command to set the viewport and scissor to the whole window
Definition cs237-window.hpp:352
bool _isVis
true when the window is visible
Definition cs237-window.hpp:259
void show()
Show the window (a no-op if it is already visible)
Definition cs237-window.hpp:93
vk::Queue graphicsQ() const
the graphics queue
Definition cs237-window.hpp:71
void enableCursorEnterEvent(bool enable)
void _recreateSwapChain()
Recreate the swap chain for this window; this redefines the _swap instance variable and is used when ...
Window(Application *app, CreateWindowInfo const &info)
the Window base-class constructor
void enableCursorPosEvent(bool enable)
int _ht
window dimensions
Definition cs237-window.hpp:258
bool _scrollEnabled
true when the Scroll callback is enabled
Definition cs237-window.hpp:264
bool _cursorEnterEnabled
true when the CursorEnter callback is enabled
Definition cs237-window.hpp:262
void setCursorMode(int mode)
bool windowShouldClose()
get the value of the "close" flag for the window
Definition cs237-window.hpp:116
virtual void cursorEnter(bool entered)
GLFWwindow * _win
the underlying window
Definition cs237-window.hpp:257
SwapChainDetails _getSwapChainDetails()
Get the swap-chain details for a physical device.
bool _mouseButtonEnabled
true when the MouseButton callback is enabled
Definition cs237-window.hpp:263
int width() const
the width of the window
Definition cs237-window.hpp:387
virtual ~Window()
destructor: it destroys the underlying GLFW window
Application * _app
the owning application
Definition cs237-window.hpp:256
bool _cursorPosEnabled
true when the CursorPos callback is enabled
Definition cs237-window.hpp:261
vk::Viewport _getViewport(bool oglView=false)
get the natural viewport for the window
Definition cs237-window.hpp:311
uint32_t _presentationQIdx() const
the presentation queue
Definition cs237-window.hpp:304
Application * app()
return the application pointer
Definition cs237-window.hpp:65
void _createSwapChain(bool depth, bool stencil)
Create the swap chain for this window; this initializes the _swap instance variable.
virtual void iconify(bool iconified)
method invoked on Iconify events.
void enableMouseButtonEvent(bool enable)
uint32_t _graphicsQIdx() const
the graphics queue-family index
Definition cs237-window.hpp:299
void refresh()
Definition cs237-window.hpp:78
virtual void scroll(double xoffset, double yoffset)
void _initAttachments(std::vector< vk::AttachmentDescription > &descs, std::vector< vk::AttachmentReference > &refs)
initialize the attachment descriptors and references for the color and optional depth/stencil-buffer
void hide()
Hide the window.
Definition cs237-window.hpp:86
virtual void reshape(int wid, int ht)
int height() const
the height of the window
Definition cs237-window.hpp:390
virtual void draw()=0
void _setViewportCmd(vk::CommandBuffer cmdBuf, int32_t x, int32_t y, int32_t wid, int32_t ht)
add a viewport command to the command buffer; this also sets the scissor rectangle.
int _wid
Definition cs237-window.hpp:258
Definition cs237-aabb.hpp:22
T present
the queue family that supports presentation
Definition cs237-application.hpp:361
T graphics
the queue family that supports graphics
Definition cs237-application.hpp:360
structure containing parameters for creating windows
Definition cs237-window.hpp:26
std::string title
window title
Definition cs237-window.hpp:29
int ht
the window height
Definition cs237-window.hpp:28
bool depth
do we need depth-buffer support?
Definition cs237-window.hpp:31
bool resizable
should the window support resizing
Definition cs237-window.hpp:30
bool needsDepthBuf() const
do we need a depth/stencil buffer for the window?
Definition cs237-window.hpp:53
CreateWindowInfo(int w, int h)
Definition cs237-window.hpp:48
CreateWindowInfo(int w, int h, std::string const &t, bool r, bool d, bool s)
Definition cs237-window.hpp:41
int wid
the window width
Definition cs237-window.hpp:27
bool stencil
do we need stencil-buffer support?
Definition cs237-window.hpp:32
information about the optional depth/stencil buffers for the window
Definition cs237-window.hpp:158
vk::ImageView view
image view for depth/image-buffer
Definition cs237-window.hpp:164
vk::Image image
depth/image-buffer image
Definition cs237-window.hpp:162
bool stencil
true if stencil-buffer is supported
Definition cs237-window.hpp:160
vk::Format format
the depth/image-buffer format
Definition cs237-window.hpp:161
vk::DeviceMemory imageMem
device memory for depth/image-buffer
Definition cs237-window.hpp:163
bool depth
true if depth-buffer is supported
Definition cs237-window.hpp:159
information about swap-chain support
Definition cs237-window.hpp:143
vk::SurfaceCapabilitiesKHR capabilities
Definition cs237-window.hpp:144
vk::PresentModeKHR choosePresentMode()
choose a presentation mode from the available modes; we prefer "mailbox" (aka triple buffering)
std::vector< vk::SurfaceFormatKHR > formats
Definition cs237-window.hpp:145
std::vector< vk::PresentModeKHR > presentModes
Definition cs237-window.hpp:146
vk::Extent2D chooseExtent(GLFWwindow *win)
get the extent of the window subject to the limits of the Vulkan device
vk::SurfaceFormatKHR chooseSurfaceFormat()
choose a surface format from the available formats
the collected information about the swap-chain for a window
Definition cs237-window.hpp:168
std::vector< vk::ImageView > views
image views for the swap buffers
Definition cs237-window.hpp:177
std::vector< vk::Image > images
images for the swap buffers
Definition cs237-window.hpp:176
std::vector< vk::Framebuffer > fBufs
frame buffers
Definition cs237-window.hpp:179
int size() const
return the number of buffers in the swap chain
Definition cs237-window.hpp:186
bool hasStencilBuffer() const
does the swap-chain support a stencil buffer?
Definition cs237-window.hpp:198
void initFramebuffers(vk::RenderPass renderPass)
allocate frame buffers for a rendering pass
std::optional< DepthStencilBuffer > dsBuf
optional depth/stencil-buffer
Definition cs237-window.hpp:178
SwapChain(vk::Device dev)
Definition cs237-window.hpp:181
bool hasDepthBuffer() const
does the swap-chain support a depth buffer?
Definition cs237-window.hpp:192
vk::SwapchainKHR chain
the swap chain object
Definition cs237-window.hpp:170
void cleanup()
destroy the Vulkan state for the swap chain
vk::Extent2D extent
size of swap buffer images
Definition cs237-window.hpp:172
vk::Format imageFormat
pixel format of image buffers
Definition cs237-window.hpp:171
int numAttachments
the number of framebuffer attachments
Definition cs237-window.hpp:173
vk::Device device
the owning logical device
Definition cs237-window.hpp:169
a container for a frame's synchronization objects
Definition cs237-window.hpp:208
vk::Semaphore renderFinished
Definition cs237-window.hpp:212
SyncObjs(SyncObjs &)=delete
SyncObjs(SyncObjs const &)=delete
vk::Fence inFlight
Definition cs237-window.hpp:214
~SyncObjs()
destroy the objects
Window * win
the owning window
Definition cs237-window.hpp:209
void submitCommands(vk::Queue q, vk::CommandBuffer const &cmdBuf)
SyncObjs(Window *w)
create a SyncObjs container
Definition cs237-window.hpp:218
vk::ResultValue< uint32_t > acquireNextImage()
acquire the next image from the window's swap chain.
SyncObjs(SyncObjs &&)=delete
vk::Semaphore imageAvailable
Definition cs237-window.hpp:210
void allocate()
helper method for allocating the synchronization objects in the constructor
vk::Result present(vk::Queue q, uint32_t imageIndex)
present the frame
void reset()
reset the in-flight fence of this frame