FocusTerra Realtime Wave Simulator
This repository contains all files that make up the realtime wave simulator exhibit at FocusTerra.
While the project was originally developed on macOS, it was later ported to Windows (Visual Studio 2019).
Dependencies are OpenGL, ImGui and SDL, and all of them are contained in this repository.
Table of Contents
- FocusTerra Realtime Wave Simulator
- Table of Contents
- Project Organization
- Resources
- Building the Project
- User Manual
-
Documentation
- Architecture
-
Classes
- Drawer (drawer.hpp, drawer.cpp)
- DrawingHandler (drawing_handler.hpp, drawing_handler.cpp)
- EfficientBlock (efficient_block.hpp, efficient_block.cpp)
- EventLogger (event_logger.hpp, event_logger.cpp)
- GuiHandler (gui_handler.hpp, gui_handler.cpp)
- Infrastructure (infrastructure.hpp, infrastructure.cpp)
- InputHandler (input_handler.hpp, input_handler.cpp)
- Message (message.hpp, message.cpp)
- PatternHandler (pattern_handler.hpp, pattern_handler.cpp)
- Pevent (pevent.hpp, pevent.cpp)
- PeventFactory (pevent.hpp, pevent.cpp)
- Shader (shader.hpp, shader.cpp)
- SlimBlockchainHandler (slim_blockchain_handler.hpp, slim_blockchain_handler.cpp)
- TimeoutHandler (timeout_handler.hpp, timeout_handler.cpp)
- Toolbox (toolbox.hpp, toolbox.cpp)
- WaveHandler (wave_handler.hpp, wave_handler.cpp)
- Enums (enums.hpp)
- ImGui Customization
- GLSL Shaders
- Workflow: Sample main function
Project Organization
The repo is organized in the following folders:
-
src
(and subfolders) contains all.cpp
-
include
(and subfolders) contains all headers -
shaders
contains all GLSL shaders -
lib
contains libraries -
FocusTerra
contains the Visual Studio project file -
FocusTerra\x64\Release
andFocusTerra\x64\Debug
contain the executables -
build
is left over from the macOS days (contains a currently brokenMakefile
)
Resources
Resources are organized in a single folder called ft_top
. It contains the following:
- A folder
bin
that contains the executable andSDL2.dll
- A folder
fonts
that contains the font used by the GUI, namelyCousine-Regular.ttf
- A folder
resource
that contains- The colour palette in
ft_palette.conf
,ft_palette.texture
- The Image Button images in the folder
images
- A folder
textures
that contains the predefined structure textures. There are two versions, one for the FocusTerra resolution (inrocket
), and one for Pascal's resolution (inhome
).
- The colour palette in
- A folder
shaders
that contains the used GLSL shaders.
The program accesses these resources at runtime.
File Formats
Todo
Building the Project
Windows
In order to build the project, the following steps must be followed
- Open the project in Visual Studio 2019
- Make sure the following files are excluded from the build (
Rightclick
,Properties
,Excluded from Build
,Yes
:block.hpp
blockchain_handler.hpp
block.cpp
blockchain_handler.cpp
imgui_example_sdl_opengl3.cpp
main.cpp
main_3d.cpp
main_refactored.cpp
- There should be a lot of include-errors reported. To fix this, adjust the paths:
- In the
Solution Explorer
, right click onFocusTerra
and selectProperties
- Navigate to
VCC Directories
and adjust theInclude Directories
. They should point toinclude
include\imgui
-
SDL2-2.0.14\include
(e.g. in WindowsSDL, or vclib (not in project))
- Also adjust the
Library Directories
. They should point toSDL2-2.0.14\lib\x64
. - The linked libraries can be seen in
Linker
,Input
,Additional Dependencies
. They should not require change, and should includeSDL2.lib
,SDL2main.lib
andopengl32.lib
.
- In the
- Open
main_testing.cpp
. This contains themain
function. Adjust it for the target system:- Adjust the screen resolution using the
#defines
at the top. Predefined are the resolutions for the FocusTerra screen and for Pascal's screen. - Adjust the
top_path
to point to the resource folder (i.e.ft_top
). - Adjust the Derived paths, where necessary. Potentially the
tex_path
may need adjustment. - Adjust the
tex_offscreen_*
variables to reflect the used texture dimensions.
- Adjust the screen resolution using the
- Open
input_handler.cpp
.- For a touchscreen, send an event after the
SDL_FINGER*
cases. On non-touch devices, send events after theSDL_MOUSE*
events.
- For a touchscreen, send an event after the
- Open
slim_blockchain_handler.cpp
- Depending on the system, uncomment or comment out the section marked
Uncomment on a touchscreen
.
- Depending on the system, uncomment or comment out the section marked
- Open
drawing_handler.cpp
- Depending on the system, uncomment or comment out the section marked
Uncomment on a touchscreen
.
- Depending on the system, uncomment or comment out the section marked
- Open
gui_handler.cpp
- Depending on the system, change the gui drawcall towards the end in
GuiHandler::update
. - Depending on the system, comment out or uncomment the two sections marked
Uncomment on a touchscreen
.
- Depending on the system, change the gui drawcall towards the end in
- Rightclick on
FocusTerra
, selectBuild
. - When the building is finished, create a
ft_top
folder according to the specifications above, and place the executable in thebin
folder. If there are library issues, also place a copy ofSDL2.dll
in thebin
folder. - Now the application is ready to be run.
macOS
Building on macOS is currently not supported.
User Manual
Todo
Documentation
The following is a documentation for all the code used in the project. As the project is currently at development halt, there are a few known issues that can not be addressed for the time being. They are pointed out in their respective sections.
Architecture
Todo
Classes
drawer.hpp, drawer.cpp)
Drawer (Description
A Drawer
is an object that can draw a single line of segments to the screen.
It handles single touch of the Zeichnen
functionality.
Note that this class relies on the caller to its methods to take care of the Opengl state (using shaders, binding buffers, etc.).
See DrawingHandler
for more information on how this class is to be used.
The coordinates used are typically OpenGL coordinates, i.e. in the range [-1, 1].
Usage
- Construct object (e.g. when new finger goes down)
-
start_drawing
with initial position (e.g. with finger down position) - Upon new location (e.g. finger motion to new position), setup Opengl state (use shader
draw
, bind FBO, bind VAO, bind VBO, set viewport, bind textures) and calldraw
. Callredraw
with all required FBO/texture combinations. - Destruct when drawing this line is finished (e.g. finger is lifted)
Constructors and Destructors
-
Drawer(Toolbox&)
-
Notes:
The argument is unused. It should be removed.
-
-
Drawer(const Drawer&) = default
-
~Drawer() = default
Public Function Members
-
void start_drawing(const float x, const float y)
:-
Description
Start drawing at coordinates (
x
,y
) (typically fingerdown location). Doesn't draw anything. -
Preconditions
-
Postconditions
Drawing start is set to (x, y).
-
Notes
Nothing is drawn by this function. (x, y) are typically the fingerdown coordinates.
-
-
bool draw(const float x, const float y, Toolbox& tb, bool drawing)
-
Description
Draw a stroke from the last position to (x, y), with strokewidth controlled by
drawing
(true
->tb.drawing_width
,false
->tb.erasing_width
). -
Preconditions
A
GL_ARRAY_BUFFER
is bound and the Opengl state is set as desired -
Postconditions
The points of the new segment have been calculated and uploaded to the currently bound
GL_ARRAY_BUFFER
, a call toglDrawArrays
has been done to draw the segment.Returns
true
if something was drawn, andfalse
else (typically when new point is same as last point). -
Notes
-
-
void redraw(Toolbox&)
-
Description:
Redraws the last drawn stroke.
-
Preconditions:
draw
has been called at least once, and since the last call todraw
, the then-boundGL_ARRAY_BUFFER
has not been modified. -
Postconditions:
A drawcall
glDrawArrays
has been issued. -
Notes:
-
-
void erase(const float x, const float y, Toolbox& tb)
DEPRECATED-
Description:
-
Preconditions:
-
Postconditions:
-
Notes:
Use
draw(x, y, tb, false)
instead.
-
-
int num_drawn()
-
Description:
Returns the number of calls to
draw
anderase
that have been performed by this object. -
Preconditions:
-
Postconditions:
-
Notes:
-
Private Function Members
-
bool calculate_points_(Toolbox& tb, bool drawing)
-
Description:
Calculate the triangulation of the stroke represented by the internal parameters, and store them in
points_
. The stroke width is controlled bydrawing
(true
->tb.drawing_width
,false
->tb.erasing_width
). -
Preconditions:
The points (
x0_
,y0_
) and (x1_
,y1_
) are set as desired. -
Postconditions:
If the points (
x0_
,y0_
), (x1_
,y1_
) are the same, false has been returned and the call has no effect. Else,true
has been returned, and the vertices that make up the segment between these two points with the desired width have been calculated and stored inpoints_
. -
Notes:
-
Public Data Members
N/A
Private Data Members
-
float x0_, y0_, x1_, y1_
Internal representation of old ('0') and new ('1') points.
-
int num_drawn_
Number of calls to
draw
anderase
. -
std::vector<float> points_
Vertex coordinates that represent the stroke between old and new points, as calculated by
calculate_points_
.
Notes
N/A
drawing_handler.hpp, drawing_handler.cpp)
DrawingHandler (Description
This class handles all aspects of the Zeichnen
/ Radieren
functionality. It constructs Drawer
objects on demand, manages their OpenGL infrastructure and chooses appropriate rendering targets.
Usage
- Construct one object at the start of the application with a fully initialized
Toolbox
- In each frame, call
update(Toolbox&)
Constructors and Destructors
-
DrawingHandler(Toolbox&)
-
Notes
Toolbox
is needed to find shader sources and obtain handles to the damping and wave textures.
-
-
~DrawingHandler()
Public Function Members
-
void update(Toolbox&)
-
Description:
Handles messages directed to it, parses events and takes the necessary actions (spawn / delete
Drawer
s, draw withDrawer
s using correct infrastructure). -
Preconditions:
The object is initialized correctly.
-
Postconditions:
The desired portions have been drawn to static damping and to the wave 1 texture.
-
Notes:
N/A
-
-
size_t num_drawers()
-
Description:
Returns the current number of active
Drawer
s. -
Preconditions:
N/A
-
Postconditions:
N/A
-
Notes:
N/A
-
Private Function Members
-
void draw_(Toolbox&)
-
Notes:
Does nothing.
-
-
bool in_wave_window_(const Toolbox&, const Pevent&) const
-
Description:
Check if an event is within the wave window.
-
Preconditions:
N/A
-
Postconditions:
Returns
true
if the event is in the wave window, andfalse
else (i.e. when event is in the GUI window). -
Notes:
N/A
-
-
std::array<float, 2> drawerCoordinates_(const Pevent&, const Toolbox&)
-
Description:
Compute drawer coordinates of a
Pevent
. -
Preconditions:
N/A
-
Postconditions:
N/A
-
Notes:
The method
Drawer::draw
should be called with the return values of this function.
-
Public Data Members
N/A
Private Data Members
-
std::list<std::pair<Drawer, SDL_FingerID> > drawpairs_
List of
Drawer
s and the correspondingSDL_FingerID
that spawned/controls them. -
int previous_mstate_
tb.mstate
from previous frame -
float draw_value_, erase_value_
DEPRECATED -
GLuint vao_, vbo_
Vertex Array Object and Vertex Buffer Object that are used by the
Drawer
s -
GLuint fbo_, fbo_wave_
Framebuffers that target the static damping texture (
fbo_
) and the wave texture 1 (fbo_wave_
) -
Shader shader_draw_
Shader that is used.
Notes
Todo
efficient_block.hpp, efficient_block.cpp)
EfficientBlock (Description
Represents a rectangular block. Scaled down version of the previous Block
, optimized for current feature set.
Usage
- Construct object
- Call
xywh
to obtain an efficient representation of the properties, convert it to vertices and draw them- Note: This should be batched for peak performance (see
SlimBlockchainHandler
)
- Note: This should be batched for peak performance (see
- Check if a point is inside the block using
is_inside
- Obtain the dimensions using
width
andheight
- Move the block to a new position using
translate
- Mark the block for removal using
request_removal
- Check if a block has been marked for removal using
needs_removal
Constructors and Destructors
-
EfficientBlock()
-
EfficientBlock(const EfficientBlock&)
-
EfficientBlock(int w, int h, int xoffset, int yoffset)
Note that (
xoffset
,yoffset
) are the coordinates of the lower left corner, with origin in the top left corner of the texture.
Public Function Members
-
void request_removal()
-
Description:
Mark block for removal
-
Preconditions:
N/A
-
Postconditions:
Block is marked for removal
-
Notes:
N/A
-
-
void translate(int new_x, int new_y)
-
Description:
Translate block to new (LLC) texel-coordinates.
-
Preconditions:
N/A
-
Postconditions:
Block has been translated to (
new_x
,new_y
). -
Notes:
N/A
-
-
bool needs_removal() const
-
Description:
Check if block is marked for removal.
-
Preconditions:
N/A
-
Postconditions:
Returns
true
if block has been marked for removal, andfalse
else. -
Notes:
N/A
-
-
glm::ivec4 xywh() const
-
Description:
Get coordinates and dimensions of block.
-
Preconditions:
N/A
-
Postconditions:
Returns a
glm::ivec4
;[0]
->x,[1]
->y,[2]
->width,[3]
->height. -
Notes:
-
-
bool is_inside(int x, int y) const
-
Description:
Check if a point lies within the block.
-
Preconditions:
N/A
-
Postconditions:
Returns
true
if point (x
,y
) is inside the block,false
else. -
Notes:
N/A
-
-
int width() const
-
Description:
Get the width of the block.
-
Preconditions:
N/A
-
Postconditions:
Returns the width of the block in texels.
-
Notes:
-
-
int height() const
-
Description:
Get the height of the block.
-
Preconditions:
-
Postconditions:
Returns the height of the block in texels.
-
Notes:
-
Private Function Members
N/A
Public Data Members
N/A
Private Data Members
-
glm::ivec4 xywh_
Coordinates and dimensions of block
-
bool needs_removal_
Is block marked for removal?
Notes
Todo
event_logger.hpp, event_logger.cpp)
EventLogger (Description
Logs events directly from the Toolbox
and sorts them by type in 16-deep circular buffers. Used exclusively for debugging.
Usage
- Construct object
- Once the events are collected in the
Toolbox
, callpush_events
- Convert
Pevent
s to strings usingptos
Constructors and Destructors
EventLogger()
EventLogger(const EventLogger&)
Public Function Members
-
void push_events(Toolbox&)
-
Description:
Catch events from
Toolbox
-
Preconditions:
New events have been captured by the
Toolbox
-
Postconditions:
The events have been sorted into their corresponding circular buffer
-
Notes:
N/A
-
Non-Member Functions
-
std::string ptos(const Pevent&, const Toolbox&)
-
Description:
Convert a
Pevent
to astd::string
. -
Preconditions:
N/A
-
Postconditions:
Returns a
std::string
that describes thePevent
. -
Notes:
The
Toolbox
argument is used for coordinate transformation.
-
Private Function Members
N/A
Public Data Members
-
unsigned i_down, i_up, i_move, i_other
Index where next event is to be inserted.
-
std::array<Pevent, 16> events_down
Events of down-type.
-
std::array<Pevent, 16> events_up
Events of up-type.
-
std::array<Pevent, 16> events_move
Events of move-type.
-
std::array<Pevent, 16> events_other
Events of other-type.
Private Data Members
N/A
Notes
N/A
gui_handler.hpp, gui_handler.cpp)
GuiHandler (Description
Draws and controls all functionality related to the GUI.
Usage
- Construct object with fraction of screen occupied by the gui
- Call
init
with aToolbox
that has a fully initializedInfrastructure
- In each frame, call
-
update
to queue drawing and update state -
render
to draw the GUI before swapping screen buffers
-
Constructors and Destructors
GuiHandler(float)
Public Function Members
-
void init(Toolbox& tb, const std::string path_img, const std::string path_ttf)
-
Description:
Initialize all GUI related infrastructure
-
Preconditions:
Fully setup
Toolbox
, correct paths to images and fonts -
Postconditions:
Fonts and images have been loaded, ImGui is fully initialized.
-
Notes:
-
-
void update(Toolbox& tb)
-
Description:
Update the GUI
-
Preconditions:
N/A
-
Postconditions:
Tasks performed:
- Check if application has to quit and handle this
- Check if there is a new finger that controls the GUI, or if the controlling finger was released
- Handle any input to the GUI, pass it on to ImGui for processing
- Construct the GUI
-
Notes:
N/A
-
-
void render(Toolbox& tb)
-
Description:
Render the GUI to the screenbuffer.
-
Preconditions:
N/A
-
Postconditions:
The GUI is drawn on the screenbuffer.
-
Notes:
N/A
-
Private Function Members
-
bool isInGuiWindow_(const Pevent&) const
-
Description:
Check if a
Pevent
is within the GUI window. -
Preconditions:
N/A
-
Postconditions:
Returns
true
if thePevent
is within the GUI window,false
else. -
Notes:
N/A
-
-
void draw_gui_(Toolbox& tb)
-
Description:
Construct a semi-old version of the GUI
-
Preconditions:
All resources have been loaded
-
Postconditions:
N/A
-
Notes:
N/A
-
-
void draw_gui_ft_(Toolbox& tb)
-
Description:
Construct the GUI used at FocusTerra
-
Preconditions:
All resources have been loaded.
-
Postconditions:
N/A
-
Notes:
N/A
-
-
void draw_old_gui_(Toolbox& tb)
-
Description:
Construct the old GUI.
-
Preconditions:
All resources have been loaded
-
Postconditions:
N/A
-
Notes:
N/A
-
-
void load_button_textures_(const std::string path)
-
Description:
Load all necessary images into textures.
-
Preconditions:
All images can be found in
path
, and they all have the correct format. -
Postconditions:
All images have been uploaded to the GPU as textures and can be accessed with the corresponding private members
btex_*_*
. -
Notes:
N/A
-
-
void load_image_to_texture_(const std::string file, GLuint& texture)
-
Description:
Load an image (rgba) from file into an OpenGL texture using stbi.
-
Preconditions:
The target image has format RGBA
-
Postconditions:
The image has been uploaded to the GPU as a texture, and its handle has been copied into the address specified by
texture
-
Notes:
-
Public Data Members
N/A
Private Data Members
-
float gui_pos_
: Portion of screen occupied by GUI -
ImGuiIO io_
: ImGui IO object -
SDL_Event next_event_
: Next event for event chains that span several frames (raising, lowering finger) -
SDL_FingerID fingerID_
: Finger currently controlling the GUI -
SDL_TouchID deviceID_
: ID of the currently connected touch device (read: touchscreen) -
bool lowering_finger_
: Are we in the event chain of lowering a finger? -
bool raising_finger_
: Are we in the event chain of raising a finger? -
ImFont* font_
: Font used by ImGui -
GLuint btex_*_on, btex_*_off
: 12 OpenGL texture handles corresponding to images drawn in GUI -
EventLogger evlog
: EventLogger, only present while debugging
Notes
N/A
infrastructure.hpp, infrastructure.cpp)
Infrastructure (Description
Creates and stores the SDL and OpenGL infrastructures (window, renderer, context).
Usage
- Default construct object
- Call
init
- at end of life, call
quit
Constructors and Destructors
Infrastructure(SDL_Window*, SDL_RENDERER*, SDL_GLContext)
Infrastructure()
Infrastructure(const Infrastructure&)
Public Function Members
-
SDL_Window* window() const
- Description: Return the window
- Preconditions: N/A
- Postconditions: The window is returned
- Notes: N/A
-
SDL_Renderer* renderer() const
- Description: Return the renderer
- Preconditions: N/A
- Postconditions: The renderer is returned
- Notes: N/A
-
SDL_Context context() const
- Description: Return the context
- Preconditions: N/A
- Postconditions: The context is returned
- Notes: N/A
-
bool init(const std::string name, const int width, const int height)
- Description: Initialize everything
- Preconditions:
name
is the window name,width
andheight
are width and height of the window in pixels - Postconditions: An OpenGL context has been created and connected to the
SDL_Window
. The OpenGL functions have been loaded via glad. - Notes: N/A
-
void destroy()
- Description: Destruct all resources
- Preconditions: N/A
- Postconditions: All resources have been destroyed
- Notes: N/A
-
void quit()
- Description: Quit the application
- Preconditions: N/A
- Postconditions: All resources have been destroyed and the application has quit
- Notes: N/A
Private Function Members
N/A
Public Data Members
N/A
Private Data Members
-
SDL_Window* window_
: Window -
SDL_Renderer* renderer_
: Renderer -
SDL_Context context_
: Context
Notes
N/A
input_handler.hpp, input_handler.cpp)
InputHandler (Description
Reads all input from SDL, preprocesses it and stores it in the Toolbox
event chain
Usage
- Construct object
- At the beginning of each frame, call
update
to update theToolbox
event chain
Constructors and Destructors
InputHandler()
Public Function Members
-
void update(Toolbox& tb)
- Description: Collect all current SDL events and store them as
Pevent
s in the toolboxevents
- Preconditions: N/A
- Postconditions: The toolbox
events
have been updated - Notes: N/A
- Description: Collect all current SDL events and store them as
Private Function Members
N/A
Public Data Members
N/A
Private Data Members
N/A
Notes
N/A
message.hpp, message.cpp)
Message (Description
Represents a message to one of the components
Usage
- Construct object using a message, e.g.
WAVEMESSAGE::RESET_DAMPING
orPATTERNMESSAGE::PATTERN_SSH
(see*MESSAGE
types in enums) - Read messages using the pattern
Message m = getSomeMessage();
if(m.target == MESSAGETARGET::WAVE){
WAVEMESSAGE message = std::get<WAVEMESSAGE>(m.message);
switch(message){
case WAVEMESSAGE::RESET_WAVE:
//process RESET_WAVE here
case WAVEMESSAGE::RESET_DAMPING:
//process RESET_DAMPING here
default:
break;
}
}
Constructors and Destructors
Message()
Message(BLOCKCHAINMESSAGE message)
Message(DRAWERMESSAGE message)
Message(GUIMESSAGE message)
Message(WAVEMESSAGE message)
Message(PATTERNMESSAGE message)
Public Function Members
N/A
Private Function Members
N/A
Public Data Members
MESSAGETARGET target
std::variant<BLOCKCHAINMESSAGE, DRAWERMESSAGE, GUIMESSAGE, WAVEMESSAGE, PATTERNMESSAGE> message
bool handled
Private Data Members
N/A
Notes
Access the correct message by first checking the target
, and then calling T message = std::get<T>(m.message)
, where T
is the message type.
pattern_handler.hpp, pattern_handler.cpp)
PatternHandler (Description
Handles the placement of predefined patterns.
Usage
Todo
Constructors and Destructors
PatternHandler()
PatternHandler(const PatternHandler&)
-
PatternHandler(Toolbox& tb)
- Notes: This constructor sets up the necessary OpenGL objects and loads the textures
Public Function Members
-
void update(Toolbox& tb)
- Description: Checks the mailbox, and upon requests draws the desired structure into the static damping texture
- Preconditions: Correctly constructed object
- Postconditions: All messages concerning patterns have been handled
- Notes: N/A
Private Function Members
-
bool load_damping_texture_(const Toolbox& tb, const std::string file, GLuint* texture_target)
- Description: Load a damping texture from file, upload it to the GPU and return the texture handler in the
GLuint
out argument. - Preconditions:
file
is the texture file without extension,texture_target
is an out argument - Postconditions: The texture has been uploaded to the GPU and has handler
texture_target
- Notes: N/A
- Description: Load a damping texture from file, upload it to the GPU and return the texture handler in the
Public Data Members
N/A
Private Data Members
GLuint texture_*_
Notes
N/A
pevent.hpp, pevent.cpp)
Pevent (Description
Stores SDL_Event
s in a more convenient way.
Usage
Todo
Constructors and Destructors
Pevent()
Pevent(const SDL_Event&)
Public Function Members
N/A
Private Function Members
N/A
Public Data Members
SDL_Event event
unsigned itcoord_x, itcoord_y
float fscoord_x, fscoord_y
PEVENTTYPE type
SDL_FingerID finger_id
Private Data Members
N/A
Notes
Objects of this type should be constructed by something like the PeventFactory
.
pevent.hpp, pevent.cpp)
PeventFactory (Description
Constructs Pevent
objects.
Usage
Todo
Constructors and Destructors
PeventFactory()
-
PeventFactory(const int screen_w, const int screen_h, const int texture_w, const int texture_h, const int texoffset_left, const int texoffset_right, const int texoffset_bottom, const int texoffset_top)
- This constructor should be changed to take a
Toolbox
instead.
- This constructor should be changed to take a
Public Function Members
-
Pevent operator()(const SDL_Event&) const
- Description: Generate a
Pevent
from andSDL_Event
- Preconditions: N/A
- Postconditions: Returns a
Pevent
that corresponds to theSDL_Event
passed as argument. - Notes: N/A
- Description: Generate a
Private Function Members
N/A
Public Data Members
N/A
Private Data Members
-
int screen_w_, screen_h_
: screen width and height in pixels -
int texture_w_, texture_h_
: texture width and height in texels -
int texoffset_left_, texoffset_right_, texoffset_bottom_, texoffset_top_
: screen offsets in the texture (buffer sizes)
Notes
N/A
shader.hpp, shader.cpp)
Shader (Description
Represents a GLSL shader.
Usage
Todo
Constructors and Destructors
Shader()
-
Shader(const char* vertexPath, const char* fragmentPath)
- Notes: This is the correct way to initialize a shader from sources.
Shader(const Shader&)
Shader& operator=(const Shader&)
~Shader()
Public Function Members
-
void clean_up()
- Description: Clean up the OpenGL resources associated with this shader
- Preconditions: N/A
- Postconditions: N/A
- Notes: N/A
-
void use()
- Description: Activates the shader
- Preconditions: Correctly constructed object
- Postconditions: The shader program is used
- Notes: N/A
-
void setBool(const std::string& name, bool value) const
- Description: Set a boolean uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setInt(const std::string& name, int value) const
- Description: Set an integer uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setFloat(const std::string& name, float value) const
- Description: Set a float uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setVec2(const std::string& name, glm::vec2 value) const
- Description: Set a vec2 uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setVec3(const std::string& name, glm::vec3 value) const
- Description: Set a vec3 uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setVec4(const std::string& name, glm::vec4 value) const
- Description: Set a vec4 uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
-
void setMat(const std::string& name, glm::mat4 value) const
- Description: Set a mat4 uniform
- Preconditions:
name
is the name of the target uniform,value
the target value. - Postconditions: The target uniform has been set to
value
- Notes: N/A
Private Function Members
N/A
Public Data Members
-
unsigned int ID
: OpenGL handle of the shader program
Private Data Members
N/A
Notes
N/A
slim_blockchain_handler.hpp, slim_blockchain_handler.cpp)
SlimBlockchainHandler (Description
Handles everything related to rectangular blocks, especially the immediate mode (Spielen
).
Drawing is handled in a batched manner: There are eraselists and drawlists that store objects that are to be drawn or erased.
The drawcalls are issued at the very end of the update-cycle, all in one.
Usage
Todo
Constructors and Destructors
SlimBlockchainHandler(Toolbox& tb)
Public Function Members
-
void update(Toolbox& tb)
- Description: Update the toolbox - first handle messages, then handle events, update erase-/drawlists (dynamic and static) and finally draw the updated state to the dynamic damping and wave 1 textures.
- Preconditions: Fully initialized object
- Postconditions: The state has been updated and drawn to the necessary textures
- Notes: N/A
-
size_t num_blocks()
- Description: Get the current number of active blocks
- Preconditions: N/A
- Postconditions: The number of blocks is returned
- Notes: N/A
Private Function Members
-
void clear_blocks_()
- Description: Erases all current blocks
- Preconditions: N/A
- Postconditions: All blocks have been erased,
dragpairs_
andblockchain_
are empty - Notes: N/A
-
void update_blocks_(const Toolbox&, bool reload_all)
- Description: Updates the state of the block tracking. First, the draw-/eraselists are checked for duplicated inputs (e.g. draw at R, erase at R, draw at R', erase at R', draw at R''). Then the vertices of blocks to be drawn / erased are calculated, and the queues are executed in the order: erase dynamic blocks, draw static blocks, draw dynamic blocks.
- Preconditions: N/A
- Postconditions: N/A
- Notes: N/A
-
bool in_wave_window_(const Toolbox&, const Pevent&) const
- Description: Check if a
Pevent
is within the wave window - Preconditions: N/A
- Postconditions: Returns
true
if the passedPevent
is in the wave window,false
else - Notes: N/A
- Description: Check if a
-
void xywhs_to_vertices_(const Toolbox&, const std::vector<bool>& duplicates, const std::vector<glm::ivec4>& xywhs, std::vector<float>& vertices) const
- Description: Calculate all vertices corresponding to the
EfficientBlock
xywh
coordinates stored inxywhs
and store them invertices
, while ignoringxywhs[i]
ifduplicates[i] == true
. - Preconditions: N/A
- Postconditions: The vertices corresponding to the non-duplicate marked
xywhs
coordinates have been calculated and are stored invertices
. Previous contents ofvertices
have been deleted. - Notes: If duplicates are not tracked (e.g. for static blocks, which can't be erased), pass
std::vector<bool>(xywhs.size(), false)
asduplicates
.
- Description: Calculate all vertices corresponding to the
-
void find_dynamic_duplicates_()
- Description: Find duplicates in the dynamic draw-/eraselists and store them in the members
drawlist_dynamic_duplicates_
anderaselist_dynamic_duplicates_
, respectively - Preconditions: N/A
- Postconditions: The dynamic duplicate storing members have been updated according to the current dynamic draw-/eraselists
- Notes: This takes care that one erase can only delete one draw.
- Description: Find duplicates in the dynamic draw-/eraselists and store them in the members
Public Data Members
N/A
Private Data Members
-
int previous_mstate_
: The tb.mstate of the previous frame -
std::list<EfficientBlock> blockchain_
: list of currentEfficientBlocks
-
std::list<std::pair<EfficientBlock*, SDL_FingerID> > dragpairs_
: list of blocks and their controlling finger -
Shader shader_drawblocks_
: Shader used to draw blocks -
GLuint vao_, vbo_
: OpenGL infrastructure, vertex array object and vertex buffer object -
GLuint fbo_wave_, fbo_dynamic_, fbo_static_
: framebuffer objects targeting the wave 1 texture, the dynamic damping texture, and the static damping texture, respectively -
std::vector<glm::ivec4> drawlist_static_
: xywh coordinates of blocks to be drawn to the static damping texture -
std::vector<glm::ivec4> drawlist_dynamic_
: xywh coordinates of blocks to be drawn to the dynamic damping texture -
std::vector<glm::ivec4> eraselist_dynamic_
: xywh coordinates of blocks to be erased from the dynamic damping texture -
std::vector<bool> drawlist_dynamic_duplicates_
: marks which indices of the dynamic drawlist are duplicates (i.e. paired with an entry of the dynamic eraselist) -
std::vector<bool> eraselist_dynamic_duplicates_
: marks which indices of the dynamic eraselist are duplicates (i.e. paired with an entry of the dynamic drawlist)
Notes
One should unify all the framebuffers used over all objects, they could be collected in the Toolbox
. The same is true for functions that check whether a Pevent
is in the GUI or in the wave window.
There used to be issues with this class, because duplicates were not tracked / tracked incorrectly. For example, one frame can contain several move commands, e.g. move to R, then move to R'.
Because erasing is completed before drawing is started, this would leave blocks at R and R', unless the draw and erase at R are marked as net-0x90.
timeout_handler.hpp, timeout_handler.cpp)
TimeoutHandler (Description
Keeps track of the time since the last user input, and resets the state of the application when a set threshold is surpassed.
Usage
Todo
Constructors and Destructors
-
TimeoutHandler(int seconds_to_timeout)
- Notes: This constructs a timeout handler with timeout
seconds_to_timeout
- Notes: This constructs a timeout handler with timeout
Public Function Members
-
void update(Toolbox& tb)
- Description: Check how much time has passed since an event has last come in, and if it exceeds the timeout, post reset messages and reset the timer.
- Preconditions: N/A
- Postconditions: If the timer exceeds the timeout, a reset has been posted. If
tb.events
is not empty or the timer has exceeded the timeout, the timer has been reset. - Notes: The reset messages need to be adjusted if more components are added. Potentially, one could add a messagetarget
ALL
, with a messageRESET
everyone listens to.
Private Function Members
N/A
Public Data Members
N/A
Private Data Members
-
int seconds_to_timeout_
: Number of seconds that constitute a timeout -
std::chrono::time_point<std::chrono::high_resolution_clock> time_last_input_
: Time of last user input
Notes
N/A
toolbox.hpp, toolbox.cpp)
Toolbox (Description
Stores all information needed by several components. Has a mailbox to which messages to other components can be posted and a event chain where user input is stored. The Toolbox is passed from one component to the next, so each component has all the tools when it needs to perform a specific task.
Usage
Todo
Constructors and Destructors
Toolbox(const int screen_w, const int screen_h, const int texture_w, const int texture_h, const int texoffset_left, const int texoffset_right, const int texoffset_bottom, const int texoffset_top, const std::string shaderpath, const std::string texturepath, const std::string resourcepath)
Public Function Members
-
void newFrame()
- Description: Start a new frame
- Preconditions: N/A
- Postconditions:
mailbox
andevents
have been cleared - Notes: Call this function at the very beginning of each frame
Private Function Members
N/A
Public Data Members
Check the source, there are ~50 members and they are well documented.
Private Data Members
N/A
Notes
For full initialization, this class needs external objects. More precisely, after construction, it needs a WaveHandler
to call WaveHandler::generate_and_transfer_textures
on it.
Else the textures stay un-initialized.
wave_handler.hpp, wave_handler.cpp)
WaveHandler (Description
Handles simulation and rendering of the wave.
Usage
Todo
Constructors and Destructors
WaveHandler(Toolbox&)
Public Function Members
-
bool initialize(const std::string damping_file, const std::string palette_file)
- Description: Fully initialize the object (especially shaders, textures, framebuffers and other OpenGL infrastructure)
- Preconditions: N/A
- Postconditions: The object is fully initialized
- Notes: Without a call to this function, the object is not functional.
-
void update(Toolbox&)
- Description: Handle all messages addressed to this object
- Preconditions: N/A
- Postconditions: All relevant messages have been handled
- Notes: N/A
-
void prepare_step()
- Description: Prepare for timestepping by combining the dynamic and static damping textures into one texture.
- Preconditions: N/A
- Postconditions: The texture
tex_comp_damp_
has been updated - Notes: N/A
-
bool step(Toolbox& tb, const int num_dsteps)
- Description: Perform
num_dsteps
double-timesteps on the wave - Preconditions: N/A
- Postconditions: 2*
num_dsteps
timesteps have been performed and the latest step is stored intex_comp_wave_1_
. - Notes: Always returns
true
- Description: Perform
-
bool render()
- Description: Render
tex_comp_damp_
andtex_wave_1_
using the color schemetex_palette_
to the current screenbuffer using shadershdr_2d_
- Preconditions: N/A
- Postconditions: The state has been rendered
- Notes: Always returns
true
- Description: Render
-
GLuint get_damping_tex() const
- Description: Get the combined damping texture
- Preconditions: N/A
- Postconditions: The combined damping texture handler is returned
- Notes: N/A
-
unsigned get_width() const
- Description: Get the window width
- Preconditions: N/A
- Postconditions: The window width is returned
- Notes: N/A
-
unsigned get_height() const
- Description: Get the window height
- Preconditions: N/A
- Postconditions: The window height is returned
- Notes: N/A
-
void generate_and_transfer_textures(Toolbox&)
- Description: Generate textures and transfer these and existing textures to the
Toolbox
. - Preconditions: N/A
- Postconditions: Three textures have been generated and initialized correctly:
tb.tex_damp_clean
,tb.tex_wave_clean
,tb.tex_const_zero
. Four textures have been transferred to theToolbox
:tex_comp_damp_dynamic_
,tex_comp_damp_static_
,tex_comp_wave_0_
,tex_comp_wave_1_
- Notes: This function has to be called to fully initialize the
Toolbox
- Description: Generate textures and transfer these and existing textures to the
Private Function Members
-
bool initialize_2D_data_()
- Description: Initialize OpenGL infrastructure to perform timesteps
- Preconditions: N/A
- Postconditions: The objects
num_elements_2d
,vao_2d_
,vbo_2d_
,ebo_2d_
have been initialized - Notes: N/A
-
bool initialize_3D_data_()
- Description: Do not call
- Preconditions: N/A
- Postconditions: N/A
- Notes: N/A
-
bool initialize_comp_data_(const std::string)
- Description: Initialize damping and wave textures, and time stepping framebuffers
- Preconditions: The argument specifies the damping file, without extension.
- Postconditions: The vanilla damping profile has been loaded from file, or a backup has been generated in case the file is missing.
The damping textures
tex_comp_damp_
,tex_comp_damp_dynamic_
,tex_comp_damp_static_
have been generated and initialized. The wave texturestex_comp_wave_0_
andtex_comp_wave_1_
have been generated and initialized. The framebuffersfb_comp_0_
,fb_comp_1_
,fb_comp_damp_
have been generated and initialized. - Notes: Always returns
true
-
bool initialize_render_data_(const std::string)
- Description: Load the colour palette and upload it to the GPU, initialize rendering infrastructure
- Preconditions: The argument specifies a file where the colour palette texture can be found (without extension)
- Postconditions: The colour palette has been loaded from file and uploaded to the GPU in texture
tex_palette_
. The rendering infrastructure, namelyvao_render_
,vbo_render_
,ebo_render_
have been generated and initialized. - Notes: Always returns
true
, except when the colour palette file is invalid.
-
bool initialize_shaders_()
- Description: Initialize all shaders
- Preconditions: N/A
- Postconditions: The following shaders have been initialized from source:
shdr_step_
(stepwave),shdr_2d_
(render2d),shdr_2d_dbg_
(render2d),shdr_damp_
(combine_damping) and their uniforms have been set - Notes: Always returns
true
-
bool load_damping_(const std::string)
- Description: Load a damping texture from file into
tex_damp_data_
- Preconditions: The argument specifies a damping file without file extension
- Postconditions: The damping texture has been loaded into
tex_damp_data_
. On success,true
has been returned,false
else. - Notes: In the same directory, there needs to be a
.conf
and a.texture
file with the specified name. This function is very inefficient and can be optimized.
- Description: Load a damping texture from file into
-
bool load_palette_(const std::string)
- Description: Load the colour palette texture from file into
palette_
- Preconditions: The argument specifies the palette file without file extension
- Postconditions: The colour palette texture has been loaded into
palette_
. On success,true
has been returned,false
else. - Notes: In the same directory, there needs to be a
.conf
and a.texture
file with the specified name.
- Description: Load the colour palette texture from file into
Public Data Members
N/A
Private Data Members Only relevant members are listed
-
const int width_, height_
: Window dimensions -
const int texwidth_, texheight_
: Texture dimensions -
const int texoffset_left_, texoffset_right_, texoffset_bottom_, texoffset_top_
: Offsets in pixels in textures from the screen -
const std::string shader_path_
: Path to shaders -
GLuint vbo_2d_, vao_2d_, ebo_2d_, num_elements_2d_
: Timestepping OpenGL infrastructure -
GLuint vbo_render_, vao_render_, ebo_render_, num_elements_render_
: Rendering OpenGL infrastructure -
GLuint fb_comp_0_, fb_comp_1_, fb_comp_damp_
: Framebuffers targeting wave texture 0tex_comp_wave_0_
, wave texture 1tex_comp_wave_1_
and the combined damping texturetex_comp_damp_
, respectively. -
GLuint tex_comp_wave_0_, tex_comp_wave_1_
: Wave textures 0 and 1, between which timestepping happens. The latest state is always intex_comp_wave_1_
, and the first step of a chain goes from 1 to 0. -
GLuint tex_comp_damp_
: Combined damping texture -
GLuint tex_comp_damp_dynamic_, tex_comp_damp_static_
: Dynamic damping texture (for obstacles that are dynamic like "Spielen" blocks) and static damping texture (for static obstacles like drawings and predefined structures) -
std::vector<float> palette_
: colour palette texture data -
GLuint tex_palette_
: Colour palette texture handle -
Shader shdr_step_, shdr_2d_, shdr_2d_dbg_, shdr_damp_
: Shaders used for timestepping, rendering (without and with debugging) and damping combination -
std::vector<float> tex_wave_data_, tex_damp_data_
: The vanilla wave and damping texture data
Notes
This class predates the Toolbox and is not yet well integrated into the flow. All member functions assume the object is fully initialized.
enums.hpp)
Enums (MSTATE
(Mouse State): IMMEDIATE
(Spielen), PLACE
, DELETE
, MOVE
, DRAW
(Zeichnen), Erase
(Radieren)
SSTATE
(Source State): RUN
, STOP
GSTATE
(Game State): RUN
, FREEZE
, EXIT
MESSAGETARGET
(Possible targets for messages): BLOCKCHAIN
, DRAWER
, GUI
, WAVE
, PATTERN
BLOCKCHAINMESSAGE
(Possible messages to SlimBlockchainHandler
): CLEAR
, PATTERN_SINGLESLIT
, PATTERN_DOUBLESLIT
, PATTER_LATTICE
, PATTERN_WAVEGUIDE
, PATTERN_SSH
, PATTERN_FRESNEL
PATTERNMESSAGE
(Possible messages to PatternHandler
): PATTERN_SINGLESLIT
, PATTERN_DOUBLESLIT
, PATTER_LATTICE
, PATTERN_WAVEGUIDE
, PATTERN_SSH
, PATTERN_FRESNEL
DRAWERMESSAGE
(Possible messages to DrawingHandler
): CLEAR
GUIMESSAGE
(Possible messages to GuiHandler
): n/a
WAVEMESSAGE
(Possible messages to WaveHandler
): RESET_WAVE
, RESET_DAMPING
, DIMENSION_2D
, DIMENSION_3D
, DEBUG_ON
, DEBUG_OFF
PEVENTTYPE
(Possible Pevent
types): DOWN
, UP
, MOVE
, OTHER
ImGui Customization
The ImGui library has been customized in the following way:
-
imgui_widgets.cpp
:-
ImGui::PlotEx
: Removed tooltip
-
-
imgui_impl_sdl.h
:-
ImGui_ImplSDL2_Touch_ProcessEvent
: Added declaration -
ImGui_ImplSDL2_Touch_UpdateMousePosAndButtons
: Added declaration
-
-
imgui_impl_sdl.cpp
:-
ImGui_ImplSDL2_Touch_ProcessEvent
: Added implementation -
ImGui_ImplSDL2_Touch_UpdateMousePosAndButtons
: Added implementation -
ImGui_ImplSDL2_NewFrame
: Changed to callImGui_ImplSDL2_Touch_UpdateMousePosAndButtons
instead ofImGui_ImplSDL2_UpdateMousePosAndButtons
-
-
Notes:
- On touch devices, one should generally call
ImGui_ImplSDL2_Touch_ProcessEvent
rather thanImGui_ImplSDL2_ProcessEvent
- To run
ImGui
demos, one must revert the change toImGui_ImplSDL2_NewFrame
(see comments).
- On touch devices, one should generally call
GLSL Shaders
combine_damping
copy_texture
draw
draw_blocks
render2d
stepwave
main
function
Workflow: Sample Todo