Skip to content

Commit 8b0d1ac

Browse files
committed
Support for runtime reloading of shaders
1 parent 1a1f25d commit 8b0d1ac

File tree

4 files changed

+32
-0
lines changed

4 files changed

+32
-0
lines changed

README.rst

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ Features
5656
- Management commands to create new projects and effects
5757
- Convenient wrappers for VAO, Shader, Texture, FBO
5858
- On-the-fly Shader and VAO negotiation of the needed buffer binding
59+
- Runtime re-loading shaders (press R)
5960
- Strict validation in most OpenGL operations with reasonable error feedback
6061
- Time line / Timer support
6162
- A highly pluggable framework

demosys/opengl/shader.py

+27
Original file line numberDiff line numberDiff line change
@@ -112,14 +112,33 @@ def prepare(self):
112112
Compiles all the shaders and links the program.
113113
If the linking is successful, it builds the uniform and attribute map.
114114
"""
115+
# Attempt to delete the current shader in case we are re-loading
116+
self.delete()
117+
118+
# Compile the separate shaders
115119
self.vert_source.compile()
116120
self.frag_source.compile()
117121
if self.geo_source:
118122
self.geo_source.compile()
119123
self.link()
124+
125+
# Build internal lookups
120126
self.build_uniform_map()
121127
self.build_attribute_map()
122128

129+
# We only need the programs for linking
130+
if self.vert_source:
131+
self.vert_source.delete(self.program)
132+
if self.frag_source:
133+
self.frag_source.delete(self.program)
134+
if self.geo_source:
135+
self.geo_source.delete(self.program)
136+
137+
def delete(self):
138+
"""Frees the memory and invalidates the name associated with the program"""
139+
if self.program:
140+
GL.glDeleteProgram(self.program)
141+
123142
def link(self):
124143
"""
125144
Links the program.
@@ -753,6 +772,14 @@ def compile(self):
753772
self.print()
754773
raise ShaderError("Failed to compile {} {}: {}".format(self.type_name(), self.name, message.decode()))
755774

775+
def delete(self, program=None):
776+
"""Frees the memory and invalidates the name associated with the shader object """
777+
# The shader will not be deleted if attached
778+
if program:
779+
GL.glDetachShader(program, self.shader)
780+
# Now we can delete it
781+
GL.glDeleteShader(self.shader)
782+
756783
def print(self):
757784
"""Print the shader lines"""
758785
print("---[ START {} ]---".format(self.name))

demosys/view/controller.py

+3
Original file line numberDiff line numberDiff line change
@@ -165,6 +165,9 @@ def key_event_callback(window, key, scancode, action, mods):
165165
if key == glfw.KEY_X and action == glfw.PRESS:
166166
screenshot.create()
167167

168+
if key == glfw.KEY_R and action == glfw.PRESS:
169+
resources.shaders.load()
170+
168171
# Forward the event to the effect manager
169172
MANAGER.key_event(key, scancode, action, mods)
170173

docs/source/controls.rst

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ Simply apply the ``view_matrix`` of the camera to your transformations.
2323
- ``D`` strafe right
2424
- ``Q`` down the y axis
2525
- ``E`` up the y axis
26+
- ``R`` reload all shaders
2627

2728
**Mouse Controls**
2829

0 commit comments

Comments
 (0)