shortcut when using literal booleans
This commit is contained in:
parent
9c54dee454
commit
f6dadf881a
|
@ -11,6 +11,8 @@ use crate::{
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use enum_as_inner::EnumAsInner;
|
||||||
|
|
||||||
#[cfg(feature = "shulkerbox")]
|
#[cfg(feature = "shulkerbox")]
|
||||||
use shulkerbox::prelude::{Command, Condition, Execute};
|
use shulkerbox::prelude::{Command, Condition, Execute};
|
||||||
|
|
||||||
|
@ -149,6 +151,15 @@ impl StorageType {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Condition
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, EnumAsInner)]
|
||||||
|
pub enum ExtendedCondition {
|
||||||
|
/// Runtime condition
|
||||||
|
Runtime(Condition),
|
||||||
|
/// Compile-time condition
|
||||||
|
Comptime(bool),
|
||||||
|
}
|
||||||
|
|
||||||
impl Expression {
|
impl Expression {
|
||||||
/// Returns whether the expression can yield a certain type.
|
/// Returns whether the expression can yield a certain type.
|
||||||
#[must_use]
|
#[must_use]
|
||||||
|
@ -276,7 +287,7 @@ impl Binary {
|
||||||
.can_yield_type(ValueType::Boolean, scope)*/ => {
|
.can_yield_type(ValueType::Boolean, scope)*/ => {
|
||||||
Some(ComptimeValue::Boolean(true))
|
Some(ComptimeValue::Boolean(true))
|
||||||
}
|
}
|
||||||
(ComptimeValue::Boolean(false), _) | (_, ComptimeValue::Boolean(false)) if matches!(self.operator(), BinaryOperator::LogicalAnd(..))
|
(ComptimeValue::Boolean(false), _) | (_, ComptimeValue::Boolean(false)) if matches!(self.operator(), BinaryOperator::LogicalAnd(..))
|
||||||
// TODO: re-enable if can_yield_type works properly
|
// TODO: re-enable if can_yield_type works properly
|
||||||
/*&& self
|
/*&& self
|
||||||
.left_operand()
|
.left_operand()
|
||||||
|
@ -731,11 +742,22 @@ impl Transpiler {
|
||||||
),
|
),
|
||||||
};
|
};
|
||||||
|
|
||||||
cmds.push(Command::Execute(Execute::If(
|
let cmd = match cond {
|
||||||
cond,
|
ExtendedCondition::Runtime(cond) => Command::Execute(Execute::If(
|
||||||
Box::new(Execute::Run(Box::new(Command::Raw(success_cmd)))),
|
cond,
|
||||||
Some(Box::new(Execute::Run(Box::new(Command::Raw(else_cmd))))),
|
Box::new(Execute::Run(Box::new(Command::Raw(success_cmd)))),
|
||||||
)));
|
Some(Box::new(Execute::Run(Box::new(Command::Raw(else_cmd))))),
|
||||||
|
)),
|
||||||
|
ExtendedCondition::Comptime(cond) => {
|
||||||
|
if cond {
|
||||||
|
Command::Raw(success_cmd)
|
||||||
|
} else {
|
||||||
|
Command::Raw(else_cmd)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
cmds.push(cmd);
|
||||||
|
|
||||||
Ok(cmds)
|
Ok(cmds)
|
||||||
}
|
}
|
||||||
|
@ -748,7 +770,7 @@ impl Transpiler {
|
||||||
expression: &Expression,
|
expression: &Expression,
|
||||||
scope: &Arc<super::Scope>,
|
scope: &Arc<super::Scope>,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<(Vec<Command>, Condition)> {
|
) -> TranspileResult<(Vec<Command>, ExtendedCondition)> {
|
||||||
match expression {
|
match expression {
|
||||||
Expression::Primary(primary) => {
|
Expression::Primary(primary) => {
|
||||||
self.transpile_primary_expression_as_condition(primary, scope, handler)
|
self.transpile_primary_expression_as_condition(primary, scope, handler)
|
||||||
|
@ -759,14 +781,17 @@ impl Transpiler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[expect(clippy::too_many_lines)]
|
||||||
fn transpile_primary_expression_as_condition(
|
fn transpile_primary_expression_as_condition(
|
||||||
&mut self,
|
&mut self,
|
||||||
primary: &Primary,
|
primary: &Primary,
|
||||||
scope: &Arc<super::Scope>,
|
scope: &Arc<super::Scope>,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<(Vec<Command>, Condition)> {
|
) -> TranspileResult<(Vec<Command>, ExtendedCondition)> {
|
||||||
match primary {
|
match primary {
|
||||||
Primary::Boolean(_) => todo!("handle boolean literal if not catched by comptime eval"),
|
Primary::Boolean(boolean) => {
|
||||||
|
Ok((Vec::new(), ExtendedCondition::Comptime(boolean.value())))
|
||||||
|
}
|
||||||
Primary::Integer(_) => {
|
Primary::Integer(_) => {
|
||||||
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
||||||
expected_type: ValueType::Boolean,
|
expected_type: ValueType::Boolean,
|
||||||
|
@ -777,11 +802,12 @@ impl Transpiler {
|
||||||
}
|
}
|
||||||
Primary::StringLiteral(s) => Ok((
|
Primary::StringLiteral(s) => Ok((
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
Condition::Atom(s.str_content().to_string().into()),
|
ExtendedCondition::Runtime(Condition::Atom(s.str_content().to_string().into())),
|
||||||
|
)),
|
||||||
|
Primary::MacroStringLiteral(macro_string) => Ok((
|
||||||
|
Vec::new(),
|
||||||
|
ExtendedCondition::Runtime(Condition::Atom(macro_string.into())),
|
||||||
)),
|
)),
|
||||||
Primary::MacroStringLiteral(macro_string) => {
|
|
||||||
Ok((Vec::new(), Condition::Atom(macro_string.into())))
|
|
||||||
}
|
|
||||||
Primary::FunctionCall(func) => {
|
Primary::FunctionCall(func) => {
|
||||||
if func
|
if func
|
||||||
.arguments()
|
.arguments()
|
||||||
|
@ -806,7 +832,9 @@ impl Transpiler {
|
||||||
|
|
||||||
Ok((
|
Ok((
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
Condition::Atom(format!("function {func_location}").into()),
|
ExtendedCondition::Runtime(Condition::Atom(
|
||||||
|
format!("function {func_location}").into(),
|
||||||
|
)),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -816,16 +844,20 @@ impl Transpiler {
|
||||||
match variable {
|
match variable {
|
||||||
VariableData::BooleanStorage { storage_name, path } => Ok((
|
VariableData::BooleanStorage { storage_name, path } => Ok((
|
||||||
Vec::new(),
|
Vec::new(),
|
||||||
Condition::Atom(format!("data storage {storage_name} {{{path}: 1b}}").into()),
|
ExtendedCondition::Runtime(Condition::Atom(
|
||||||
|
format!("data storage {storage_name} {{{path}: 1b}}").into(),
|
||||||
|
)),
|
||||||
|
)),
|
||||||
|
VariableData::FunctionArgument { .. } => Ok((
|
||||||
|
Vec::new(),
|
||||||
|
ExtendedCondition::Runtime(Condition::Atom(
|
||||||
|
shulkerbox::util::MacroString::MacroString(vec![
|
||||||
|
shulkerbox::util::MacroStringPart::MacroUsage(
|
||||||
|
ident.span.str().to_string(),
|
||||||
|
),
|
||||||
|
]),
|
||||||
|
)),
|
||||||
)),
|
)),
|
||||||
VariableData::FunctionArgument { .. } => {
|
|
||||||
Ok((
|
|
||||||
Vec::new(),
|
|
||||||
Condition::Atom(shulkerbox::util::MacroString::MacroString(vec![
|
|
||||||
shulkerbox::util::MacroStringPart::MacroUsage(ident.span.str().to_string()),
|
|
||||||
]))
|
|
||||||
))
|
|
||||||
}
|
|
||||||
_ => {
|
_ => {
|
||||||
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
||||||
expected_type: ValueType::Boolean,
|
expected_type: ValueType::Boolean,
|
||||||
|
@ -842,8 +874,7 @@ impl Transpiler {
|
||||||
handler.receive(err.clone());
|
handler.receive(err.clone());
|
||||||
Err(err)
|
Err(err)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
},
|
|
||||||
Primary::Parenthesized(parenthesized) => {
|
Primary::Parenthesized(parenthesized) => {
|
||||||
self.transpile_expression_as_condition(parenthesized.expression(), scope, handler)
|
self.transpile_expression_as_condition(parenthesized.expression(), scope, handler)
|
||||||
}
|
}
|
||||||
|
@ -854,7 +885,15 @@ impl Transpiler {
|
||||||
scope,
|
scope,
|
||||||
handler,
|
handler,
|
||||||
)?;
|
)?;
|
||||||
Ok((cmds, Condition::Not(Box::new(cond))))
|
Ok((
|
||||||
|
cmds,
|
||||||
|
match cond {
|
||||||
|
ExtendedCondition::Runtime(cond) => {
|
||||||
|
ExtendedCondition::Runtime(Condition::Not(Box::new(cond)))
|
||||||
|
}
|
||||||
|
ExtendedCondition::Comptime(cond) => ExtendedCondition::Comptime(!cond),
|
||||||
|
},
|
||||||
|
))
|
||||||
}
|
}
|
||||||
PrefixOperator::Negate(_) => {
|
PrefixOperator::Negate(_) => {
|
||||||
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
let err = TranspileError::MismatchedTypes(MismatchedTypes {
|
||||||
|
@ -863,7 +902,7 @@ impl Transpiler {
|
||||||
});
|
});
|
||||||
handler.receive(err.clone());
|
handler.receive(err.clone());
|
||||||
Err(err)
|
Err(err)
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
Primary::Lua(_) => todo!("Lua code as condition"),
|
Primary::Lua(_) => todo!("Lua code as condition"),
|
||||||
}
|
}
|
||||||
|
@ -874,16 +913,16 @@ impl Transpiler {
|
||||||
binary: &Binary,
|
binary: &Binary,
|
||||||
scope: &Arc<super::Scope>,
|
scope: &Arc<super::Scope>,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<(Vec<Command>, Condition)> {
|
) -> TranspileResult<(Vec<Command>, ExtendedCondition)> {
|
||||||
match binary.operator() {
|
match binary.operator() {
|
||||||
BinaryOperator::Equal(..)
|
BinaryOperator::Equal(..)
|
||||||
| BinaryOperator::NotEqual(..)
|
| BinaryOperator::NotEqual(..)
|
||||||
| BinaryOperator::GreaterThan(_)
|
| BinaryOperator::GreaterThan(_)
|
||||||
| BinaryOperator::GreaterThanOrEqual(..)
|
| BinaryOperator::GreaterThanOrEqual(..)
|
||||||
| BinaryOperator::LessThan(_)
|
| BinaryOperator::LessThan(_)
|
||||||
| BinaryOperator::LessThanOrEqual(..) => {
|
| BinaryOperator::LessThanOrEqual(..) => self
|
||||||
self.transpile_comparison_operator(binary, scope, handler)
|
.transpile_comparison_operator(binary, scope, handler)
|
||||||
}
|
.map(|(cmds, cond)| (cmds, ExtendedCondition::Runtime(cond))),
|
||||||
BinaryOperator::LogicalAnd(..) | BinaryOperator::LogicalOr(..) => {
|
BinaryOperator::LogicalAnd(..) | BinaryOperator::LogicalOr(..) => {
|
||||||
self.transpile_logic_operator(binary, scope, handler)
|
self.transpile_logic_operator(binary, scope, handler)
|
||||||
}
|
}
|
||||||
|
@ -1066,7 +1105,7 @@ impl Transpiler {
|
||||||
binary: &Binary,
|
binary: &Binary,
|
||||||
scope: &Arc<super::Scope>,
|
scope: &Arc<super::Scope>,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<(Vec<Command>, Condition)> {
|
) -> TranspileResult<(Vec<Command>, ExtendedCondition)> {
|
||||||
let left = binary.left_operand().as_ref();
|
let left = binary.left_operand().as_ref();
|
||||||
let right = binary.right_operand().as_ref();
|
let right = binary.right_operand().as_ref();
|
||||||
|
|
||||||
|
@ -1075,16 +1114,44 @@ impl Transpiler {
|
||||||
let (right_cmds, right_cond) =
|
let (right_cmds, right_cond) =
|
||||||
self.transpile_expression_as_condition(right, scope, handler)?;
|
self.transpile_expression_as_condition(right, scope, handler)?;
|
||||||
|
|
||||||
let combined_cmds = left_cmds.into_iter().chain(right_cmds).collect();
|
match (binary.operator(), left_cond, right_cond) {
|
||||||
|
(BinaryOperator::LogicalAnd(..), ExtendedCondition::Comptime(true), other)
|
||||||
match binary.operator() {
|
| (BinaryOperator::LogicalOr(..), ExtendedCondition::Comptime(false), other) => {
|
||||||
BinaryOperator::LogicalAnd(..) => Ok((
|
Ok((right_cmds, other))
|
||||||
combined_cmds,
|
}
|
||||||
Condition::And(Box::new(left_cond), Box::new(right_cond)),
|
(BinaryOperator::LogicalAnd(..), other, ExtendedCondition::Comptime(true))
|
||||||
|
| (BinaryOperator::LogicalOr(..), other, ExtendedCondition::Comptime(false)) => {
|
||||||
|
Ok((left_cmds, other))
|
||||||
|
}
|
||||||
|
(BinaryOperator::LogicalAnd(..), ExtendedCondition::Comptime(false), _)
|
||||||
|
| (BinaryOperator::LogicalAnd(..), _, ExtendedCondition::Comptime(false)) => {
|
||||||
|
Ok((Vec::new(), ExtendedCondition::Comptime(false)))
|
||||||
|
}
|
||||||
|
(BinaryOperator::LogicalOr(..), ExtendedCondition::Comptime(true), _)
|
||||||
|
| (BinaryOperator::LogicalOr(..), _, ExtendedCondition::Comptime(true)) => {
|
||||||
|
Ok((Vec::new(), ExtendedCondition::Comptime(true)))
|
||||||
|
}
|
||||||
|
(
|
||||||
|
BinaryOperator::LogicalAnd(..),
|
||||||
|
ExtendedCondition::Runtime(left_cond),
|
||||||
|
ExtendedCondition::Runtime(right_cond),
|
||||||
|
) => Ok((
|
||||||
|
left_cmds.into_iter().chain(right_cmds).collect(),
|
||||||
|
ExtendedCondition::Runtime(Condition::And(
|
||||||
|
Box::new(left_cond),
|
||||||
|
Box::new(right_cond),
|
||||||
|
)),
|
||||||
)),
|
)),
|
||||||
BinaryOperator::LogicalOr(..) => Ok((
|
(
|
||||||
combined_cmds,
|
BinaryOperator::LogicalOr(..),
|
||||||
Condition::Or(Box::new(left_cond), Box::new(right_cond)),
|
ExtendedCondition::Runtime(left_cond),
|
||||||
|
ExtendedCondition::Runtime(right_cond),
|
||||||
|
) => Ok((
|
||||||
|
left_cmds.into_iter().chain(right_cmds).collect(),
|
||||||
|
ExtendedCondition::Runtime(Condition::Or(
|
||||||
|
Box::new(left_cond),
|
||||||
|
Box::new(right_cond),
|
||||||
|
)),
|
||||||
)),
|
)),
|
||||||
_ => unreachable!("This function should only be called for logical operators."),
|
_ => unreachable!("This function should only be called for logical operators."),
|
||||||
}
|
}
|
||||||
|
@ -1187,9 +1254,13 @@ impl Transpiler {
|
||||||
|
|
||||||
let targets = (0..amount)
|
let targets = (0..amount)
|
||||||
.map(|i| {
|
.map(|i| {
|
||||||
chksum_md5::hash(format!("{namespace}\0{j}", namespace = self.main_namespace_name, j = i + self.temp_counter))
|
chksum_md5::hash(format!(
|
||||||
.to_hex_lowercase()
|
"{namespace}\0{j}",
|
||||||
.split_off(16)
|
namespace = self.main_namespace_name,
|
||||||
|
j = i + self.temp_counter
|
||||||
|
))
|
||||||
|
.to_hex_lowercase()
|
||||||
|
.split_off(16)
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,7 @@ use crate::{
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
error::{TranspileError, TranspileResult},
|
error::{TranspileError, TranspileResult},
|
||||||
expression::ComptimeValue,
|
expression::{ComptimeValue, ExtendedCondition},
|
||||||
variables::{Scope, VariableData},
|
variables::{Scope, VariableData},
|
||||||
FunctionData, TranspileAnnotationValue,
|
FunctionData, TranspileAnnotationValue,
|
||||||
};
|
};
|
||||||
|
@ -810,10 +810,19 @@ impl Transpiler {
|
||||||
let (pre_cond_cmds, cond) =
|
let (pre_cond_cmds, cond) =
|
||||||
self.transpile_expression_as_condition(cond_expression, scope, handler)?;
|
self.transpile_expression_as_condition(cond_expression, scope, handler)?;
|
||||||
|
|
||||||
Ok(Some((
|
match cond {
|
||||||
pre_cond_cmds,
|
ExtendedCondition::Runtime(cond) => Ok(Some((
|
||||||
Execute::If(cond, Box::new(then), el.map(Box::new)),
|
pre_cond_cmds,
|
||||||
)))
|
Execute::If(cond, Box::new(then), el.map(Box::new)),
|
||||||
|
))),
|
||||||
|
ExtendedCondition::Comptime(cond) => {
|
||||||
|
if cond {
|
||||||
|
Ok(Some((Vec::new(), then)))
|
||||||
|
} else {
|
||||||
|
Ok(el.map(|el| (Vec::new(), el)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue