use cairo_lang_diagnostics::ToOption;
use cairo_lang_utils::{define_short_id, try_extract_matches};
use id_arena::Arena;
use num_bigint::BigInt;
use crate::corelib::get_core_trait;
use crate::db::SemanticGroup;
use crate::items::functions::GenericFunctionId;
use crate::{Expr, ExprFunctionCall, ExprFunctionCallArg};
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
pub struct LiteralLongId {
pub value: BigInt,
}
define_short_id!(LiteralId, LiteralLongId, SemanticGroup, lookup_intern_literal);
impl LiteralId {
pub fn format(&self, db: &dyn SemanticGroup) -> String {
db.lookup_intern_literal(*self).value.to_string()
}
}
pub fn try_extract_minus_literal(
db: &dyn SemanticGroup,
exprs: &Arena<Expr>,
expr: &ExprFunctionCall,
) -> Option<BigInt> {
let [ExprFunctionCallArg::Value(expr_id)] = &expr.args[..] else {
return None;
};
let literal = try_extract_matches!(&exprs[*expr_id], Expr::Literal)?;
let imp = try_extract_matches!(
expr.function.get_concrete(db).generic_function,
GenericFunctionId::Impl
)?;
let trait_id = imp.impl_id.concrete_trait(db).to_option()?.trait_id(db);
if trait_id != get_core_trait(db, "Neg".into()) {
return None;
}
if imp.function.name(db.upcast()) != "neg" { None } else { Some(-literal.value.clone()) }
}