mirror of
https://github.com/hexedtech/codemp.git
synced 2025-04-05 02:51:34 +02:00
156 lines
4.5 KiB
Rust
156 lines
4.5 KiB
Rust
use jni::{objects::{JClass, JObject, JValueGen}, sys::{jlong, jobject, jstring}, JNIEnv};
|
|
|
|
use crate::{api::Controller, ffi::java::handle_callback};
|
|
|
|
use super::{JExceptable, RT};
|
|
|
|
/// Gets the name of the buffer.
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_get_1name(
|
|
mut env: JNIEnv,
|
|
_class: JClass,
|
|
self_ptr: jlong,
|
|
) -> jstring {
|
|
let controller = unsafe { Box::leak(Box::from_raw(self_ptr as *mut crate::buffer::Controller)) };
|
|
let content = controller.path();
|
|
env.new_string(content)
|
|
.jexcept(&mut env)
|
|
.as_raw()
|
|
}
|
|
|
|
/// Gets the contents of the buffers.
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_get_1content(
|
|
mut env: JNIEnv,
|
|
_class: JClass,
|
|
self_ptr: jlong,
|
|
) -> jstring {
|
|
let controller = unsafe { Box::leak(Box::from_raw(self_ptr as *mut crate::buffer::Controller)) };
|
|
let content = RT.block_on(controller.content())
|
|
.jexcept(&mut env);
|
|
env.new_string(content)
|
|
.jexcept(&mut env)
|
|
.as_raw()
|
|
}
|
|
|
|
/// Tries to fetch a [crate::api::TextChange], or returns null if there's nothing.
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_try_1recv(
|
|
mut env: JNIEnv,
|
|
_class: JClass,
|
|
self_ptr: jlong,
|
|
) -> jobject {
|
|
let controller = unsafe { Box::leak(Box::from_raw(self_ptr as *mut crate::buffer::Controller)) };
|
|
let change = RT.block_on(controller.try_recv()).jexcept(&mut env);
|
|
recv_jni(&mut env, change)
|
|
}
|
|
|
|
/// Blocks until it receives a [crate::api::TextChange].
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_recv(
|
|
mut env: JNIEnv,
|
|
_class: JClass,
|
|
self_ptr: jlong,
|
|
) -> jobject {
|
|
let controller = unsafe { Box::leak(Box::from_raw(self_ptr as *mut crate::buffer::Controller)) };
|
|
let change = RT.block_on(controller.recv()).map(Some).jexcept(&mut env);
|
|
recv_jni(&mut env, change)
|
|
}
|
|
|
|
/// Utility method to convert a [crate::api::TextChange] to its Java equivalent.
|
|
fn recv_jni(env: &mut JNIEnv, change: Option<crate::api::TextChange>) -> jobject {
|
|
match change {
|
|
None => JObject::default(),
|
|
Some(event) => {
|
|
let content = env.new_string(event.content).jexcept(env);
|
|
|
|
let hash = env.find_class("java/util/OptionalLong").and_then(|class| {
|
|
if let Some(h) = event.hash {
|
|
env.call_static_method(class, "of", "(J)Ljava/util/OptionalLong;", &[JValueGen::Long(h)])
|
|
} else {
|
|
env.call_static_method(class, "empty", "()Ljava/util/OptionalLong;", &[])
|
|
}
|
|
}).and_then(|o| o.l()).jexcept(env);
|
|
env.find_class("mp/code/data/TextChange")
|
|
.and_then(|class| {
|
|
env.new_object(
|
|
class,
|
|
"(JJLjava/lang/String;Ljava/util/OptionalLong;)V",
|
|
&[
|
|
JValueGen::Long(jlong::from(event.start)),
|
|
JValueGen::Long(jlong::from(event.end)),
|
|
JValueGen::Object(&content),
|
|
JValueGen::Object(&hash)
|
|
]
|
|
)
|
|
}).jexcept(env)
|
|
}
|
|
}.as_raw()
|
|
}
|
|
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_callback<'local>(
|
|
mut env: JNIEnv,
|
|
_class: JClass<'local>,
|
|
self_ptr: jlong,
|
|
cb: JObject<'local>,
|
|
) {
|
|
handle_callback!("mp/code/BufferController", env, self_ptr, cb, crate::buffer::Controller);
|
|
}
|
|
|
|
/// Receive from Java, converts and sends a [crate::api::TextChange].
|
|
#[no_mangle]
|
|
pub extern "system" fn Java_mp_code_BufferController_send<'local>(
|
|
mut env: JNIEnv,
|
|
_class: JClass<'local>,
|
|
self_ptr: jlong,
|
|
input: JObject<'local>,
|
|
) {
|
|
let Ok(start) = env.get_field(&input, "start", "I")
|
|
.and_then(|sr| sr.i())
|
|
.jexcept(&mut env)
|
|
.try_into()
|
|
else {
|
|
env.throw_new("java/lang/IllegalArgumentException", "Start index cannot be negative!")
|
|
.expect("Failed to throw exception!");
|
|
return;
|
|
};
|
|
|
|
let Ok(end) = env.get_field(&input, "end", "I")
|
|
.and_then(|er| er.i())
|
|
.jexcept(&mut env)
|
|
.try_into()
|
|
else {
|
|
env.throw_new("java/lang/IllegalArgumentException", "End index cannot be negative!")
|
|
.expect("Failed to throw exception!");
|
|
return;
|
|
};
|
|
|
|
let content = env.get_field(&input, "content", "Ljava/lang/String;")
|
|
.and_then(|b| b.l())
|
|
.map(|b| b.into())
|
|
.jexcept(&mut env);
|
|
let content = env.get_string(&content)
|
|
.map(|b| b.into())
|
|
.jexcept(&mut env);
|
|
|
|
let hash = env.get_field(&input, "hash", "Ljava/util/OptionalLong")
|
|
.and_then(|hash| hash.l())
|
|
.and_then(|hash| {
|
|
if env.call_method(&hash, "isPresent", "()Z", &[]).and_then(|r| r.z()).jexcept(&mut env) {
|
|
env.call_method(&hash, "getAsLong", "()J", &[])
|
|
.and_then(|r| r.j())
|
|
.map(Some)
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}).jexcept(&mut env);
|
|
|
|
let controller = unsafe { Box::leak(Box::from_raw(self_ptr as *mut crate::buffer::Controller)) };
|
|
RT.block_on(controller.send(crate::api::TextChange {
|
|
start,
|
|
end,
|
|
content,
|
|
hash,
|
|
})).jexcept(&mut env);
|
|
}
|