mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-28 03:32:45 +01:00
exp/textinput: bug fix: a virtual keybord didn't show on iOS Safari
This change fixes the issue by processing a textarea element in a click or touchstart event. There is still little odd behaviors, like a keyboard is sometimes not shown. Let's fix them later. Updates #2898
This commit is contained in:
parent
e3b54b41c2
commit
7eee3929de
@ -43,6 +43,7 @@ var theTextInput textInput
|
|||||||
|
|
||||||
func (t *textInput) init() {
|
func (t *textInput) init() {
|
||||||
t.textareaElement = document.Call("createElement", "textarea")
|
t.textareaElement = document.Call("createElement", "textarea")
|
||||||
|
t.textareaElement.Set("id", "ebitengine-textinput")
|
||||||
t.textareaElement.Set("autocapitalize", "off")
|
t.textareaElement.Set("autocapitalize", "off")
|
||||||
t.textareaElement.Set("spellcheck", false)
|
t.textareaElement.Set("spellcheck", false)
|
||||||
t.textareaElement.Set("translate", "no")
|
t.textareaElement.Set("translate", "no")
|
||||||
@ -111,6 +112,29 @@ func (t *textInput) init() {
|
|||||||
t.trySend(true)
|
t.trySend(true)
|
||||||
return nil
|
return nil
|
||||||
}))
|
}))
|
||||||
|
body.Call("appendChild", t.textareaElement)
|
||||||
|
|
||||||
|
js.Global().Call("eval", `
|
||||||
|
// Process the textarea element under user-interaction events.
|
||||||
|
// This is due to an iOS Safari restriction (#2898).
|
||||||
|
let handler = (e) => {
|
||||||
|
if (window._ebitengine_textinput_x === undefined || window._ebitengine_textinput_y === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let textarea = document.getElementById("ebitengine-textinput");
|
||||||
|
textarea.value = '';
|
||||||
|
textarea.focus();
|
||||||
|
textarea.style.left = _ebitengine_textinput_x + 'px';
|
||||||
|
textarea.style.top = _ebitengine_textinput_y + 'px';
|
||||||
|
window._ebitengine_textinput_x = undefined;
|
||||||
|
window._ebitengine_textinput_y = undefined;
|
||||||
|
window._ebitengine_textinput_ready = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
let body = window.document.body;
|
||||||
|
body.addEventListener("click", handler);
|
||||||
|
body.addEventListener("touchstart", handler);`)
|
||||||
|
|
||||||
// TODO: What about other events like wheel?
|
// TODO: What about other events like wheel?
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -119,17 +143,23 @@ func (t *textInput) Start(x, y int) (chan State, func()) {
|
|||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if js.Global().Get("_ebitengine_textinput_ready").Truthy() {
|
||||||
|
s := newSession()
|
||||||
|
t.session = s
|
||||||
|
js.Global().Get("window").Set("_ebitengine_textinput_ready", js.Undefined())
|
||||||
|
return s.ch, s.end
|
||||||
|
}
|
||||||
|
|
||||||
if t.session != nil {
|
if t.session != nil {
|
||||||
t.session.end()
|
t.session.end()
|
||||||
t.session = nil
|
t.session = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if !body.Call("contains", t.textareaElement).Bool() {
|
// If a textarea is focused, create a session immediately.
|
||||||
body.Call("appendChild", t.textareaElement)
|
// A virtual keyboard should already be shown on mobile browsers.
|
||||||
}
|
if document.Get("activeElement").Equal(t.textareaElement) {
|
||||||
t.textareaElement.Set("value", "")
|
t.textareaElement.Set("value", "")
|
||||||
t.textareaElement.Call("focus")
|
t.textareaElement.Call("focus")
|
||||||
|
|
||||||
style := t.textareaElement.Get("style")
|
style := t.textareaElement.Get("style")
|
||||||
style.Set("left", fmt.Sprintf("%dpx", x))
|
style.Set("left", fmt.Sprintf("%dpx", x))
|
||||||
style.Set("top", fmt.Sprintf("%dpx", y))
|
style.Set("top", fmt.Sprintf("%dpx", y))
|
||||||
@ -137,6 +167,13 @@ func (t *textInput) Start(x, y int) (chan State, func()) {
|
|||||||
s := newSession()
|
s := newSession()
|
||||||
t.session = s
|
t.session = s
|
||||||
return s.ch, s.end
|
return s.ch, s.end
|
||||||
|
}
|
||||||
|
|
||||||
|
// On iOS Safari, `focus` works only in user-interaction events (#2898).
|
||||||
|
// Assuming Start is called every tick, defer the starting process to the next user-interaction event.
|
||||||
|
js.Global().Get("window").Set("_ebitengine_textinput_x", x)
|
||||||
|
js.Global().Get("window").Set("_ebitengine_textinput_y", y)
|
||||||
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *textInput) trySend(committed bool) {
|
func (t *textInput) trySend(committed bool) {
|
||||||
|
Loading…
Reference in New Issue
Block a user