mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-26 02:42:02 +01:00
internal/ui: freeze the input state for each frame
After this change, the input APIs will return more consistent results for one frame. Closes #2496
This commit is contained in:
parent
e1804eca64
commit
d1b9a0a9a1
@ -147,7 +147,9 @@ func (g *gameForUI) Layout(outsideWidth, outsideHeight float64) (float64, float6
|
|||||||
return float64(sw), float64(sh)
|
return float64(sw), float64(sh)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *gameForUI) Update() error {
|
func (g *gameForUI) Update(inputState ui.InputState) error {
|
||||||
|
theInputState.set(inputState)
|
||||||
|
|
||||||
if err := g.game.Update(); err != nil {
|
if err := g.game.Update(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -485,6 +485,7 @@ const (
|
|||||||
KeyReserved1
|
KeyReserved1
|
||||||
KeyReserved2
|
KeyReserved2
|
||||||
KeyReserved3
|
KeyReserved3
|
||||||
|
KeyMax = KeyReserved3
|
||||||
)
|
)
|
||||||
|
|
||||||
func (k Key) String() string {
|
func (k Key) String() string {
|
||||||
|
126
input.go
126
input.go
@ -15,6 +15,8 @@
|
|||||||
package ebiten
|
package ebiten
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"sync"
|
||||||
|
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/gamepaddb"
|
"github.com/hajimehoshi/ebiten/v2/internal/gamepaddb"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/ui"
|
"github.com/hajimehoshi/ebiten/v2/internal/ui"
|
||||||
@ -35,7 +37,7 @@ import (
|
|||||||
//
|
//
|
||||||
// Keyboards don't work on iOS yet (#1090).
|
// Keyboards don't work on iOS yet (#1090).
|
||||||
func AppendInputChars(runes []rune) []rune {
|
func AppendInputChars(runes []rune) []rune {
|
||||||
return ui.Get().Input().AppendInputChars(runes)
|
return theInputState.appendInputChars(runes)
|
||||||
}
|
}
|
||||||
|
|
||||||
// InputChars return "printable" runes read from the keyboard at the time update is called.
|
// InputChars return "printable" runes read from the keyboard at the time update is called.
|
||||||
@ -64,29 +66,7 @@ func InputChars() []rune {
|
|||||||
//
|
//
|
||||||
// Keyboards don't work on iOS yet (#1090).
|
// Keyboards don't work on iOS yet (#1090).
|
||||||
func IsKeyPressed(key Key) bool {
|
func IsKeyPressed(key Key) bool {
|
||||||
if !key.isValid() {
|
return theInputState.isKeyPressed(key)
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
var keys []ui.Key
|
|
||||||
switch key {
|
|
||||||
case KeyAlt:
|
|
||||||
keys = []ui.Key{ui.KeyAltLeft, ui.KeyAltRight}
|
|
||||||
case KeyControl:
|
|
||||||
keys = []ui.Key{ui.KeyControlLeft, ui.KeyControlRight}
|
|
||||||
case KeyShift:
|
|
||||||
keys = []ui.Key{ui.KeyShiftLeft, ui.KeyShiftRight}
|
|
||||||
case KeyMeta:
|
|
||||||
keys = []ui.Key{ui.KeyMetaLeft, ui.KeyMetaRight}
|
|
||||||
default:
|
|
||||||
keys = []ui.Key{ui.Key(key)}
|
|
||||||
}
|
|
||||||
for _, k := range keys {
|
|
||||||
if ui.Get().Input().IsKeyPressed(k) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// CursorPosition returns a position of a mouse cursor relative to the game screen (window). The cursor position is
|
// CursorPosition returns a position of a mouse cursor relative to the game screen (window). The cursor position is
|
||||||
@ -98,7 +78,7 @@ func IsKeyPressed(key Key) bool {
|
|||||||
//
|
//
|
||||||
// CursorPosition is concurrent-safe.
|
// CursorPosition is concurrent-safe.
|
||||||
func CursorPosition() (x, y int) {
|
func CursorPosition() (x, y int) {
|
||||||
return ui.Get().Input().CursorPosition()
|
return theInputState.cursorPosition()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wheel returns x and y offsets of the mouse wheel or touchpad scroll.
|
// Wheel returns x and y offsets of the mouse wheel or touchpad scroll.
|
||||||
@ -106,7 +86,7 @@ func CursorPosition() (x, y int) {
|
|||||||
//
|
//
|
||||||
// Wheel is concurrent-safe.
|
// Wheel is concurrent-safe.
|
||||||
func Wheel() (xoff, yoff float64) {
|
func Wheel() (xoff, yoff float64) {
|
||||||
return ui.Get().Input().Wheel()
|
return theInputState.wheel()
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsMouseButtonPressed returns a boolean indicating whether mouseButton is pressed.
|
// IsMouseButtonPressed returns a boolean indicating whether mouseButton is pressed.
|
||||||
@ -116,7 +96,7 @@ func Wheel() (xoff, yoff float64) {
|
|||||||
//
|
//
|
||||||
// IsMouseButtonPressed is concurrent-safe.
|
// IsMouseButtonPressed is concurrent-safe.
|
||||||
func IsMouseButtonPressed(mouseButton MouseButton) bool {
|
func IsMouseButtonPressed(mouseButton MouseButton) bool {
|
||||||
return ui.Get().Input().IsMouseButtonPressed(mouseButton)
|
return theInputState.isMouseButtonPressed(mouseButton)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GamepadID represents a gamepad's identifier.
|
// GamepadID represents a gamepad's identifier.
|
||||||
@ -377,7 +357,7 @@ type TouchID = ui.TouchID
|
|||||||
//
|
//
|
||||||
// AppendTouchIDs is concurrent-safe.
|
// AppendTouchIDs is concurrent-safe.
|
||||||
func AppendTouchIDs(touches []TouchID) []TouchID {
|
func AppendTouchIDs(touches []TouchID) []TouchID {
|
||||||
return ui.Get().Input().AppendTouchIDs(touches)
|
return theInputState.appendTouchIDs(touches)
|
||||||
}
|
}
|
||||||
|
|
||||||
// TouchIDs returns the current touch states.
|
// TouchIDs returns the current touch states.
|
||||||
@ -393,5 +373,93 @@ func TouchIDs() []TouchID {
|
|||||||
//
|
//
|
||||||
// TouchPosition is cuncurrent-safe.
|
// TouchPosition is cuncurrent-safe.
|
||||||
func TouchPosition(id TouchID) (int, int) {
|
func TouchPosition(id TouchID) (int, int) {
|
||||||
return ui.Get().Input().TouchPosition(id)
|
return theInputState.touchPosition(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
var theInputState inputState
|
||||||
|
|
||||||
|
type inputState struct {
|
||||||
|
state ui.InputState
|
||||||
|
m sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) set(inputState ui.InputState) {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
i.state = inputState
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) appendInputChars(runes []rune) []rune {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
return append(runes, i.state.Runes[:i.state.RunesCount]...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) isKeyPressed(key Key) bool {
|
||||||
|
if !key.isValid() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
|
||||||
|
switch key {
|
||||||
|
case KeyAlt:
|
||||||
|
return i.state.KeyPressed[ui.KeyAltLeft] && i.state.KeyPressed[ui.KeyAltRight]
|
||||||
|
case KeyControl:
|
||||||
|
return i.state.KeyPressed[ui.KeyControlLeft] && i.state.KeyPressed[ui.KeyControlRight]
|
||||||
|
case KeyShift:
|
||||||
|
return i.state.KeyPressed[ui.KeyShiftLeft] && i.state.KeyPressed[ui.KeyShiftRight]
|
||||||
|
case KeyMeta:
|
||||||
|
return i.state.KeyPressed[ui.KeyMetaLeft] && i.state.KeyPressed[ui.KeyMetaRight]
|
||||||
|
default:
|
||||||
|
return i.state.KeyPressed[ui.Key(key)]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) cursorPosition() (int, int) {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
return i.state.CursorX, i.state.CursorY
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) wheel() (float64, float64) {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
return i.state.WheelX, i.state.WheelY
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) isMouseButtonPressed(mouseButton MouseButton) bool {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
return i.state.MouseButtonPressed[mouseButton]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) appendTouchIDs(touches []TouchID) []TouchID {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
|
||||||
|
for _, t := range i.state.Touches {
|
||||||
|
if !t.Valid {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
touches = append(touches, t.ID)
|
||||||
|
}
|
||||||
|
return touches
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *inputState) touchPosition(id TouchID) (int, int) {
|
||||||
|
i.m.Lock()
|
||||||
|
defer i.m.Unlock()
|
||||||
|
|
||||||
|
for _, t := range i.state.Touches {
|
||||||
|
if !t.Valid {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if id != t.ID {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return t.X, t.Y
|
||||||
|
}
|
||||||
|
return 0, 0
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,7 @@ type Game interface {
|
|||||||
NewOffscreenImage(width, height int) *Image
|
NewOffscreenImage(width, height int) *Image
|
||||||
NewScreenImage(width, height int) *Image
|
NewScreenImage(width, height int) *Image
|
||||||
Layout(outsideWidth, outsideHeight float64) (screenWidth, screenHeight float64)
|
Layout(outsideWidth, outsideHeight float64) (screenWidth, screenHeight float64)
|
||||||
Update() error
|
Update(InputState) error
|
||||||
DrawOffscreen() error
|
DrawOffscreen() error
|
||||||
DrawFinalScreen(scale, offsetX, offsetY float64)
|
DrawFinalScreen(scale, offsetX, offsetY float64)
|
||||||
}
|
}
|
||||||
@ -89,7 +89,9 @@ func (c *context) updateFrameImpl(graphicsDriver graphicsdriver.Graphics, update
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
ui.beginFrame()
|
// Read the input state and use it for one frame to give a consistent result for one frame (#2496).
|
||||||
|
var inputState InputState
|
||||||
|
ui.beginFrame(&inputState)
|
||||||
defer ui.endFrame()
|
defer ui.endFrame()
|
||||||
|
|
||||||
// The given outside size can be 0 e.g. just after restoring from the fullscreen mode on Windows (#1589)
|
// The given outside size can be 0 e.g. just after restoring from the fullscreen mode on Windows (#1589)
|
||||||
@ -126,7 +128,7 @@ func (c *context) updateFrameImpl(graphicsDriver graphicsdriver.Graphics, update
|
|||||||
if err := hooks.RunBeforeUpdateHooks(); err != nil {
|
if err := hooks.RunBeforeUpdateHooks(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err := c.game.Update(); err != nil {
|
if err := c.game.Update(inputState); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Catch the error that happened at (*Image).At.
|
// Catch the error that happened at (*Image).At.
|
||||||
|
69
internal/ui/input.go
Normal file
69
internal/ui/input.go
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
// Copyright 2022 The Ebitengine Authors
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package ui
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unicode"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MouseButton int
|
||||||
|
|
||||||
|
const (
|
||||||
|
MouseButton0 MouseButton = iota // The 'left' button
|
||||||
|
MouseButton1 // The 'right' button
|
||||||
|
MouseButton2 // The 'middle' button
|
||||||
|
MouseButton3 // The additional button (usually browser-back)
|
||||||
|
MouseButton4 // The additional button (usually browser-forward)
|
||||||
|
MouseButtonMax = MouseButton4
|
||||||
|
)
|
||||||
|
|
||||||
|
type TouchID int
|
||||||
|
|
||||||
|
type Touch struct {
|
||||||
|
Valid bool
|
||||||
|
ID TouchID
|
||||||
|
X int
|
||||||
|
Y int
|
||||||
|
}
|
||||||
|
|
||||||
|
type InputState struct {
|
||||||
|
KeyPressed [KeyMax + 1]bool
|
||||||
|
MouseButtonPressed [MouseButtonMax + 1]bool
|
||||||
|
CursorX int
|
||||||
|
CursorY int
|
||||||
|
WheelX float64
|
||||||
|
WheelY float64
|
||||||
|
Touches [16]Touch
|
||||||
|
Runes [16]rune
|
||||||
|
RunesCount int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InputState) resetForFrame() {
|
||||||
|
i.WheelX = 0
|
||||||
|
i.WheelY = 0
|
||||||
|
i.RunesCount = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InputState) appendRune(r rune) {
|
||||||
|
if !unicode.IsPrint(r) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if i.RunesCount >= len(i.Runes) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
i.Runes[i.RunesCount] = r
|
||||||
|
i.RunesCount++
|
||||||
|
}
|
@ -18,136 +18,11 @@ package ui
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
"sync"
|
|
||||||
"unicode"
|
|
||||||
|
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/glfw"
|
"github.com/hajimehoshi/ebiten/v2/internal/glfw"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Input struct {
|
|
||||||
keyPressed map[glfw.Key]bool
|
|
||||||
mouseButtonPressed map[glfw.MouseButton]bool
|
|
||||||
onceCallback sync.Once
|
|
||||||
scrollX float64
|
|
||||||
scrollY float64
|
|
||||||
cursorX int
|
|
||||||
cursorY int
|
|
||||||
touches map[TouchID]pos // TODO: Implement this (#417)
|
|
||||||
runeBuffer []rune
|
|
||||||
ui *userInterfaceImpl
|
|
||||||
}
|
|
||||||
|
|
||||||
type pos struct {
|
|
||||||
X int
|
|
||||||
Y int
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) CursorPosition() (x, y int) {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
return i.cursorX, i.cursorY
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) AppendTouchIDs(touchIDs []TouchID) []TouchID {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
for id := range i.touches {
|
|
||||||
touchIDs = append(touchIDs, id)
|
|
||||||
}
|
|
||||||
return touchIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) TouchPosition(id TouchID) (x, y int) {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
for tid, pos := range i.touches {
|
|
||||||
if id == tid {
|
|
||||||
return pos.X, pos.Y
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsKeyPressed(key Key) bool {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
|
|
||||||
gk, ok := uiKeyToGLFWKey[key]
|
|
||||||
if !ok {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return i.keyPressed[gk]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) AppendInputChars(runes []rune) []rune {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
return append(runes, i.runeBuffer...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) resetForTick() {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
i.runeBuffer = i.runeBuffer[:0]
|
|
||||||
i.scrollX, i.scrollY = 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsMouseButtonPressed(button MouseButton) bool {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
if i.mouseButtonPressed == nil {
|
|
||||||
i.mouseButtonPressed = map[glfw.MouseButton]bool{}
|
|
||||||
}
|
|
||||||
for gb, b := range glfwMouseButtonToMouseButton {
|
|
||||||
if b != button {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if i.mouseButtonPressed[gb] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) Wheel() (xoff, yoff float64) {
|
|
||||||
if !i.ui.isRunning() {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
return i.scrollX, i.scrollY
|
|
||||||
}
|
|
||||||
|
|
||||||
var glfwMouseButtonToMouseButton = map[glfw.MouseButton]MouseButton{
|
var glfwMouseButtonToMouseButton = map[glfw.MouseButton]MouseButton{
|
||||||
glfw.MouseButtonLeft: MouseButton0,
|
glfw.MouseButtonLeft: MouseButton0,
|
||||||
glfw.MouseButtonMiddle: MouseButton1,
|
glfw.MouseButtonMiddle: MouseButton1,
|
||||||
@ -156,53 +31,44 @@ var glfwMouseButtonToMouseButton = map[glfw.MouseButton]MouseButton{
|
|||||||
glfw.MouseButton4: MouseButton4,
|
glfw.MouseButton4: MouseButton4,
|
||||||
}
|
}
|
||||||
|
|
||||||
// update must be called from the main thread.
|
func (u *userInterfaceImpl) registerInputCallbacks() {
|
||||||
func (i *Input) update(window *glfw.Window, context *context) error {
|
u.window.SetCharModsCallback(glfw.ToCharModsCallback(func(w *glfw.Window, char rune, mods glfw.ModifierKey) {
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
|
|
||||||
i.onceCallback.Do(func() {
|
|
||||||
window.SetCharModsCallback(glfw.ToCharModsCallback(func(w *glfw.Window, char rune, mods glfw.ModifierKey) {
|
|
||||||
// As this function is called from GLFW callbacks, the current thread is main.
|
// As this function is called from GLFW callbacks, the current thread is main.
|
||||||
if !unicode.IsPrint(char) {
|
u.m.Lock()
|
||||||
return
|
defer u.m.Unlock()
|
||||||
}
|
u.inputState.appendRune(char)
|
||||||
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
i.runeBuffer = append(i.runeBuffer, char)
|
|
||||||
}))
|
}))
|
||||||
window.SetScrollCallback(glfw.ToScrollCallback(func(w *glfw.Window, xoff float64, yoff float64) {
|
u.window.SetScrollCallback(glfw.ToScrollCallback(func(w *glfw.Window, xoff float64, yoff float64) {
|
||||||
// As this function is called from GLFW callbacks, the current thread is main.
|
// As this function is called from GLFW callbacks, the current thread is main.
|
||||||
i.ui.m.Lock()
|
u.m.Lock()
|
||||||
defer i.ui.m.Unlock()
|
defer u.m.Unlock()
|
||||||
i.scrollX += xoff
|
u.inputState.WheelX += xoff
|
||||||
i.scrollY += yoff
|
u.inputState.WheelY += yoff
|
||||||
}))
|
}))
|
||||||
})
|
}
|
||||||
if i.keyPressed == nil {
|
|
||||||
i.keyPressed = map[glfw.Key]bool{}
|
// updateInput must be called from the main thread.
|
||||||
|
func (u *userInterfaceImpl) updateInputState() error {
|
||||||
|
u.m.Lock()
|
||||||
|
defer u.m.Unlock()
|
||||||
|
|
||||||
|
for uk, gk := range uiKeyToGLFWKey {
|
||||||
|
u.inputState.KeyPressed[uk] = u.window.GetKey(gk) == glfw.Press
|
||||||
}
|
}
|
||||||
for _, gk := range uiKeyToGLFWKey {
|
for gb, ub := range glfwMouseButtonToMouseButton {
|
||||||
i.keyPressed[gk] = window.GetKey(gk) == glfw.Press
|
u.inputState.MouseButtonPressed[ub] = u.window.GetMouseButton(gb) == glfw.Press
|
||||||
}
|
}
|
||||||
if i.mouseButtonPressed == nil {
|
cx, cy := u.window.GetCursorPos()
|
||||||
i.mouseButtonPressed = map[glfw.MouseButton]bool{}
|
|
||||||
}
|
|
||||||
for gb := range glfwMouseButtonToMouseButton {
|
|
||||||
i.mouseButtonPressed[gb] = window.GetMouseButton(gb) == glfw.Press
|
|
||||||
}
|
|
||||||
cx, cy := window.GetCursorPos()
|
|
||||||
// TODO: This is tricky. Rename the function?
|
// TODO: This is tricky. Rename the function?
|
||||||
m := i.ui.currentMonitor()
|
m := u.currentMonitor()
|
||||||
s := i.ui.deviceScaleFactor(m)
|
s := u.deviceScaleFactor(m)
|
||||||
cx = i.ui.dipFromGLFWPixel(cx, m)
|
cx = u.dipFromGLFWPixel(cx, m)
|
||||||
cy = i.ui.dipFromGLFWPixel(cy, m)
|
cy = u.dipFromGLFWPixel(cy, m)
|
||||||
cx, cy = context.adjustPosition(cx, cy, s)
|
cx, cy = u.context.adjustPosition(cx, cy, s)
|
||||||
|
|
||||||
// AdjustPosition can return NaN at the initialization.
|
// AdjustPosition can return NaN at the initialization.
|
||||||
if !math.IsNaN(cx) && !math.IsNaN(cy) {
|
if !math.IsNaN(cx) && !math.IsNaN(cy) {
|
||||||
i.cursorX, i.cursorY = int(cx), int(cy)
|
u.inputState.CursorX, u.inputState.CursorY = int(cx), int(cy)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := gamepad.Update(); err != nil {
|
if err := gamepad.Update(); err != nil {
|
||||||
|
@ -31,89 +31,17 @@ var (
|
|||||||
stringTouchmove = js.ValueOf("touchmove")
|
stringTouchmove = js.ValueOf("touchmove")
|
||||||
)
|
)
|
||||||
|
|
||||||
var jsKeys []js.Value
|
func jsKeyToID(key js.Value) Key {
|
||||||
|
|
||||||
func init() {
|
|
||||||
for _, k := range uiKeyToJSKey {
|
|
||||||
jsKeys = append(jsKeys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func jsKeyToID(key js.Value) int {
|
|
||||||
// js.Value cannot be used as a map key.
|
// js.Value cannot be used as a map key.
|
||||||
// As the number of keys is around 100, just a dumb loop should work.
|
// As the number of keys is around 100, just a dumb loop should work.
|
||||||
for i, k := range jsKeys {
|
for uiKey, jsKey := range uiKeyToJSKey {
|
||||||
if k.Equal(key) {
|
if jsKey.Equal(key) {
|
||||||
return i
|
return uiKey
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
|
|
||||||
type pos struct {
|
|
||||||
X int
|
|
||||||
Y int
|
|
||||||
}
|
|
||||||
|
|
||||||
type Input struct {
|
|
||||||
keyPressed map[int]bool
|
|
||||||
mouseButtonPressed map[int]bool
|
|
||||||
cursorX int
|
|
||||||
cursorY int
|
|
||||||
origCursorX int
|
|
||||||
origCursorY int
|
|
||||||
wheelX float64
|
|
||||||
wheelY float64
|
|
||||||
touches map[TouchID]pos
|
|
||||||
runeBuffer []rune
|
|
||||||
ui *userInterfaceImpl
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) CursorPosition() (x, y int) {
|
|
||||||
if i.ui.context == nil {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
xf, yf := i.ui.context.adjustPosition(float64(i.cursorX), float64(i.cursorY), i.ui.DeviceScaleFactor())
|
|
||||||
return int(xf), int(yf)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) AppendTouchIDs(touchIDs []TouchID) []TouchID {
|
|
||||||
for id := range i.touches {
|
|
||||||
touchIDs = append(touchIDs, id)
|
|
||||||
}
|
|
||||||
return touchIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) TouchPosition(id TouchID) (x, y int) {
|
|
||||||
d := i.ui.DeviceScaleFactor()
|
|
||||||
for tid, pos := range i.touches {
|
|
||||||
if id == tid {
|
|
||||||
x, y := i.ui.context.adjustPosition(float64(pos.X), float64(pos.Y), d)
|
|
||||||
return int(x), int(y)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) AppendInputChars(runes []rune) []rune {
|
|
||||||
return append(runes, i.runeBuffer...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) resetForTick() {
|
|
||||||
i.runeBuffer = nil
|
|
||||||
i.wheelX = 0
|
|
||||||
i.wheelY = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsKeyPressed(key Key) bool {
|
|
||||||
if i.keyPressed != nil {
|
|
||||||
if i.keyPressed[jsKeyToID(uiKeyToJSKey[key])] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
var codeToMouseButton = map[int]MouseButton{
|
var codeToMouseButton = map[int]MouseButton{
|
||||||
0: MouseButton0, // Left
|
0: MouseButton0, // Left
|
||||||
1: MouseButton1, // Middle
|
1: MouseButton1, // Middle
|
||||||
@ -122,123 +50,92 @@ var codeToMouseButton = map[int]MouseButton{
|
|||||||
4: MouseButton4,
|
4: MouseButton4,
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) IsMouseButtonPressed(button MouseButton) bool {
|
func (u *userInterfaceImpl) keyDown(code js.Value) {
|
||||||
if i.mouseButtonPressed == nil {
|
u.inputState.KeyPressed[jsKeyToID(code)] = true
|
||||||
i.mouseButtonPressed = map[int]bool{}
|
|
||||||
}
|
|
||||||
for c, b := range codeToMouseButton {
|
|
||||||
if b != button {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if i.mouseButtonPressed[c] {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) Wheel() (xoff, yoff float64) {
|
func (u *userInterfaceImpl) keyUp(code js.Value) {
|
||||||
return i.wheelX, i.wheelY
|
u.inputState.KeyPressed[jsKeyToID(code)] = false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) keyDown(code js.Value) {
|
func (u *userInterfaceImpl) mouseDown(code int) {
|
||||||
if i.keyPressed == nil {
|
u.inputState.MouseButtonPressed[codeToMouseButton[code]] = true
|
||||||
i.keyPressed = map[int]bool{}
|
|
||||||
}
|
|
||||||
i.keyPressed[jsKeyToID(code)] = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) keyUp(code js.Value) {
|
func (u *userInterfaceImpl) mouseUp(code int) {
|
||||||
if i.keyPressed == nil {
|
u.inputState.MouseButtonPressed[codeToMouseButton[code]] = false
|
||||||
i.keyPressed = map[int]bool{}
|
|
||||||
}
|
|
||||||
i.keyPressed[jsKeyToID(code)] = false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) mouseDown(code int) {
|
func (u *userInterfaceImpl) updateInputFromEvent(e js.Value) error {
|
||||||
if i.mouseButtonPressed == nil {
|
|
||||||
i.mouseButtonPressed = map[int]bool{}
|
|
||||||
}
|
|
||||||
i.mouseButtonPressed[code] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) mouseUp(code int) {
|
|
||||||
if i.mouseButtonPressed == nil {
|
|
||||||
i.mouseButtonPressed = map[int]bool{}
|
|
||||||
}
|
|
||||||
i.mouseButtonPressed[code] = false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) updateFromEvent(e js.Value) error {
|
|
||||||
// Avoid using js.Value.String() as String creates a Uint8Array via a TextEncoder and causes a heavy
|
// Avoid using js.Value.String() as String creates a Uint8Array via a TextEncoder and causes a heavy
|
||||||
// overhead (#1437).
|
// overhead (#1437).
|
||||||
switch t := e.Get("type"); {
|
switch t := e.Get("type"); {
|
||||||
case t.Equal(stringKeydown):
|
case t.Equal(stringKeydown):
|
||||||
if str := e.Get("key").String(); isKeyString(str) {
|
if str := e.Get("key").String(); isKeyString(str) {
|
||||||
for _, r := range str {
|
for _, r := range str {
|
||||||
if unicode.IsPrint(r) {
|
u.inputState.appendRune(r)
|
||||||
i.runeBuffer = append(i.runeBuffer, r)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
u.keyDown(e.Get("code"))
|
||||||
i.keyDown(e.Get("code"))
|
|
||||||
case t.Equal(stringKeyup):
|
case t.Equal(stringKeyup):
|
||||||
i.keyUp(e.Get("code"))
|
u.keyUp(e.Get("code"))
|
||||||
case t.Equal(stringMousedown):
|
case t.Equal(stringMousedown):
|
||||||
button := e.Get("button").Int()
|
u.mouseDown(e.Get("button").Int())
|
||||||
i.mouseDown(button)
|
u.setMouseCursorFromEvent(e)
|
||||||
i.setMouseCursorFromEvent(e)
|
|
||||||
case t.Equal(stringMouseup):
|
case t.Equal(stringMouseup):
|
||||||
button := e.Get("button").Int()
|
u.mouseUp(e.Get("button").Int())
|
||||||
i.mouseUp(button)
|
u.setMouseCursorFromEvent(e)
|
||||||
i.setMouseCursorFromEvent(e)
|
|
||||||
case t.Equal(stringMousemove):
|
case t.Equal(stringMousemove):
|
||||||
i.setMouseCursorFromEvent(e)
|
u.setMouseCursorFromEvent(e)
|
||||||
case t.Equal(stringWheel):
|
case t.Equal(stringWheel):
|
||||||
// TODO: What if e.deltaMode is not DOM_DELTA_PIXEL?
|
// TODO: What if e.deltaMode is not DOM_DELTA_PIXEL?
|
||||||
i.wheelX = -e.Get("deltaX").Float()
|
u.inputState.WheelX = -e.Get("deltaX").Float()
|
||||||
i.wheelY = -e.Get("deltaY").Float()
|
u.inputState.WheelY = -e.Get("deltaY").Float()
|
||||||
case t.Equal(stringTouchstart) || t.Equal(stringTouchend) || t.Equal(stringTouchmove):
|
case t.Equal(stringTouchstart) || t.Equal(stringTouchend) || t.Equal(stringTouchmove):
|
||||||
i.updateTouchesFromEvent(e)
|
u.updateTouchesFromEvent(e)
|
||||||
}
|
}
|
||||||
|
|
||||||
i.ui.forceUpdateOnMinimumFPSMode()
|
u.forceUpdateOnMinimumFPSMode()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) setMouseCursorFromEvent(e js.Value) {
|
func (u *userInterfaceImpl) setMouseCursorFromEvent(e js.Value) {
|
||||||
if i.ui.cursorMode == CursorModeCaptured {
|
if u.context == nil {
|
||||||
x, y := e.Get("clientX").Int(), e.Get("clientY").Int()
|
|
||||||
i.origCursorX, i.origCursorY = x, y
|
|
||||||
dx, dy := e.Get("movementX").Int(), e.Get("movementY").Int()
|
|
||||||
i.cursorX += dx
|
|
||||||
i.cursorY += dy
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if u.cursorMode == CursorModeCaptured {
|
||||||
x, y := e.Get("clientX").Int(), e.Get("clientY").Int()
|
x, y := e.Get("clientX").Int(), e.Get("clientY").Int()
|
||||||
i.cursorX, i.cursorY = x, y
|
u.origCursorX, u.origCursorY = x, y
|
||||||
i.origCursorX, i.origCursorY = x, y
|
dx, dy := u.context.adjustPosition(e.Get("movementX").Float(), e.Get("movementY").Float(), u.DeviceScaleFactor())
|
||||||
|
u.inputState.CursorX += int(dx)
|
||||||
|
u.inputState.CursorY += int(dy)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
x, y := u.context.adjustPosition(e.Get("clientX").Float(), e.Get("clientY").Float(), u.DeviceScaleFactor())
|
||||||
|
u.inputState.CursorX, u.inputState.CursorY = int(x), int(y)
|
||||||
|
u.origCursorX, u.origCursorY = int(x), int(y)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) recoverCursorPosition() {
|
func (u *userInterfaceImpl) recoverCursorPosition() {
|
||||||
i.cursorX, i.cursorY = i.origCursorX, i.origCursorY
|
u.inputState.CursorX, u.inputState.CursorY = u.origCursorX, u.origCursorY
|
||||||
}
|
}
|
||||||
|
|
||||||
func (in *Input) updateTouchesFromEvent(e js.Value) {
|
func (u *userInterfaceImpl) updateTouchesFromEvent(e js.Value) {
|
||||||
j := e.Get("targetTouches")
|
for i := range u.inputState.Touches {
|
||||||
for k := range in.touches {
|
u.inputState.Touches[i].Valid = false
|
||||||
delete(in.touches, k)
|
|
||||||
}
|
}
|
||||||
for i := 0; i < j.Length(); i++ {
|
|
||||||
jj := j.Call("item", i)
|
touches := e.Get("targetTouches")
|
||||||
id := TouchID(jj.Get("identifier").Int())
|
for i := 0; i < touches.Length(); i++ {
|
||||||
if in.touches == nil {
|
t := touches.Call("item", i)
|
||||||
in.touches = map[TouchID]pos{}
|
x, y := u.context.adjustPosition(t.Get("clientX").Float(), t.Get("clientY").Float(), u.DeviceScaleFactor())
|
||||||
}
|
u.inputState.Touches[i] = Touch{
|
||||||
in.touches[id] = pos{
|
Valid: true,
|
||||||
X: jj.Get("clientX").Int(),
|
ID: TouchID(t.Get("identifier").Int()),
|
||||||
Y: jj.Get("clientY").Int(),
|
X: int(x),
|
||||||
|
Y: int(y),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,82 +16,38 @@
|
|||||||
|
|
||||||
package ui
|
package ui
|
||||||
|
|
||||||
type Input struct {
|
type TouchForInput struct {
|
||||||
keys map[Key]struct{}
|
ID TouchID
|
||||||
runes []rune
|
|
||||||
touches []Touch
|
// X is in device-independent pixels.
|
||||||
ui *userInterfaceImpl
|
X float64
|
||||||
|
|
||||||
|
// Y is in device-independent pixels.
|
||||||
|
Y float64
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Input) CursorPosition() (x, y int) {
|
func (u *userInterfaceImpl) updateInputState(keys map[Key]struct{}, runes []rune, touches []TouchForInput) {
|
||||||
return 0, 0
|
u.m.Lock()
|
||||||
}
|
defer u.m.Unlock()
|
||||||
|
|
||||||
func (i *Input) AppendTouchIDs(touchIDs []TouchID) []TouchID {
|
for k := range u.inputState.KeyPressed {
|
||||||
i.ui.m.RLock()
|
_, ok := keys[Key(k)]
|
||||||
defer i.ui.m.RUnlock()
|
u.inputState.KeyPressed[k] = ok
|
||||||
|
|
||||||
for _, t := range i.touches {
|
|
||||||
touchIDs = append(touchIDs, t.ID)
|
|
||||||
}
|
|
||||||
return touchIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) TouchPosition(id TouchID) (x, y int) {
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
|
|
||||||
for _, t := range i.touches {
|
|
||||||
if t.ID == id {
|
|
||||||
return i.ui.adjustPosition(t.X, t.Y)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) AppendInputChars(runes []rune) []rune {
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
return append(runes, i.runes...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsKeyPressed(key Key) bool {
|
|
||||||
i.ui.m.RLock()
|
|
||||||
defer i.ui.m.RUnlock()
|
|
||||||
|
|
||||||
_, ok := i.keys[key]
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) Wheel() (xoff, yoff float64) {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsMouseButtonPressed(key MouseButton) bool {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) update(keys map[Key]struct{}, runes []rune, touches []Touch) {
|
|
||||||
i.ui.m.Lock()
|
|
||||||
defer i.ui.m.Unlock()
|
|
||||||
|
|
||||||
if i.keys == nil {
|
|
||||||
i.keys = map[Key]struct{}{}
|
|
||||||
}
|
|
||||||
for k := range i.keys {
|
|
||||||
delete(i.keys, k)
|
|
||||||
}
|
|
||||||
for k := range keys {
|
|
||||||
i.keys[k] = struct{}{}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
i.runes = i.runes[:0]
|
copy(u.inputState.Runes[:], runes)
|
||||||
i.runes = append(i.runes, runes...)
|
u.inputState.RunesCount = len(runes)
|
||||||
|
|
||||||
i.touches = i.touches[:0]
|
for i := range u.inputState.Touches {
|
||||||
i.touches = append(i.touches, touches...)
|
u.inputState.Touches[i].Valid = false
|
||||||
}
|
}
|
||||||
|
for i, t := range touches {
|
||||||
func (i *Input) resetForTick() {
|
x, y := u.context.adjustPosition(t.X, t.Y, u.DeviceScaleFactor())
|
||||||
i.runes = nil
|
u.inputState.Touches[i] = Touch{
|
||||||
|
Valid: true,
|
||||||
|
ID: t.ID,
|
||||||
|
X: int(x),
|
||||||
|
Y: int(y),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,73 +17,23 @@
|
|||||||
package ui
|
package ui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sync"
|
|
||||||
|
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/nintendosdk"
|
"github.com/hajimehoshi/ebiten/v2/internal/nintendosdk"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Input struct {
|
func (u *userInterfaceImpl) updateInputState() {
|
||||||
gamepads []nintendosdk.Gamepad
|
u.nativeTouches = u.nativeTouches[:0]
|
||||||
touches []nintendosdk.Touch
|
u.nativeTouches = nintendosdk.AppendTouches(u.nativeTouches)
|
||||||
|
|
||||||
m sync.Mutex
|
for i := range u.inputState.Touches {
|
||||||
}
|
u.inputState.Touches[i].Valid = false
|
||||||
|
|
||||||
func (i *Input) update(context *context) {
|
|
||||||
i.m.Lock()
|
|
||||||
defer i.m.Unlock()
|
|
||||||
|
|
||||||
gamepad.Update()
|
|
||||||
|
|
||||||
i.touches = i.touches[:0]
|
|
||||||
i.touches = nintendosdk.AppendTouches(i.touches)
|
|
||||||
|
|
||||||
for idx, t := range i.touches {
|
|
||||||
x, y := context.adjustPosition(float64(t.X), float64(t.Y), deviceScaleFactor)
|
|
||||||
i.touches[idx].X = int(x)
|
|
||||||
i.touches[idx].Y = int(y)
|
|
||||||
}
|
}
|
||||||
}
|
for i, t := range u.nativeTouches {
|
||||||
|
x, y := u.context.adjustPosition(float64(t.X), float64(t.Y), deviceScaleFactor)
|
||||||
func (i *Input) AppendInputChars(runes []rune) []rune {
|
u.inputState.Touches[i] = Touch{
|
||||||
return nil
|
Valid: true,
|
||||||
}
|
ID: TouchID(t.ID),
|
||||||
|
X: int(x),
|
||||||
func (i *Input) AppendTouchIDs(touchIDs []TouchID) []TouchID {
|
Y: int(y),
|
||||||
i.m.Lock()
|
|
||||||
defer i.m.Unlock()
|
|
||||||
|
|
||||||
for _, t := range i.touches {
|
|
||||||
touchIDs = append(touchIDs, TouchID(t.ID))
|
|
||||||
}
|
|
||||||
return touchIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) CursorPosition() (x, y int) {
|
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsKeyPressed(key Key) bool {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) IsMouseButtonPressed(button MouseButton) bool {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) TouchPosition(id TouchID) (x, y int) {
|
|
||||||
i.m.Lock()
|
|
||||||
defer i.m.Unlock()
|
|
||||||
|
|
||||||
for _, t := range i.touches {
|
|
||||||
if TouchID(t.ID) == id {
|
|
||||||
return t.X, t.Y
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0, 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Input) Wheel() (xoff, yoff float64) {
|
|
||||||
return 0, 0
|
|
||||||
}
|
}
|
||||||
|
@ -132,6 +132,7 @@ const (
|
|||||||
KeyReserved1
|
KeyReserved1
|
||||||
KeyReserved2
|
KeyReserved2
|
||||||
KeyReserved3
|
KeyReserved3
|
||||||
|
KeyMax = KeyReserved3
|
||||||
)
|
)
|
||||||
|
|
||||||
func (k Key) String() string {
|
func (k Key) String() string {
|
||||||
|
@ -21,18 +21,6 @@ import (
|
|||||||
"github.com/hajimehoshi/ebiten/v2/internal/mipmap"
|
"github.com/hajimehoshi/ebiten/v2/internal/mipmap"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MouseButton int
|
|
||||||
|
|
||||||
const (
|
|
||||||
MouseButton0 MouseButton = iota // The 'left' button
|
|
||||||
MouseButton1 // The 'right' button
|
|
||||||
MouseButton2 // The 'middle' button
|
|
||||||
MouseButton3 // The additional button (usually browser-back)
|
|
||||||
MouseButton4 // The additional button (usually browser-forward)
|
|
||||||
)
|
|
||||||
|
|
||||||
type TouchID int
|
|
||||||
|
|
||||||
// RegularTermination represents a regular termination.
|
// RegularTermination represents a regular termination.
|
||||||
// Run can return this error, and if this error is received,
|
// Run can return this error, and if this error is received,
|
||||||
// the game loop should be terminated as soon as possible.
|
// the game loop should be terminated as soon as possible.
|
||||||
|
@ -100,7 +100,7 @@ type userInterfaceImpl struct {
|
|||||||
|
|
||||||
fpsModeInited bool
|
fpsModeInited bool
|
||||||
|
|
||||||
input Input
|
inputState InputState
|
||||||
iwindow glfwWindow
|
iwindow glfwWindow
|
||||||
|
|
||||||
sizeCallback glfw.SizeCallback
|
sizeCallback glfw.SizeCallback
|
||||||
@ -137,7 +137,6 @@ func init() {
|
|||||||
origWindowPosX: invalidPos,
|
origWindowPosX: invalidPos,
|
||||||
origWindowPosY: invalidPos,
|
origWindowPosY: invalidPos,
|
||||||
}
|
}
|
||||||
theUI.input.ui = &theUI.userInterfaceImpl
|
|
||||||
theUI.iwindow.ui = &theUI.userInterfaceImpl
|
theUI.iwindow.ui = &theUI.userInterfaceImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -697,8 +696,13 @@ func (u *userInterfaceImpl) createWindow(width, height int) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) beginFrame() {
|
func (u *userInterfaceImpl) beginFrame(inputState *InputState) {
|
||||||
atomic.StoreUint32(&u.inFrame, 1)
|
atomic.StoreUint32(&u.inFrame, 1)
|
||||||
|
|
||||||
|
u.m.Lock()
|
||||||
|
defer u.m.Unlock()
|
||||||
|
*inputState = u.inputState
|
||||||
|
u.inputState.resetForFrame()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) endFrame() {
|
func (u *userInterfaceImpl) endFrame() {
|
||||||
@ -958,6 +962,7 @@ func (u *userInterfaceImpl) init(options *RunOptions) error {
|
|||||||
u.registerWindowSetSizeCallback()
|
u.registerWindowSetSizeCallback()
|
||||||
u.registerWindowCloseCallback()
|
u.registerWindowCloseCallback()
|
||||||
u.registerWindowFramebufferSizeCallback()
|
u.registerWindowFramebufferSizeCallback()
|
||||||
|
u.registerInputCallbacks()
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -1046,7 +1051,7 @@ func (u *userInterfaceImpl) update() (float64, float64, error) {
|
|||||||
} else {
|
} else {
|
||||||
glfw.WaitEvents()
|
glfw.WaitEvents()
|
||||||
}
|
}
|
||||||
if err := u.input.update(u.window, u.context); err != nil {
|
if err := u.updateInputState(); err != nil {
|
||||||
return 0, 0, err
|
return 0, 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1408,15 +1413,15 @@ func monitorFromWindow(window *glfw.Window) *glfw.Monitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) resetForTick() {
|
func (u *userInterfaceImpl) resetForTick() {
|
||||||
u.input.resetForTick()
|
|
||||||
|
|
||||||
u.m.Lock()
|
u.m.Lock()
|
||||||
|
defer u.m.Unlock()
|
||||||
u.windowBeingClosed = false
|
u.windowBeingClosed = false
|
||||||
u.m.Unlock()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Input() *Input {
|
func (u *userInterfaceImpl) readInputState(inputState *InputState) {
|
||||||
return &u.input
|
u.m.Lock()
|
||||||
|
defer u.m.Unlock()
|
||||||
|
*inputState = u.inputState
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Window() Window {
|
func (u *userInterfaceImpl) Window() Window {
|
||||||
|
@ -87,7 +87,9 @@ type userInterfaceImpl struct {
|
|||||||
err error
|
err error
|
||||||
|
|
||||||
context *context
|
context *context
|
||||||
input Input
|
inputState InputState
|
||||||
|
origCursorX int
|
||||||
|
origCursorY int
|
||||||
|
|
||||||
m sync.Mutex
|
m sync.Mutex
|
||||||
}
|
}
|
||||||
@ -96,7 +98,6 @@ func init() {
|
|||||||
theUI.userInterfaceImpl = userInterfaceImpl{
|
theUI.userInterfaceImpl = userInterfaceImpl{
|
||||||
runnableOnUnfocused: true,
|
runnableOnUnfocused: true,
|
||||||
}
|
}
|
||||||
theUI.input.ui = &theUI.userInterfaceImpl
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -475,7 +476,7 @@ func init() {
|
|||||||
if theUI.cursorMode == CursorModeCaptured {
|
if theUI.cursorMode == CursorModeCaptured {
|
||||||
theUI.recoverCursorMode()
|
theUI.recoverCursorMode()
|
||||||
}
|
}
|
||||||
theUI.input.recoverCursorPosition()
|
theUI.recoverCursorPosition()
|
||||||
return nil
|
return nil
|
||||||
}))
|
}))
|
||||||
document.Call("addEventListener", "pointerlockerror", js.FuncOf(func(this js.Value, args []js.Value) any {
|
document.Call("addEventListener", "pointerlockerror", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
@ -516,7 +517,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
|
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -525,7 +526,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "keyup", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "keyup", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -539,7 +540,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
|
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -548,7 +549,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "mouseup", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "mouseup", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -557,7 +558,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "mousemove", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "mousemove", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -566,7 +567,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "wheel", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "wheel", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -580,7 +581,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
|
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -589,7 +590,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "touchend", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "touchend", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -598,7 +599,7 @@ func setCanvasEventHandlers(v js.Value) {
|
|||||||
v.Call("addEventListener", "touchmove", js.FuncOf(func(this js.Value, args []js.Value) any {
|
v.Call("addEventListener", "touchmove", js.FuncOf(func(this js.Value, args []js.Value) any {
|
||||||
e := args[0]
|
e := args[0]
|
||||||
e.Call("preventDefault")
|
e.Call("preventDefault")
|
||||||
if err := theUI.input.updateFromEvent(e); err != nil && theUI.err != nil {
|
if err := theUI.updateInputFromEvent(e); err != nil && theUI.err != nil {
|
||||||
theUI.err = err
|
theUI.err = err
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -680,18 +681,15 @@ func (u *userInterfaceImpl) SetScreenTransparent(transparent bool) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) resetForTick() {
|
func (u *userInterfaceImpl) resetForTick() {
|
||||||
u.input.resetForTick()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Input() *Input {
|
|
||||||
return &u.input
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Window() Window {
|
func (u *userInterfaceImpl) Window() Window {
|
||||||
return &nullWindow{}
|
return &nullWindow{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) beginFrame() {
|
func (u *userInterfaceImpl) beginFrame(inputState *InputState) {
|
||||||
|
*inputState = u.inputState
|
||||||
|
u.inputState.resetForFrame()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) endFrame() {
|
func (u *userInterfaceImpl) endFrame() {
|
||||||
|
@ -60,7 +60,6 @@ func init() {
|
|||||||
outsideWidth: 640,
|
outsideWidth: 640,
|
||||||
outsideHeight: 480,
|
outsideHeight: 480,
|
||||||
}
|
}
|
||||||
theUI.input.ui = &theUI.userInterfaceImpl
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update is called from mobile/ebitenmobileview.
|
// Update is called from mobile/ebitenmobileview.
|
||||||
@ -108,7 +107,7 @@ type userInterfaceImpl struct {
|
|||||||
|
|
||||||
context *context
|
context *context
|
||||||
|
|
||||||
input Input
|
inputState InputState
|
||||||
|
|
||||||
fpsMode FPSModeType
|
fpsMode FPSModeType
|
||||||
renderRequester RenderRequester
|
renderRequester RenderRequester
|
||||||
@ -127,7 +126,7 @@ func (u *userInterfaceImpl) appMain(a app.App) {
|
|||||||
var glctx gl.Context
|
var glctx gl.Context
|
||||||
var sizeInited bool
|
var sizeInited bool
|
||||||
|
|
||||||
touches := map[touch.Sequence]Touch{}
|
touches := map[touch.Sequence]TouchForInput{}
|
||||||
keys := map[Key]struct{}{}
|
keys := map[Key]struct{}{}
|
||||||
|
|
||||||
for e := range a.Events() {
|
for e := range a.Events() {
|
||||||
@ -180,12 +179,10 @@ func (u *userInterfaceImpl) appMain(a app.App) {
|
|||||||
switch e.Type {
|
switch e.Type {
|
||||||
case touch.TypeBegin, touch.TypeMove:
|
case touch.TypeBegin, touch.TypeMove:
|
||||||
s := deviceScale()
|
s := deviceScale()
|
||||||
x, y := float64(e.X)/s, float64(e.Y)/s
|
touches[e.Sequence] = TouchForInput{
|
||||||
// TODO: Is it ok to cast from int64 to int here?
|
|
||||||
touches[e.Sequence] = Touch{
|
|
||||||
ID: TouchID(e.Sequence),
|
ID: TouchID(e.Sequence),
|
||||||
X: int(x),
|
X: float64(e.X) / s,
|
||||||
Y: int(y),
|
Y: float64(e.Y) / s,
|
||||||
}
|
}
|
||||||
case touch.TypeEnd:
|
case touch.TypeEnd:
|
||||||
delete(touches, e.Sequence)
|
delete(touches, e.Sequence)
|
||||||
@ -212,11 +209,11 @@ func (u *userInterfaceImpl) appMain(a app.App) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if updateInput {
|
if updateInput {
|
||||||
var ts []Touch
|
var ts []TouchForInput
|
||||||
for _, t := range touches {
|
for _, t := range touches {
|
||||||
ts = append(ts, t)
|
ts = append(ts, t)
|
||||||
}
|
}
|
||||||
u.input.update(keys, runes, ts)
|
u.updateInputState(keys, runes, ts)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -422,25 +419,14 @@ func (u *userInterfaceImpl) DeviceScaleFactor() float64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) resetForTick() {
|
func (u *userInterfaceImpl) resetForTick() {
|
||||||
u.input.resetForTick()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Input() *Input {
|
|
||||||
return &u.input
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Window() Window {
|
func (u *userInterfaceImpl) Window() Window {
|
||||||
return &nullWindow{}
|
return &nullWindow{}
|
||||||
}
|
}
|
||||||
|
|
||||||
type Touch struct {
|
func (u *userInterfaceImpl) UpdateInput(keys map[Key]struct{}, runes []rune, touches []TouchForInput) {
|
||||||
ID TouchID
|
u.updateInputState(keys, runes, touches)
|
||||||
X int
|
|
||||||
Y int
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u *userInterfaceImpl) UpdateInput(keys map[Key]struct{}, runes []rune, touches []Touch) {
|
|
||||||
u.input.update(keys, runes, touches)
|
|
||||||
if u.fpsMode == FPSModeVsyncOffMinimum {
|
if u.fpsMode == FPSModeVsyncOffMinimum {
|
||||||
u.renderRequester.RequestRenderIfNeeded()
|
u.renderRequester.RequestRenderIfNeeded()
|
||||||
}
|
}
|
||||||
@ -462,7 +448,12 @@ func (u *userInterfaceImpl) ScheduleFrame() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) beginFrame() {
|
func (u *userInterfaceImpl) beginFrame(inputState *InputState) {
|
||||||
|
u.m.Lock()
|
||||||
|
defer u.m.Unlock()
|
||||||
|
|
||||||
|
*inputState = u.inputState
|
||||||
|
u.inputState.resetForFrame()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) endFrame() {
|
func (u *userInterfaceImpl) endFrame() {
|
||||||
|
@ -19,6 +19,7 @@ package ui
|
|||||||
import (
|
import (
|
||||||
"runtime"
|
"runtime"
|
||||||
|
|
||||||
|
"github.com/hajimehoshi/ebiten/v2/internal/gamepad"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver"
|
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/opengl"
|
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/opengl"
|
||||||
"github.com/hajimehoshi/ebiten/v2/internal/nintendosdk"
|
"github.com/hajimehoshi/ebiten/v2/internal/nintendosdk"
|
||||||
@ -53,7 +54,8 @@ type userInterfaceImpl struct {
|
|||||||
graphicsDriver graphicsdriver.Graphics
|
graphicsDriver graphicsdriver.Graphics
|
||||||
|
|
||||||
context *context
|
context *context
|
||||||
input Input
|
inputState InputState
|
||||||
|
nativeTouches []nintendosdk.Touch
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) Run(game Game, options *RunOptions) error {
|
func (u *userInterfaceImpl) Run(game Game, options *RunOptions) error {
|
||||||
@ -66,7 +68,8 @@ func (u *userInterfaceImpl) Run(game Game, options *RunOptions) error {
|
|||||||
nintendosdk.InitializeGame()
|
nintendosdk.InitializeGame()
|
||||||
for {
|
for {
|
||||||
nintendosdk.BeginFrame()
|
nintendosdk.BeginFrame()
|
||||||
u.input.update(u.context)
|
gamepad.Update()
|
||||||
|
u.updateInputState()
|
||||||
|
|
||||||
w, h := nintendosdk.ScreenSize()
|
w, h := nintendosdk.ScreenSize()
|
||||||
if err := u.context.updateFrame(u.graphicsDriver, float64(w), float64(h), deviceScaleFactor, u); err != nil {
|
if err := u.context.updateFrame(u.graphicsDriver, float64(w), float64(h), deviceScaleFactor, u); err != nil {
|
||||||
@ -126,15 +129,13 @@ func (*userInterfaceImpl) SetFPSMode(mode FPSModeType) {
|
|||||||
func (*userInterfaceImpl) ScheduleFrame() {
|
func (*userInterfaceImpl) ScheduleFrame() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (*userInterfaceImpl) Input() *Input {
|
|
||||||
return &theUI.input
|
|
||||||
}
|
|
||||||
|
|
||||||
func (*userInterfaceImpl) Window() Window {
|
func (*userInterfaceImpl) Window() Window {
|
||||||
return &nullWindow{}
|
return &nullWindow{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) beginFrame() {
|
func (u *userInterfaceImpl) beginFrame(inputState *InputState) {
|
||||||
|
*inputState = u.inputState
|
||||||
|
u.inputState.resetForFrame()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *userInterfaceImpl) endFrame() {
|
func (u *userInterfaceImpl) endFrame() {
|
||||||
|
@ -32,16 +32,16 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
touchSlice []ui.Touch
|
touchSlice []ui.TouchForInput
|
||||||
)
|
)
|
||||||
|
|
||||||
func updateInput() {
|
func updateInput() {
|
||||||
touchSlice = touchSlice[:0]
|
touchSlice = touchSlice[:0]
|
||||||
for id, position := range touches {
|
for id, position := range touches {
|
||||||
touchSlice = append(touchSlice, ui.Touch{
|
touchSlice = append(touchSlice, ui.TouchForInput{
|
||||||
ID: id,
|
ID: id,
|
||||||
X: position.x,
|
X: float64(position.x),
|
||||||
Y: position.y,
|
Y: float64(position.y),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user