Skip to content

Commit e202cfc

Browse files
authored
Improved some Camera2D methods, and filled in temporary docs (#2467)
* slight correction of Rect.__bool__ * Update `Camera2D.match_...` based on feedback. * Update Camera.rst with temporary docs pulled from README.md * finished converting from md to rst * small doc updates, example updating, and method name changes
1 parent 33c368d commit e202cfc

21 files changed

+194
-92
lines changed

arcade/camera/README.md

+6-5
Original file line numberDiff line numberDiff line change
@@ -30,11 +30,11 @@ world space is equal to one unit in view space
3030

3131
### Projection Matrices
3232
The projection matrix takes the positions of objects in view space and projects them into screen space.
33-
depending on the type of projection matrix how this exactly applies changes. Projection matrices along
33+
depending on the type of projection matrix how this exactly applies changes. Projection matrices alone
3434
do not fully project objects into screen space, instead they transform positions into unit space. This
3535
special coordinate space ranges from -1 to 1 in the x, y, and z axis. Anything within this range will
3636
be transformed into screen space, and everything outside this range is discarded and left undrawn.
37-
you can conceptualise projection matrices as taking a 6 sided 3D prism volume in view space and
37+
you can conceptualise projection matrices as taking a 6 sided 3D volume in view space and
3838
squashing it down into a uniformly sized cube. In every case the closest position projected along the
3939
z-axis is given by the near value, while the furthest is given by the far value.
4040

@@ -71,19 +71,20 @@ not be drawn.
7171
- `arcade.camera.Projector` is a `Protocol` used internally by arcade
7272
- `Projector.use()` sets the internal projection and view matrices used by Arcade and Pyglet
7373
- `Projector.activate()` is the same as use, but works within a context manager using the `with` syntax
74-
- `Projector.map_screen_to_world_coordinate(screen_coordinate, depth)`
74+
- `Projector.unproject(screen_coordinate)`
7575
provides a way to find the world position of any pixel position on screen.
76+
- `Projector.project(world_coordinate)`
77+
provides a way to find the screen position of any position in the world.
7678
- There are multiple data types which provide the information required to make view and projection matrices
7779
- `camera.CameraData` holds the position, forward, and up vectors along with a zoom value used to create the
7880
view matrix
7981
- `camera.OrthographicProjectionData` holds the left, right, bottom, top, near, far values needed to create a
8082
orthographic projection matrix
8183
- `camera.PerspectiveProjectionData` holds the aspect ratio, field of view, near and far needed to create a
8284
perspective projection matrix.
83-
- both ProjectionData data types also provide a viewport for setting the draw area when using the camera.
8485
- There are three primary `Projectors` in `arcade.camera`
8586
- `arcade.camera.Camera2D` is locked to the x-y plane and is perfect for most use cases within arcade.
8687
- `arcade.camera.OrthographicProjector` can be freely positioned in 3D space, but the scale of objects does not
8788
depend on the distance to the projector
88-
- [not yet implemented ] `arcade.camera.PerspectiveProjector` can be freely position in 3D space,
89+
- `arcade.camera.PerspectiveProjector` can be freely position in 3D space,
8990
and objects look smaller the further from the camera they are

arcade/camera/camera_2d.py

+64-59
Original file line numberDiff line numberDiff line change
@@ -329,16 +329,17 @@ def equalise(self) -> None:
329329
x, y = self._projection_data.rect.x, self._projection_data.rect.y
330330
self._projection_data.rect = XYWH(x, y, self.viewport_width, self.viewport_height)
331331

332-
def match_screen(
332+
def match_window(
333333
self,
334-
and_projection: bool = True,
335-
and_scissor: bool = True,
336-
and_position: bool = False,
334+
viewport: bool = True,
335+
projection: bool = True,
336+
scissor: bool = True,
337+
position: bool = False,
337338
aspect: float | None = None,
338339
) -> None:
339340
"""
340-
Sets the viewport to the size of the screen.
341-
Should be called when the screen is resized.
341+
Sets the viewport to the size of the window.
342+
Should be called when the window is resized.
342343
343344
Args:
344345
and_projection: Flag whether to also equalize the projection to the viewport.
@@ -353,96 +354,102 @@ def match_screen(
353354
compared to the height. i.e. for an aspect ratio of ``4:3`` you should
354355
input ``4.0/3.0`` or ``1.33333...``. Cannot be equal to zero.
355356
"""
356-
self.update_viewport(
357+
self.update_values(
357358
self._window.rect,
358-
and_projection=and_projection,
359-
and_scissor=and_scissor,
360-
and_position=and_position,
359+
viewport=viewport,
360+
projection=projection,
361+
scissor=scissor,
362+
position=position,
361363
aspect=aspect,
362364
)
363365

364366
def match_target(
365367
self,
366-
and_projection: bool = True,
367-
and_scissor: bool = True,
368-
and_position: bool = False,
368+
viewport: bool = True,
369+
projection: bool = True,
370+
scissor: bool = True,
371+
position: bool = False,
369372
aspect: float | None = None,
370373
) -> None:
371374
"""
372375
Sets the viewport to the size of the Camera2D's render target.
373376
374377
Args:
375-
and_projection: Flag whether to also equalize the projection to the viewport.
376-
On by default
377-
and_scissor: Flag whether to also equalize the scissor box to the viewport.
378-
On by default
379-
and_position: Flag whether to also center the camera to the viewport.
378+
viewport: Flag whether to equalise the viewport to the area of the render target
379+
projection: Flag whether to equalise the size of the projection to
380+
match the render target
381+
The projection center stays fixed, and the new projection matches only in size.
382+
scissor: Flag whether to update the scissor value.
383+
position: Flag whether to also center the camera to the value.
380384
Off by default
381-
aspect_ratio: The ratio between width and height that the viewport should
382-
be constrained to. If unset then the viewport just matches the window
383-
size. The aspect ratio describes how much larger the width should be
384-
compared to the height. i.e. for an aspect ratio of ``4:3`` you should
385+
aspect_ratio: The ratio between width and height that the value should
386+
be constrained to. i.e. for an aspect ratio of ``4:3`` you should
385387
input ``4.0/3.0`` or ``1.33333...``. Cannot be equal to zero.
388+
If unset then the value will not be updated.
386389
Raises:
387390
ValueError: Will be raised if the Camera2D was has no render target.
388391
"""
389392
if self.render_target is None:
390393
raise ValueError(
391-
"Tried to match a non-exsistant render target. Please use `match_screen` instead"
394+
"Tried to match a non-exsistant render target. Please use `match_window` instead"
392395
)
393396

394-
self.update_viewport(
397+
self.update_values(
395398
LRBT(*self.render_target.viewport),
396-
and_projection=and_projection,
397-
and_scissor=and_scissor,
398-
and_position=and_position,
399+
viewport,
400+
projection,
401+
scissor,
402+
position,
399403
aspect=aspect,
400404
)
401405

402-
def update_viewport(
406+
def update_values(
403407
self,
404-
new_viewport: Rect,
405-
and_projection: bool = True,
406-
and_scissor: bool = True,
407-
and_position: bool = False,
408+
value: Rect,
409+
viewport: bool = True,
410+
projection: bool = True,
411+
scissor: bool = True,
412+
position: bool = False,
408413
aspect: float | None = None,
409414
):
410415
"""
411-
Convienence method for updating the viewport of the camera. To simply change
412-
the viewport you can safely set the projection property.
416+
Convienence method for updating the viewport, projection, position
417+
and a few others with the same value.
413418
414419
Args:
415-
and_projection: Flag whether to also equalize the projection to the viewport.
416-
On by default
417-
and_scissor: Flag whether to also equalize the scissor box to the viewport.
418-
On by default
419-
and_position: Flag whether to also center the camera to the viewport.
420+
value: The rect that the values will be derived from.
421+
viewport: Flag whether to equalise the viewport to the value.
422+
projection: Flag whether to equalise the size of the projection to match the value.
423+
The projection center stays fixed, and the new projection matches only in size.
424+
scissor: Flag whether to update the scissor value.
425+
position: Flag whether to also center the camera to the value.
420426
Off by default
421-
aspect_ratio: The ratio between width and height that the viewport should
422-
be constrained to. If unset then the viewport just matches the window
423-
size. The aspect ratio describes how much larger the width should be
424-
compared to the height. i.e. for an aspect ratio of ``4:3`` you should
427+
aspect_ratio: The ratio between width and height that the value should
428+
be constrained to. i.e. for an aspect ratio of ``4:3`` you should
425429
input ``4.0/3.0`` or ``1.33333...``. Cannot be equal to zero.
430+
If unset then the value will not be updated.
426431
"""
427432
if aspect is not None:
428-
if new_viewport.height * aspect < new_viewport.width:
429-
w = new_viewport.height * aspect
430-
h = new_viewport.height
433+
if value.height * aspect < value.width:
434+
w = value.height * aspect
435+
h = value.height
431436
else:
432-
w = new_viewport.width
433-
h = new_viewport.width / aspect
434-
self.viewport = XYWH(new_viewport.x, new_viewport.y, w, h)
435-
else:
436-
self.viewport = new_viewport
437+
w = value.width
438+
h = value.width / aspect
439+
value = XYWH(value.x, value.y, w, h)
440+
441+
if viewport:
442+
self.viewport = value
437443

438-
if and_projection:
439-
self.equalise()
444+
if projection:
445+
x, y = self._projection_data.rect.x, self._projection_data.rect.y
446+
self._projection_data.rect = XYWH(x, y, value.width, value.height)
440447

441-
if and_scissor and self.scissor:
442-
self.scissor = self.viewport
448+
if scissor and self.scissor:
449+
self.scissor = value
443450

444-
if and_position:
445-
self.position = self.viewport.center
451+
if position:
452+
self.position = value.center
446453

447454
def aabb(self) -> Rect:
448455
"""
@@ -898,7 +905,6 @@ def top_left(self, new_corner: Point2):
898905
# top_center
899906
@property
900907
def top_center(self) -> Vec2:
901-
# TODO correct
902908
"""Get the top most position the camera can see"""
903909
pos = self.position
904910

@@ -908,7 +914,6 @@ def top_center(self) -> Vec2:
908914

909915
@top_center.setter
910916
def top_center(self, new_top: Point2):
911-
# TODO correct
912917
ux, uy, *_ = self._camera_data.up
913918
top = self.top
914919

arcade/examples/background_blending.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -128,7 +128,7 @@ def on_key_release(self, symbol: int, modifiers: int):
128128

129129
def on_resize(self, width: int, height: int):
130130
super().on_resize(width, height)
131-
self.camera.match_screen(and_projection=True)
131+
self.camera.match_window()
132132

133133
# This is to ensure the background covers the entire screen.
134134
self.background_1.size = (width, height)

arcade/examples/background_groups.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -130,7 +130,7 @@ def on_key_release(self, symbol: int, modifiers: int):
130130

131131
def on_resize(self, width: int, height: int):
132132
super().on_resize(width, height)
133-
self.camera.match_screen(and_projection=True)
133+
self.camera.match_window()
134134

135135

136136
def main():

arcade/examples/background_parallax.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -152,7 +152,7 @@ def on_key_release(self, symbol: int, modifiers: int):
152152

153153
def on_resize(self, width: int, height: int):
154154
super().on_resize(width, height)
155-
self.camera.match_screen(and_projection=True)
155+
self.camera.match_window()
156156
full_width_size = (width, SCALED_BG_LAYER_HEIGHT_PX)
157157

158158
# We can iterate through a background group,

arcade/examples/background_scrolling.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -98,7 +98,7 @@ def on_key_release(self, symbol: int, modifiers: int):
9898

9999
def on_resize(self, width: int, height: int):
100100
super().on_resize(width, height)
101-
self.camera.match_screen(and_projection=True)
101+
self.camera.match_window()
102102

103103
# This is to ensure the background covers the entire screen.
104104
self.background.size = (width, height)

arcade/examples/background_stationary.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -102,7 +102,7 @@ def on_key_release(self, symbol: int, modifiers: int):
102102

103103
def on_resize(self, width: int, height: int):
104104
super().on_resize(width, height)
105-
self.camera.match_screen(and_projection=True)
105+
self.camera.match_window()
106106

107107

108108
def main():

arcade/examples/camera_platform.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -163,7 +163,7 @@ def setup(self):
163163
def on_resize(self, width, height):
164164
"""Resize window"""
165165
super().on_resize(width, height)
166-
self.camera.match_screen(and_projection=True)
166+
self.camera.match_window()
167167

168168
def on_draw(self):
169169
"""Render the screen."""

arcade/examples/gl/custom_sprite.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -150,7 +150,7 @@ def __init__(self):
150150

151151
def on_resize(self, width: int, height: int):
152152
super().on_resize(width, height)
153-
self.camera.match_screen(and_position=True)
153+
self.camera.match_window(position=True)
154154

155155
def on_draw(self):
156156
self.clear()

arcade/examples/light_demo.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -225,7 +225,7 @@ def on_draw(self):
225225
def on_resize(self, width, height):
226226
""" User resizes the screen. """
227227
super().on_resize(width, height)
228-
self.camera.match_screen()
228+
self.camera.match_window()
229229

230230
# --- Light related ---
231231
# We need to resize the light layer to

arcade/examples/minimap_camera.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -222,8 +222,8 @@ def on_resize(self, width: int, height: int):
222222
Handle the user grabbing the edge and resizing the window.
223223
"""
224224
super().on_resize(width, height)
225-
self.camera_sprites.match_screen()
226-
self.camera_gui.match_screen(and_position=True)
225+
self.camera_sprites.match_window()
226+
self.camera_gui.match_window(position=True)
227227
self.camera_minimap.viewport = arcade.LBWH(
228228
width - self.camera_minimap.viewport_width,
229229
height - self.camera_minimap.viewport_height,

arcade/examples/minimap_texture.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -203,8 +203,8 @@ def on_resize(self, width: int, height: int):
203203
Handle the user grabbing the edge and resizing the window.
204204
"""
205205
super().on_resize(width, height)
206-
self.camera_sprites.match_screen(and_projection=True)
207-
self.camera_gui.match_screen(and_projection=True)
206+
self.camera_sprites.match_window()
207+
self.camera_gui.match_window()
208208

209209

210210
def main():

arcade/examples/procedural_caves_cellular.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -328,8 +328,8 @@ def on_resize(self, width: int, height: int):
328328
Handle the user grabbing the edge and resizing the window.
329329
"""
330330
super().on_resize(width, height)
331-
self.camera_sprites.match_screen(and_projection=True)
332-
self.camera_gui.match_screen(and_projection=True)
331+
self.camera_sprites.match_window()
332+
self.camera_gui.match_window()
333333

334334
def on_update(self, delta_time):
335335
""" Movement and game logic """

arcade/examples/sprite_move_scrolling.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -181,8 +181,8 @@ def on_resize(self, width: int, height: int):
181181
Handle the user grabbing the edge and resizing the window.
182182
"""
183183
super().on_resize(width, height)
184-
self.camera_sprites.match_screen(and_projection=True)
185-
self.camera_gui.match_screen(and_projection=True)
184+
self.camera_sprites.match_window()
185+
self.camera_gui.match_window()
186186

187187

188188
def main():

arcade/examples/sprite_move_scrolling_box.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -211,8 +211,8 @@ def on_resize(self, width: int, height: int):
211211
Handle the user grabbing the edge and resizing the window.
212212
"""
213213
super().on_resize(width, height)
214-
self.camera_sprites.match_screen(and_projection=True)
215-
self.camera_gui.match_screen(and_projection=True, and_position=True)
214+
self.camera_sprites.match_window()
215+
self.camera_gui.match_window(position=True)
216216

217217

218218
def main():

arcade/examples/sprite_move_scrolling_shake.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -196,8 +196,8 @@ def on_resize(self, width: int, height: int):
196196
Handle the user grabbing the edge and resizing the window.
197197
"""
198198
super().on_resize(width, height)
199-
self.camera_sprites.match_screen(and_projection=True)
200-
self.camera_gui.match_screen(and_projection=True, and_position=True)
199+
self.camera_sprites.match_window()
200+
self.camera_gui.match_window(position=True)
201201

202202

203203
def main():

arcade/examples/template_platformer.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -216,9 +216,9 @@ def on_resize(self, width: int, height: int):
216216
""" Resize window """
217217
super().on_resize(width, height)
218218
# Update the cameras to match the new window size
219-
self.camera_sprites.match_screen(and_projection=True)
220-
# The and_position property keeps `0, 0` in the bottom left corner.
221-
self.camera_gui.match_screen(and_projection=True, and_position=True)
219+
self.camera_sprites.match_window()
220+
# The position argument keeps `0, 0` in the bottom left corner.
221+
self.camera_gui.match_window(position=True)
222222

223223

224224
def main():

arcade/gui/ui_manager.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -432,12 +432,12 @@ def on_resize(self, width, height):
432432
"""Resize the UIManager and all of its surfaces."""
433433
# resize ui camera
434434
bottom_left = self.camera.bottom_left
435-
self.camera.match_screen()
435+
self.camera.match_window()
436436
self.camera.bottom_left = bottom_left
437437

438438
# resize render to surface camera
439439
bottom_left = self._render_to_surface_camera.bottom_left
440-
self._render_to_surface_camera.match_screen()
440+
self._render_to_surface_camera.match_window()
441441
self._render_to_surface_camera.bottom_left = bottom_left
442442

443443
scale = self.window.get_pixel_ratio()

0 commit comments

Comments
 (0)