use crate::{
common::{BitArray, Result},
point, BarcodeFormat, Binarizer, Exceptions, RXingResult, RXingResultMetadataType,
RXingResultMetadataValue, Reader,
};
use super::{one_d_reader, EANManufacturerOrgSupport, OneDReader, UPCEANExtensionSupport};
use once_cell::sync::Lazy;
pub static EAN_MANUFACTURER_SUPPORT: Lazy<EANManufacturerOrgSupport> =
Lazy::new(EANManufacturerOrgSupport::default);
pub static UPC_EAN_EXTENSION_SUPPORT: Lazy<UPCEANExtensionSupport> =
Lazy::new(UPCEANExtensionSupport::default);
pub const MAX_AVG_VARIANCE: f32 = 0.48;
pub const MAX_INDIVIDUAL_VARIANCE: f32 = 0.7;
pub const START_END_PATTERN: [u32; 3] = [1, 1, 1];
pub const MIDDLE_PATTERN: [u32; 5] = [1, 1, 1, 1, 1];
pub const END_PATTERN: [u32; 6] = [1, 1, 1, 1, 1, 1];
pub const L_PATTERNS: [[u32; 4]; 10] = [
[3, 2, 1, 1], [2, 2, 2, 1], [2, 1, 2, 2], [1, 4, 1, 1], [1, 1, 3, 2], [1, 2, 3, 1], [1, 1, 1, 4], [1, 3, 1, 2], [1, 2, 1, 3], [3, 1, 1, 2], ];
pub const L_AND_G_PATTERNS: [[u32; 4]; 20] = {
let mut new_array = [[0_u32; 4]; 20];
let mut i = 0;
while i < 10 {
new_array[i] = L_PATTERNS[i];
i += 1;
}
let mut i = 10;
while i < 20 {
let widths = &L_PATTERNS[i - 10];
let mut reversedWidths = [0_u32; 4];
let mut j = 0;
while j < 4 {
reversedWidths[j] = widths[4 - j - 1];
j += 1;
}
new_array[i] = reversedWidths;
i += 1;
}
new_array
};
pub trait UPCEANReader: OneDReader {
fn find_start_guard_pattern(&self, row: &BitArray) -> Result<[usize; 2]> {
let mut foundStart = false;
let mut startRange = [0; 2];
let mut nextStart = 0;
let mut counters = [0_u32; 3];
while !foundStart {
counters.fill(0);
startRange = self.findGuardPatternWithCounters(
row,
nextStart,
false,
&START_END_PATTERN,
&mut counters,
)?;
let start = startRange[0];
nextStart = startRange[1];
let quietStart = start as isize - (nextStart as isize - start as isize);
if quietStart >= 0 {
foundStart = row.isRange(quietStart as usize, start, false)?;
}
}
Ok(startRange)
}
fn decodeRowWithGuardRange(
&self,
rowNumber: u32,
row: &BitArray,
startGuardRange: &[usize; 2],
hints: &crate::DecodeHints,
) -> Result<RXingResult> {
let resultPointCallback = &hints.NeedResultPointCallback;
let mut symbologyIdentifier = 0;
if let Some(cb) = resultPointCallback {
cb(point(
(startGuardRange[0] + startGuardRange[1]) as f32 / 2.0,
rowNumber as f32,
));
}
let mut result = String::new();
let endStart = self.decodeMiddle(row, startGuardRange, &mut result)?;
if let Some(cb) = resultPointCallback {
cb(point(endStart as f32, rowNumber as f32));
}
let endRange = self.decodeEnd(row, endStart)?;
if let Some(cb) = resultPointCallback {
cb(point(
(endRange[0] + endRange[1]) as f32 / 2.0,
rowNumber as f32,
));
}
let end = endRange[1];
let quietEnd = end + (end - endRange[0]);
if quietEnd >= row.get_size() || !row.isRange(end, quietEnd, false)? {
return Err(Exceptions::NOT_FOUND);
}
let resultString = result;
if resultString.chars().count() < 8 {
return Err(Exceptions::FORMAT);
}
if !self.checkChecksum(&resultString)? {
return Err(Exceptions::CHECKSUM);
}
let left = (startGuardRange[1] + startGuardRange[0]) as f32 / 2.0;
let right: f32 = (endRange[1] + endRange[0]) as f32 / 2.0;
let format = self.getBarcodeFormat();
let mut decodeRXingResult = RXingResult::new(
&resultString,
Vec::new(), vec![
point(left, rowNumber as f32),
point(right, rowNumber as f32),
],
format,
);
let mut extensionLength = 0;
let mut attempt = || -> Result<()> {
let extensionRXingResult =
UPC_EAN_EXTENSION_SUPPORT.decodeRow(rowNumber, row, endRange[1])?;
decodeRXingResult.putMetadata(
RXingResultMetadataType::UPC_EAN_EXTENSION,
RXingResultMetadataValue::UpcEanExtension(
extensionRXingResult.getText().to_owned(),
),
);
decodeRXingResult.putAllMetadata(extensionRXingResult.getRXingResultMetadata().clone());
decodeRXingResult.addPoints(&mut extensionRXingResult.getPoints().to_vec());
extensionLength = extensionRXingResult.getText().chars().count();
Ok(())
};
let _try_result = attempt();
if let Some(allowedExtensions) = &hints.AllowedEanExtensions {
let mut valid = false;
for length in allowedExtensions {
if extensionLength == *length as usize {
valid = true;
break;
}
}
if !valid {
return Err(Exceptions::NOT_FOUND);
}
}
if format == BarcodeFormat::EAN_13 || format == BarcodeFormat::UPC_A {
let countryID = EAN_MANUFACTURER_SUPPORT.lookupCountryIdentifier(&resultString);
if let Some(cid) = countryID {
decodeRXingResult.putMetadata(
RXingResultMetadataType::POSSIBLE_COUNTRY,
RXingResultMetadataValue::PossibleCountry(cid.to_owned()),
);
}
}
if format == BarcodeFormat::EAN_8 {
symbologyIdentifier = 4;
}
decodeRXingResult.putMetadata(
RXingResultMetadataType::SYMBOLOGY_IDENTIFIER,
RXingResultMetadataValue::SymbologyIdentifier(format!("]E{symbologyIdentifier}")),
);
Ok(decodeRXingResult)
}
fn checkChecksum(&self, s: &str) -> Result<bool> {
self.checkStandardUPCEANChecksum(s)
}
fn checkStandardUPCEANChecksum(&self, s: &str) -> Result<bool> {
let s = s.chars().collect::<Vec<_>>();
let length = s.len();
if length == 0 {
return Ok(false);
}
let char_in_question = *s.get(length - 1).ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)?;
let check = char_in_question.is_ascii_digit();
let check_against = &s[..length - 1]; let calculated_checksum = self.getStandardUPCEANChecksum(check_against)?;
Ok(calculated_checksum
== if check {
char_in_question.to_digit(10).ok_or(Exceptions::PARSE)?
} else {
u32::MAX
})
}
fn getStandardUPCEANChecksum(&self, s: &[char]) -> Result<u32> {
let length = s.len();
let mut sum = 0;
let mut i = length as isize - 1;
while i >= 0 {
let digit =
(*s.get(i as usize).ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)? as i32) - ('0' as i32);
if !(0..=9).contains(&digit) {
return Err(Exceptions::FORMAT);
}
sum += digit;
i -= 2;
}
sum *= 3;
let mut i = length as isize - 2;
while i >= 0 {
let digit =
(*s.get(i as usize).ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)? as i32) - ('0' as i32);
if !(0..=9).contains(&digit) {
return Err(Exceptions::FORMAT);
}
sum += digit;
i -= 2;
}
Ok(((1000 - sum) % 10) as u32)
}
fn decodeEnd(&self, row: &BitArray, endStart: usize) -> Result<[usize; 2]> {
self.findGuardPattern(row, endStart, false, &START_END_PATTERN)
}
fn findGuardPattern(
&self,
row: &BitArray,
rowOffset: usize,
whiteFirst: bool,
pattern: &[u32],
) -> Result<[usize; 2]> {
self.findGuardPatternWithCounters(
row,
rowOffset,
whiteFirst,
pattern,
&mut vec![0u32; pattern.len()],
)
}
fn findGuardPatternWithCounters(
&self,
row: &BitArray,
rowOffset: usize,
whiteFirst: bool,
pattern: &[u32],
counters: &mut [u32],
) -> Result<[usize; 2]> {
let width = row.get_size();
let rowOffset = if whiteFirst {
row.getNextUnset(rowOffset)
} else {
row.getNextSet(rowOffset)
};
let mut counterPosition = 0;
let mut patternStart = rowOffset;
let patternLength = pattern.len();
let mut isWhite = whiteFirst;
for x in rowOffset..width {
if row.get(x) != isWhite {
counters[counterPosition] += 1;
} else {
if counterPosition == patternLength - 1 {
if one_d_reader::pattern_match_variance(
counters,
pattern,
MAX_INDIVIDUAL_VARIANCE,
) < MAX_AVG_VARIANCE
{
return Ok([patternStart, x]);
}
patternStart += (counters[0] + counters[1]) as usize;
counters.copy_within(2..(counterPosition - 1 + 2), 0);
counters[counterPosition - 1] = 0;
counters[counterPosition] = 0;
counterPosition -= 1;
} else {
counterPosition += 1;
}
counters[counterPosition] = 1;
isWhite = !isWhite;
}
}
Err(Exceptions::NOT_FOUND)
}
fn decodeDigit(
&self,
row: &BitArray,
counters: &mut [u32; 4],
rowOffset: usize,
patterns: &[[u32; 4]],
) -> Result<usize> {
one_d_reader::record_pattern(row, rowOffset, counters)?;
let mut bestVariance = MAX_AVG_VARIANCE; let mut bestMatch = -1_isize;
let max = patterns.len();
for (i, pattern) in patterns.iter().enumerate().take(max) {
let variance: f32 =
one_d_reader::pattern_match_variance(counters, pattern, MAX_INDIVIDUAL_VARIANCE);
if variance < bestVariance {
bestVariance = variance;
bestMatch = i as isize;
}
}
if bestMatch >= 0 {
Ok(bestMatch as usize)
} else {
Err(Exceptions::NOT_FOUND)
}
}
fn getBarcodeFormat(&self) -> BarcodeFormat;
fn decodeMiddle(
&self,
row: &BitArray,
startRange: &[usize; 2],
resultString: &mut String,
) -> Result<usize>;
}
pub(crate) struct StandInStruct;
impl UPCEANReader for StandInStruct {
fn getBarcodeFormat(&self) -> BarcodeFormat {
unimplemented!()
}
fn decodeMiddle(
&self,
_row: &BitArray,
_startRange: &[usize; 2],
_resultString: &mut String,
) -> Result<usize> {
unimplemented!()
}
}
impl OneDReader for StandInStruct {
fn decode_row(
&mut self,
_rowNumber: u32,
_row: &BitArray,
_hints: &crate::DecodeHints,
) -> Result<RXingResult> {
unimplemented!()
}
}
impl Reader for StandInStruct {
fn decode<B: Binarizer>(&mut self, _image: &mut crate::BinaryBitmap<B>) -> Result<RXingResult> {
unimplemented!()
}
fn decode_with_hints<B: Binarizer>(
&mut self,
_image: &mut crate::BinaryBitmap<B>,
_hints: &crate::DecodeHints,
) -> Result<RXingResult> {
unimplemented!()
}
}
pub(crate) const STAND_IN: StandInStruct = StandInStruct {};