use crate::common::WebviewSource;
use crate::focus::FocusedWebview;
use bevy::input::keyboard::KeyboardInput;
use bevy::prelude::*;
#[cfg(not(target_os = "windows"))]
use bevy_cef_core::prelude::Browsers;
#[cfg(target_os = "windows")]
use bevy_cef_core::prelude::BrowsersProxy;
use bevy_cef_core::prelude::{create_cef_key_events, keyboard_modifiers};
use serde::{Deserialize, Serialize};
pub(super) struct KeyboardPlugin;
impl Plugin for KeyboardPlugin {
fn build(&self, app: &mut App) {
app.init_resource::<IsImeCommiting>()
.init_resource::<IsImeComposing>();
#[cfg(not(target_os = "windows"))]
app.add_systems(
Update,
(
activate_ime,
ime_event.run_if(on_message::<Ime>),
send_key_event.run_if(on_message::<KeyboardInput>),
)
.chain(),
);
#[cfg(target_os = "windows")]
app.add_systems(
Update,
(
activate_ime,
ime_event_win.run_if(on_message::<Ime>),
send_key_event_win.run_if(on_message::<KeyboardInput>),
)
.chain(),
);
}
}
fn activate_ime(mut windows: Query<&mut Window>, mut state: Local<ImeActivationState>) {
match *state {
ImeActivationState::Pending => {
for mut window in windows.iter_mut() {
if window.ime_enabled {
window.ime_enabled = false;
*state = ImeActivationState::Toggled;
}
}
}
ImeActivationState::Toggled => {
for mut window in windows.iter_mut() {
if !window.ime_enabled {
window.ime_enabled = true;
*state = ImeActivationState::Done;
}
}
}
ImeActivationState::Done => {}
}
}
#[derive(Default)]
enum ImeActivationState {
#[default]
Pending,
Toggled,
Done,
}
#[derive(Resource, Default, Serialize, Deserialize, Reflect)]
#[reflect(Default, Serialize, Deserialize)]
struct IsImeCommiting(bool);
#[derive(Resource, Default, Serialize, Deserialize, Reflect)]
#[reflect(Default, Serialize, Deserialize)]
struct IsImeComposing(bool);
#[cfg(not(target_os = "windows"))]
fn send_key_event(
mut er: MessageReader<KeyboardInput>,
mut is_ime_commiting: ResMut<IsImeCommiting>,
mut is_ime_composing: ResMut<IsImeComposing>,
input: Res<ButtonInput<KeyCode>>,
browsers: NonSend<Browsers>,
focused: Res<FocusedWebview>,
webviews: Query<Entity, With<WebviewSource>>,
) {
let modifiers = keyboard_modifiers(&input);
let target = focused.0.filter(|e| webviews.get(*e).is_ok());
for event in er.read() {
if (event.key_code == KeyCode::Enter || event.key_code == KeyCode::Backspace)
&& is_ime_commiting.0
{
is_ime_commiting.0 = false;
continue;
}
if event.key_code == KeyCode::Backspace && is_ime_composing.0 {
is_ime_composing.0 = false;
continue;
}
let Some(webview) = target else {
continue;
};
for key_event in create_cef_key_events(modifiers, event) {
browsers.send_key(&webview, key_event);
}
}
}
#[cfg(not(target_os = "windows"))]
fn ime_event(
mut er: MessageReader<Ime>,
mut is_ime_commiting: ResMut<IsImeCommiting>,
mut is_ime_composing: ResMut<IsImeComposing>,
browsers: NonSend<Browsers>,
focused: Res<FocusedWebview>,
webviews: Query<Entity, With<WebviewSource>>,
) {
let has_target = focused.0.filter(|e| webviews.get(*e).is_ok()).is_some();
if !has_target {
if is_ime_composing.0 {
browsers.ime_cancel_composition();
}
is_ime_composing.0 = false;
is_ime_commiting.0 = false;
}
for event in er.read() {
if !has_target {
continue;
}
match event {
Ime::Preedit { value, cursor, .. } => {
if value.is_empty() {
browsers.ime_cancel_composition();
} else {
browsers.set_ime_composition(value, cursor.map(|(_, e)| e as u32));
is_ime_composing.0 = true;
}
}
Ime::Commit { value, .. } => {
browsers.set_ime_commit_text(value);
is_ime_commiting.0 = true;
is_ime_composing.0 = false;
}
Ime::Disabled { .. } => {
browsers.ime_cancel_composition();
is_ime_composing.0 = false;
}
_ => {}
}
}
}
#[cfg(target_os = "windows")]
fn send_key_event_win(
mut er: MessageReader<KeyboardInput>,
mut is_ime_commiting: ResMut<IsImeCommiting>,
mut is_ime_composing: ResMut<IsImeComposing>,
input: Res<ButtonInput<KeyCode>>,
proxy: Res<BrowsersProxy>,
focused: Res<FocusedWebview>,
webviews: Query<Entity, With<WebviewSource>>,
) {
let modifiers = keyboard_modifiers(&input);
let target = focused.0.filter(|e| webviews.get(*e).is_ok());
for event in er.read() {
if (event.key_code == KeyCode::Enter || event.key_code == KeyCode::Backspace)
&& is_ime_commiting.0
{
is_ime_commiting.0 = false;
continue;
}
if event.key_code == KeyCode::Backspace && is_ime_composing.0 {
is_ime_composing.0 = false;
continue;
}
let Some(webview) = target else {
continue;
};
for key_event in create_cef_key_events(modifiers, event) {
proxy.send_key(&webview, key_event);
}
}
}
#[cfg(target_os = "windows")]
fn ime_event_win(
mut er: MessageReader<Ime>,
mut is_ime_commiting: ResMut<IsImeCommiting>,
mut is_ime_composing: ResMut<IsImeComposing>,
proxy: Res<BrowsersProxy>,
focused: Res<FocusedWebview>,
webviews: Query<Entity, With<WebviewSource>>,
) {
let has_target = focused.0.filter(|e| webviews.get(*e).is_ok()).is_some();
if !has_target {
if is_ime_composing.0 {
proxy.ime_cancel_composition();
}
is_ime_composing.0 = false;
is_ime_commiting.0 = false;
}
for event in er.read() {
if !has_target {
continue;
}
match event {
Ime::Preedit { value, cursor, .. } => {
if value.is_empty() {
proxy.ime_cancel_composition();
} else {
proxy.set_ime_composition(value, cursor.map(|(_, e)| e as u32));
is_ime_composing.0 = true;
}
}
Ime::Commit { value, .. } => {
proxy.set_ime_commit_text(value);
is_ime_commiting.0 = true;
is_ime_composing.0 = false;
}
Ime::Disabled { .. } => {
proxy.ime_cancel_composition();
is_ime_composing.0 = false;
}
_ => {}
}
}
}