Note
Go to the end to download the full example code.
GUI Direct
Direct integration of glfw and wgpu-py without using the RenderCanvas library.
Demonstration for hardcore users that need total low-level control.
# run_example = false
import time
import atexit
import glfw
import wgpu
from wgpu.utils.glfw_present_info import get_glfw_present_info
# from triangle import setup_drawing_sync
from cube import setup_drawing_sync
# Setup glfw
glfw.init()
atexit.register(glfw.terminate)
# disable automatic API selection, we are not using opengl
glfw.window_hint(glfw.CLIENT_API, glfw.NO_API)
glfw.window_hint(glfw.RESIZABLE, True)
title = "wgpu glfw direct"
window = glfw.create_window(640, 480, title, None, None)
present_info = get_glfw_present_info(window)
context = wgpu.gpu.get_canvas_context(present_info)
# Initialize physical size once. For robust apps update this on resize events.
context.set_physical_size(*glfw.get_framebuffer_size(window))
def main():
draw_frame = setup_drawing_sync(context)
last_frame_time = time.perf_counter()
frame_count = 0
# render loop
while not glfw.window_should_close(window):
# process inputs
glfw.poll_events()
# resize handling
context.set_physical_size(*glfw.get_framebuffer_size(window))
# draw a frame
draw_frame()
# present the frame to the screen
context.present()
# stats
frame_count += 1
etime = time.perf_counter() - last_frame_time
if etime > 1:
print(f"{frame_count / etime:0.1f} FPS")
last_frame_time, frame_count = time.perf_counter(), 0
# dispose resources
context.unconfigure()
glfw.destroy_window(window)
# allow proper cleanup (workaround for glfw bug)
end_time = time.perf_counter() + 0.1
while time.perf_counter() < end_time:
glfw.wait_events_timeout(end_time - time.perf_counter())
if __name__ == "__main__":
main()