1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
|
.. _programming-guide-eventloop:
The application event loop
==========================
In order to let pyglet process operating system events such as mouse and
keyboard events, applications need to enter an application event loop. The
event loop continuously checks for new events, dispatches those events, and
updates the contents of all open windows.
pyglet provides an application event loop that is tuned for performance and
low power usage on Windows, Linux and Mac OS X. Most applications need only
call::
pyglet.app.run()
to enter the event loop after creating their initial set of windows and
attaching event handlers. The :py:func:`~pyglet.app.run` function does not
return until all open windows have been closed, or until
:py:func:`pyglet.app.exit()` is called.
The pyglet application event loop dispatches window events (such as for mouse
and keyboard input) as they occur and dispatches the
:py:meth:`~pyglet.window.Window.on_draw` event to
each window after every iteration through the loop.
To have additional code run periodically or every iteration through the loop,
schedule functions on the clock (see
:ref:`guide_calling-functions-periodically`). pyglet ensures that the loop
iterates only as often as necessary
to fulfill all scheduled functions and user input.
Customising the event loop
--------------------------
The pyglet event loop is encapsulated in the
:py:class:`~pyglet.app.EventLoop` class, which provides
several hooks that can be overridden for customising its behaviour. This is
recommended only for advanced users -- typical applications and games are
unlikely to require this functionality.
To use the :py:class:`~pyglet.app.EventLoop` class directly, instantiate it and call `run`::
event_loop = pyglet.app.EventLoop()
event_loop.run()
Only one :py:class:`~pyglet.app.EventLoop` can be running at a time; when the
:py:meth:`~pyglet.app.EventLoop.run` method is called
the module variable :py:attr:`pyglet.app.event_loop` is set to the running
instance. Other pyglet modules such as :py:mod:`pyglet.window` depend on this.
Event loop events
^^^^^^^^^^^^^^^^^
You can listen for several events on the event loop instance. The most useful
of these is :py:meth:`~pyglet.app.EventLoop.on_window_close`, which is
dispatched whenever a window is closed. The default handler for this event
exits the event loop if there are no more windows. The following example
overrides this behaviour to exit the application whenever any window is
closed::
event_loop = pyglet.app.EventLoop()
@event_loop.event
def on_window_close(window):
event_loop.exit()
return pyglet.event.EVENT_HANDLED
event_loop.run()
Overriding the default idle policy
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
The :py:meth:`pyglet.app.EventLoop.idle` method is called every iteration of
the event loop. It is responsible for calling scheduled clock functions,
redrawing windows, and deciding how idle the application is. You can override
this method if you have specific requirements for tuning the performance
of your application; especially if it uses many windows.
The default implementation has the following algorithm:
1. Call :py:func:`pyglet.clock.tick` with ``poll=True`` to call any scheduled
functions.
2. Dispatch the :py:meth:`~pyglet.window.Window.on_draw` event and call
:py:meth:`~pyglet.window.Window.flip` on every open window.
3. Return the value of :py:func:`pyglet.clock.get_sleep_time`.
The return value of the :py:meth:`~pyglet.clock.get_sleep_time` method is
the number of seconds until the event loop needs to iterate again (unless
there is an earlier user-input event); or ``None`` if the loop can wait
for input indefinitely.
Note that this default policy causes every window to be redrawn during every
user event -- if you have more knowledge about which events have an effect on
which windows you can improve on the performance of this method.
Dispatching events manually
---------------------------
Earlier versions of pyglet and certain other windowing toolkits such as
PyGame and SDL require the application developer to write their own event
loop. This is usually just an inconvenience compared to
:py:func:`pyglet.app.run`, but can be necessary in some situations when
combining pyglet with other toolkits.
A simple event loop usually has the following form::
while True:
pyglet.clock.tick()
for window in pyglet.app.windows:
window.switch_to()
window.dispatch_events()
window.dispatch_event('on_draw')
window.flip()
The :py:meth:`~pyglet.window.Window.dispatch_events` method checks the window's
operating system event queue for user input and dispatches any events found.
The method does not wait for input -- if ther are no events pending, control is
returned to the program immediately.
The call to :py:func:`pyglet.clock.tick` is required for ensuring scheduled
functions are called, including the internal data pump functions for playing
sounds, animations, and video.
While it is possible to write your own event loop in this way, it is strongly
discouraged for the following reasons:
* The :py:class:`~pyglet.app.EventLoop` class provides plenty of hooks for most
toolkits to be integrated without needing to resort to a manual event loop.
* Because :py:class:`~pyglet.app.EventLoop` is tuned for specific operating
systems, it is more responsive to user events, and continues calling clock
functions while windows are being resized, and (on Mac OS X) the menu bar is
being tracked.
* It is difficult to write a manual event loop that does not consume
100% CPU while still remaining responsive to user input.
The capability for writing manual event loops remains for legacy support and
extreme cases where the developer knows what they are doing.
|