myou-engine/src/screen.nim

248 lines
10 KiB
Nim

# The contents of this file are subject to the Common Public Attribution License
# Version 1.0 (the “License”); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
# https://myou.dev/licenses/LICENSE-CPAL. The License is based on the Mozilla
# Public License Version 1.1 but Sections 14 and 15 have been added to cover use
# of software over a computer network and provide for limited attribution for
# the Original Developer. In addition, Exhibit A has been modified to be
# consistent with Exhibit B.
#
# Software distributed under the License is distributed on an “AS IS” basis,
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
# the specific language governing rights and limitations under the License.
#
# The Original Code is Myou Engine.
#
# the Original Developer is the Initial Developer.
#
# The Initial Developer of the Original Code is the Myou Engine developers.
# All portions of the code written by the Myou Engine developers are Copyright
# (c) 2024. All Rights Reserved.
#
# Alternatively, the contents of this file may be used under the terms of the
# GNU Affero General Public License version 3 (the [AGPL-3] License), in which
# case the provisions of [AGPL-3] License are applicable instead of those above.
#
# If you wish to allow use of your version of this file only under the terms of
# the [AGPL-3] License and not to allow others to use your version of this file
# under the CPAL, indicate your decision by deleting the provisions above and
# replace them with the notice and other provisions required by the [AGPL-3]
# License. If you do not delete the provisions above, a recipient may use your
# version of this file under either the CPAL or the [AGPL-3] License.
import ./types
# Forward declarations
proc newScreen*(engine: MyouEngine, width, height: int32, title: string): Screen
proc destroy*(self: Screen)
proc resize*(self: Screen, width, height: int32, orientation = self.orientation)
proc add_viewport*(self: Screen, camera: Camera)
proc clear_all_callbacks*(self: Screen)
proc emulateMouseWithTouch*(screen: Screen, touch: int32, ending: bool, x, y: float32)
# End forward declarations
import std/bitops
import std/sequtils
import std/math
import vmath except Quat, quat
import ./graphics/framebuffer
import ./objects/camera
import ./platform/platform
import ./input
import ./util
proc newScreen*(engine: MyouEngine, width, height: int32, title: string): Screen =
## Creates a new screen or window. The first one is created by the engine for you.
result = new Screen
result.engine = engine
result.width = width
result.height = height
# note: the window will overwrite width and height in some platforms
let window = make_window(width, height, title)
window.screen = result
result.window = cast[pointer](window)
result.framebuffer = engine.newMainFramebuffer()
result.enabled = true
engine.screens.add result
if engine.screen == nil:
engine.screen = result
result.resize(result.width, result.height)
result.pre_draw = proc(self: Screen) = discard
result.post_draw = proc(self: Screen) = discard
result.frame_interval = 1
result.display_scale = 1.0
proc destroy*(self: Screen) =
## Destroys the screen/window and all its resources. If you destroy the main
## screen, the first screen created after it will become the main screen. If
## there are no screens left, the engine will exit.
self.engine.screens.remove self
if self.engine.screen == self and self.engine.screens.len != 0:
self.engine.screen = self.engine.screens[0]
proc resize*(self: Screen, width, height: int32, orientation = self.orientation) =
## Resize the screen/window and all its viewports.
self.width = width
self.height = height
self.orientation = orientation
for vp in self.viewports:
let (x,y,w,h) = vp.rect
# TODO: verify this doesn't leave gaps/overlaps
vp.rect_pix = (
round(x*width.float32).int32,
round(y*height.float32).int32,
round(w*width.float32).int32,
round(h*height.float32).int32,
)
if height != 0:
vp.camera.aspect_ratio = width / height
vp.camera.update_projection()
for f in self.resize_callbacks:
f(self)
proc add_viewport*(self: Screen, camera: Camera) =
## Add a viewport to the screen with a given camera.
assert camera.scene != nil, "Camera needs to added to a scene first."
let vp = new Viewport
vp.camera = camera
vp.clear_color = true
vp.clear_depth = true
vp.rect = (0'f32, 0'f32, 1'f32, 1'f32)
self.viewports.add vp
self.resize(self.width, self.height)
proc `vsync=`*(self: Screen, vsync: bool) =
## Change the vsync setting of the screen/window.
cast[Window](self.window).set_vsync vsync
proc get_ray_direction*(viewport: Viewport, position: Vec2): Vec3 =
## Calculates a vector that points from the camera towards the given screen
## coordinates, in world space.
let x = (position.x - viewport.rect_pix[0].float32) / viewport.rect_pix[2].float32
let y = (position.y - viewport.rect_pix[1].float32) / viewport.rect_pix[3].float32
return viewport.camera.get_ray_direction(x,y)
proc get_ray_direction_local*(viewport: Viewport, position: Vec2): Vec3 =
## Calculates a vector that points from the camera towards the given screen
## coordinates, in camera space.
let x = (position.x - viewport.rect_pix[0].float32) / viewport.rect_pix[2].float32
let y = (position.y - viewport.rect_pix[1].float32) / viewport.rect_pix[3].float32
return viewport.camera.get_ray_direction_local(x,y)
proc get_pixels_at_depth*(viewport: Viewport, depth: float32): float32 =
## Returns the length of one world unit (usually one meter) in screen pixels,
## at a given depth.
# TODO: shift_x
let p = vec4(1.0, 0.0, -depth, 1.0)
let s = viewport.camera.projection_matrix * p
return viewport.rect_pix[2].float32 * 0.5 * s.x/s.w
proc clear_all_callbacks*(self: Screen) =
self.key_callbacks.setLen 0
self.mouse_button_callbacks.setLen 0
self.mouse_move_callbacks.setLen 0
proc size*(self: Screen): IVec2 {.inline.} = ivec2(self.width, self.height)
proc emulateMouseWithTouch*(screen: Screen, touch: int32, ending: bool, x, y: float32) =
template send_move(x, y: untyped) =
let buttons = screen.last_buttons
let e = MouseMoveEvent(
left: (buttons and 1).bool,
middle: (buttons and 2).bool,
right: (buttons and 4).bool,
# shiftKey: shiftKey, ctrlKey: ctrlKey, altKey: altKey, metaKey: metaKey,
position: vec2(x,y),
movement: vec2(x-screen.last_x, y-screen.last_y),
)
for cb in screen.mouse_move_callbacks:
cb(e)
if screen.break_current_callbacks:
screen.break_current_callbacks = false
break
screen.last_x = x
screen.last_y = y
template send_btn(pressed1, btn, x, y: untyped) =
let e = MouseButtonEvent(
pressed: pressed1,
button: cast[MouseButton](btn),
position: vec2(x,y)
)
for cb in screen.mouse_button_callbacks:
cb(e)
if screen.break_current_callbacks:
screen.break_current_callbacks = false
break
if pressed1:
screen.last_buttons = screen.last_buttons or (1'i8 shl btn)
else:
screen.last_buttons = screen.last_buttons and not (1'i8 shl btn)
screen.last_x = x
screen.last_y = y
template send_wheel(x,y: untyped) =
let e = MouseWheelEvent(
movement: vec2(x,y),
)
for cb in screen.mouse_wheel_callbacks:
cb(e)
if screen.break_current_callbacks:
screen.break_current_callbacks = false
break
let prev_count = screen.touches.len
var index = -1
for i,t in screen.touches.mpairs:
if t[0] == touch:
index = i
t[1] = vec2(x,y)
break
if not ending and index != -1: # move
if index == screen.touches.high:
# when we receive the last touch, we received them all, time to
# calculate mousemove and scroll wheel
var delta: Vec2
for i,(_,v) in screen.touches:
delta += v - screen.prev_touches[i][1]
if delta.lengthSq != 0.0:
delta /= screen.touches.len.float32
send_move(screen.last_x + delta.x, screen.last_y + delta.y)
# when we have more than one touch we calculate zoom and angle
# from a pinch gesture
if index != 0:
let v = screen.touches[0][1] -
screen.touches[^1][1]
let v0 = screen.prev_touches[0][1] -
screen.prev_touches[^1][1]
let zoom = v0.length/v.length
let angle = fixAngle(v.angle - v0.angle)
if zoom != 1.0 or angle != 0.0:
send_wheel(angle, -log(zoom, 1.2))
# finally, store the current touches to use next time
screen.prev_touches = screen.touches
elif not ending: # start
screen.touches.add (touch.int32, vec2(x,y))
let button = prev_count
if screen.last_buttons == 0:
send_btn(true, button, x, y)
else:
let previous = screen.last_buttons.countTrailingZeroBits
if previous < button:
send_btn(true, button, screen.last_x, screen.last_y)
send_btn(false, previous, screen.last_x, screen.last_y)
screen.prev_touches = screen.touches
else: # end
screen.touches.keepItIf it[0] != touch
if screen.touches.len == 0:
let previous = screen.last_buttons.countTrailingZeroBits
send_btn(false, previous, screen.last_x, screen.last_y)
screen.prev_touches = screen.touches
proc switch*(self: Screen): bool {.discardable.} =
self.platform_switch_screen()