use std::fmt;
use crate::{
common::{BitArray, CharacterSet, Eci, Result},
exceptions::Exceptions,
};
use super::{HighLevelEncoder, Token};
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct State {
mode: u32,
token: Token,
binary_shift_byte_count: u32,
bit_count: u32,
binary_shift_cost: u32,
}
impl State {
pub fn new(token: Token, mode: u32, binary_bytes: u32, bit_count: u32) -> Self {
Self {
mode,
token,
binary_shift_byte_count: binary_bytes,
bit_count,
binary_shift_cost: Self::calculate_binary_shift_cost(binary_bytes),
}
}
pub fn getMode(&self) -> u32 {
self.mode
}
pub fn getToken(&self) -> &Token {
&self.token
}
pub fn getBinaryShiftByteCount(&self) -> u32 {
self.binary_shift_byte_count
}
pub fn getBitCount(&self) -> u32 {
self.bit_count
}
pub fn appendFLGn(self, eci: Eci) -> Result<Self> {
let bit_count = self.bit_count;
let mode = self.mode;
let result = self.shiftAndAppend(HighLevelEncoder::MODE_PUNCT as u32, 0); let mut token = result.token;
let mut bits_added = 3;
if eci as u32 > 999999 {
return Err(Exceptions::illegal_argument_with(
"ECI code must be between 0 and 999999",
));
} else {
let Ok(eci_digits) = CharacterSet::ISO8859_1.encode(&format!("{eci}")) else {
return Err(Exceptions::ILLEGAL_ARGUMENT);
};
token.add(eci_digits.len() as i32, 3); for eci_digit in &eci_digits {
token.add((eci_digit - b'0' + 2) as i32, 4);
}
bits_added += eci_digits.len() * 4;
}
Ok(State::new(token, mode, 0, bit_count + bits_added as u32))
}
pub fn latchAndAppend(self, mode: u32, value: u32) -> State {
let mut bitCount = self.bit_count;
let mut token = self.token;
if mode != self.mode {
let latch = HighLevelEncoder::LATCH_TABLE[self.mode as usize][mode as usize];
token.add(latch as i32 & 0xFFFF, latch >> 16);
bitCount += latch >> 16;
}
let latchModeBitCount = if mode == HighLevelEncoder::MODE_DIGIT as u32 {
4
} else {
5
};
token.add(value as i32, latchModeBitCount);
State::new(token, mode, 0, bitCount + latchModeBitCount)
}
pub fn shiftAndAppend(self, mode: u32, value: u32) -> State {
let mut token = self.token;
let thisModeBitCount = if self.mode == HighLevelEncoder::MODE_DIGIT as u32 {
4
} else {
5
};
token.add(
HighLevelEncoder::SHIFT_TABLE[self.mode as usize][mode as usize],
thisModeBitCount,
);
token.add(value as i32, 5);
State::new(token, self.mode, 0, self.bit_count + thisModeBitCount + 5)
}
pub fn addBinaryShiftChar(self, index: u32) -> State {
let mut token = self.token;
let mut mode = self.mode;
let mut bitCount = self.bit_count;
if self.mode == HighLevelEncoder::MODE_PUNCT as u32
|| self.mode == HighLevelEncoder::MODE_DIGIT as u32
{
let latch = HighLevelEncoder::LATCH_TABLE[mode as usize][HighLevelEncoder::MODE_UPPER];
token.add(latch as i32 & 0xFFFF, latch >> 16);
bitCount += latch >> 16;
mode = HighLevelEncoder::MODE_UPPER as u32;
}
let deltaBitCount =
if self.binary_shift_byte_count == 0 || self.binary_shift_byte_count == 31 {
18
} else if self.binary_shift_byte_count == 62 {
9
} else {
8
};
let mut result = State::new(
token,
mode,
self.binary_shift_byte_count + 1,
bitCount + deltaBitCount,
);
if result.binary_shift_byte_count == 2047 + 31 {
result = result.endBinaryShift(index + 1);
}
result
}
pub fn endBinaryShift(self, index: u32) -> State {
if self.binary_shift_byte_count == 0 {
return self;
}
let mut token = self.token;
token.addBinaryShift(
index - self.binary_shift_byte_count,
self.binary_shift_byte_count,
);
State::new(token, self.mode, 0, self.bit_count)
}
pub fn isBetterThanOrEqualTo(&self, other: &State) -> bool {
let mut new_mode_bit_count = self.bit_count
+ (HighLevelEncoder::LATCH_TABLE[self.mode as usize][other.mode as usize] >> 16);
if self.binary_shift_byte_count < other.binary_shift_byte_count {
new_mode_bit_count += other.binary_shift_cost - self.binary_shift_cost;
} else if self.binary_shift_byte_count > other.binary_shift_byte_count
&& other.binary_shift_byte_count > 0
{
new_mode_bit_count += 10;
}
new_mode_bit_count <= other.bit_count
}
pub fn toBitArray(self, text: &[u8]) -> Result<BitArray> {
let mut symbols = Vec::new();
let tok = self.endBinaryShift(text.len() as u32).token;
for tkn in tok.into_iter() {
symbols.push(tkn);
}
let mut bit_array = BitArray::new();
for symbol in symbols.into_iter().rev() {
symbol.appendTo(&mut bit_array, text)?;
}
Ok(bit_array)
}
#[inline(always)]
fn calculate_binary_shift_cost(binary_shift_byte_count: u32) -> u32 {
if binary_shift_byte_count > 62 {
21 } else if binary_shift_byte_count > 31 {
20 } else if binary_shift_byte_count > 0 {
10 } else {
0
}
}
}
impl fmt::Display for State {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(
f,
"{} bits={} bytes={}",
HighLevelEncoder::MODE_NAMES[self.mode as usize],
self.bit_count,
self.binary_shift_byte_count
)
}
}