use runmat_builtins::{StructValue, Value};
use runmat_macros::runtime_builtin;
use super::plotting_error;
use super::state::{current_axes_state, encode_axes_handle, FigureAxesState};
use super::style::value_as_string;
use crate::builtins::plotting::type_resolvers::gca_type;
#[runtime_builtin(
name = "gca",
category = "plotting",
summary = "Return the handle for the current axes.",
keywords = "gca,axes,plotting",
suppress_auto_output = true,
type_resolver(gca_type),
builtin_path = "crate::builtins::plotting::gca"
)]
pub fn gca_builtin(rest: Vec<Value>) -> crate::BuiltinResult<Value> {
let state = current_axes_state();
if rest.is_empty() {
return Ok(Value::Num(encode_axes_handle(
state.handle,
state.active_index,
)));
}
if rest.len() == 1 {
if let Some(mode) = value_as_string(&rest[0]) {
if mode.trim().eq_ignore_ascii_case("struct") {
return Ok(axes_struct_response(state));
}
}
}
Err(plotting_error(
"gca",
"gca: unsupported arguments (pass no inputs or 'struct')",
))
}
fn axes_struct_response(state: FigureAxesState) -> Value {
let mut st = StructValue::new();
st.insert(
"handle",
Value::Num(encode_axes_handle(state.handle, state.active_index)),
);
st.insert("figure", Value::Num(state.handle.as_u32() as f64));
st.insert("rows", Value::Num(state.rows as f64));
st.insert("cols", Value::Num(state.cols as f64));
st.insert("index", Value::Num((state.active_index + 1) as f64));
Value::Struct(st)
}
#[cfg(test)]
pub(crate) mod tests {
use super::*;
use crate::builtins::plotting::tests::ensure_plot_test_env;
use runmat_builtins::{ResolveContext, Type};
fn setup_plot_tests() {
ensure_plot_test_env();
}
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)]
#[test]
fn default_returns_scalar_handle() {
setup_plot_tests();
let handle = gca_builtin(Vec::new()).unwrap();
match handle {
Value::Num(v) => assert!(v > 0.0),
other => panic!("expected scalar handle, got {other:?}"),
}
}
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)]
#[test]
fn struct_mode_returns_struct() {
setup_plot_tests();
let value = gca_builtin(vec![Value::String("struct".to_string())]).unwrap();
assert!(matches!(value, Value::Struct(_)));
}
#[test]
fn gca_type_no_args_returns_num() {
assert_eq!(gca_type(&[], &ResolveContext::new(Vec::new())), Type::Num);
}
#[test]
fn gca_type_with_args_returns_struct() {
let out = gca_type(&[Type::String], &ResolveContext::new(Vec::new()));
assert!(matches!(out, Type::Struct { .. }));
}
}