use std::collections::BTreeSet;
use runmat_builtins::Value;
use runmat_builtins::{
BuiltinCompletionPolicy, BuiltinDescriptor, BuiltinErrorDescriptor, BuiltinOutputMode,
BuiltinParamArity, BuiltinParamDescriptor, BuiltinParamType, BuiltinSignatureDescriptor,
};
use runmat_macros::runtime_builtin;
use super::op_common::figure_actions::{parse_clf_action, FigureAction};
use super::state::{clear_figure, clear_figure_with_builtin, figure_handles, FigureHandle};
use crate::builtins::plotting::type_resolvers::handle_scalar_type;
const CLF_OUTPUT_RESULT: [BuiltinParamDescriptor; 1] = [BuiltinParamDescriptor {
name: "result",
ty: BuiltinParamType::NumericScalar,
arity: BuiltinParamArity::Required,
default: None,
description: "Cleared handle for single-target calls or count for multi/all clears.",
}];
const CLF_INPUTS_NONE: [BuiltinParamDescriptor; 0] = [];
const CLF_INPUTS_TARGETS: [BuiltinParamDescriptor; 1] = [BuiltinParamDescriptor {
name: "targets",
ty: BuiltinParamType::Any,
arity: BuiltinParamArity::Variadic,
default: None,
description: "Figure handle(s) or mode tokens ('all', 'reset').",
}];
const CLF_SIGNATURES: [BuiltinSignatureDescriptor; 2] = [
BuiltinSignatureDescriptor {
label: "result = clf()",
inputs: &CLF_INPUTS_NONE,
outputs: &CLF_OUTPUT_RESULT,
},
BuiltinSignatureDescriptor {
label: "result = clf(targets...)",
inputs: &CLF_INPUTS_TARGETS,
outputs: &CLF_OUTPUT_RESULT,
},
];
const CLF_ERROR_INVALID_ARGUMENT: BuiltinErrorDescriptor = BuiltinErrorDescriptor {
code: "RM.CLF.INVALID_ARGUMENT",
identifier: Some("RunMat:clf:InvalidArgument"),
when: "One or more clear targets are invalid.",
message: "clf: invalid argument",
};
const CLF_ERROR_INTERNAL: BuiltinErrorDescriptor = BuiltinErrorDescriptor {
code: "RM.CLF.INTERNAL",
identifier: Some("RunMat:clf:Internal"),
when: "Internal figure clear operation fails.",
message: "clf: internal operation failed",
};
const CLF_ERRORS: [BuiltinErrorDescriptor; 2] = [CLF_ERROR_INVALID_ARGUMENT, CLF_ERROR_INTERNAL];
pub const CLF_DESCRIPTOR: BuiltinDescriptor = BuiltinDescriptor {
signatures: &CLF_SIGNATURES,
output_mode: BuiltinOutputMode::Fixed,
completion_policy: BuiltinCompletionPolicy::Public,
errors: &CLF_ERRORS,
};
#[runtime_builtin(
name = "clf",
category = "plotting",
summary = "Clear figure contents.",
keywords = "clf,clear figure,plotting",
sink = true,
suppress_auto_output = true,
type_resolver(handle_scalar_type),
descriptor(crate::builtins::plotting::clf::CLF_DESCRIPTOR),
builtin_path = "crate::builtins::plotting::clf"
)]
pub fn clf_builtin(rest: Vec<Value>) -> crate::BuiltinResult<f64> {
let (action, _reset) = parse_clf_action(&rest)?;
match action {
FigureAction::Current => {
let cleared = clear_figure_with_builtin("clf", None)?;
Ok(cleared.as_u32() as f64)
}
FigureAction::Handles(handles) => {
let ordered: BTreeSet<u32> = handles.into_iter().map(|h| h.as_u32()).collect();
if ordered.is_empty() {
let cleared = clear_figure_with_builtin("clf", None)?;
return Ok(cleared.as_u32() as f64);
}
for id in &ordered {
clear_figure_with_builtin("clf", Some(FigureHandle::from(*id)))?;
}
if ordered.len() == 1 {
Ok(*ordered.iter().next().unwrap() as f64)
} else {
Ok(ordered.len() as f64)
}
}
FigureAction::All => {
let handles = figure_handles();
if handles.is_empty() {
return Ok(0.0);
}
let count = handles.len();
for handle in handles {
let _ = clear_figure(Some(handle));
}
Ok(count as f64)
}
}
}
#[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 clf_descriptor_signatures_cover_core_forms() {
let labels: Vec<&str> = CLF_DESCRIPTOR
.signatures
.iter()
.map(|sig| sig.label)
.collect();
assert!(labels.contains(&"result = clf()"));
assert!(labels.contains(&"result = clf(targets...)"));
}
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)]
#[test]
fn defaults_to_current() {
setup_plot_tests();
assert!(matches!(
parse_clf_action(&[]).unwrap(),
(FigureAction::Current, false)
));
}
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)]
#[test]
fn parses_all_flag() {
setup_plot_tests();
let values = vec![Value::String("all".to_string())];
assert!(matches!(
parse_clf_action(&values).unwrap(),
(FigureAction::All, false)
));
}
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test::wasm_bindgen_test)]
#[test]
fn parses_handles() {
setup_plot_tests();
let values = vec![Value::Num(2.0)];
match parse_clf_action(&values).unwrap() {
(FigureAction::Handles(handles), _) => {
assert_eq!(handles.len(), 1);
assert_eq!(handles[0].as_u32(), 2);
}
other => panic!("unexpected: {other:?}"),
}
}
#[test]
fn clf_type_is_numeric_handle() {
assert_eq!(
handle_scalar_type(&[Type::tensor()], &ResolveContext::new(Vec::new())),
Type::Num
);
}
}