use crate::{
Exceptions,
common::{BitMatrix, Result},
};
use super::DataMask;
use crate::qrcode::common::{FormatInformation, Version, VersionRef};
pub struct BitMatrixParser {
bitMatrix: BitMatrix,
parsedVersion: Option<VersionRef>,
parsedFormatInfo: Option<FormatInformation>,
mirror: bool,
}
impl BitMatrixParser {
pub fn new(bit_matrix: BitMatrix) -> Result<Self> {
let dimension = bit_matrix.getHeight();
if dimension < 21 || (dimension & 0x03) != 1 {
Err(Exceptions::format_with(format!(
"{dimension} < 21 || ({dimension} % 0x03) != 1"
)))
} else {
Ok(Self {
bitMatrix: bit_matrix,
parsedVersion: None,
parsedFormatInfo: None,
mirror: false,
})
}
}
pub fn readFormatInformation(&mut self) -> Result<&FormatInformation> {
if self.parsedFormatInfo.is_some() {
return self.parsedFormatInfo.as_ref().ok_or(Exceptions::PARSE);
}
let mut formatInfoBits1 = 0;
for i in 0..6 {
formatInfoBits1 = self.copyBit(i, 8, formatInfoBits1);
}
formatInfoBits1 = self.copyBit(7, 8, formatInfoBits1);
formatInfoBits1 = self.copyBit(8, 8, formatInfoBits1);
formatInfoBits1 = self.copyBit(8, 7, formatInfoBits1);
for j in (0..=5).rev() {
formatInfoBits1 = self.copyBit(8, j, formatInfoBits1);
}
let dimension = self.bitMatrix.getHeight();
let mut formatInfoBits2 = 0;
let jMin = dimension - 7;
for j in (jMin..=dimension - 1).rev() {
formatInfoBits2 = self.copyBit(8, j, formatInfoBits2);
}
for i in (dimension - 8)..dimension {
formatInfoBits2 = self.copyBit(i, 8, formatInfoBits2);
}
self.parsedFormatInfo =
FormatInformation::decodeFormatInformation(formatInfoBits1, formatInfoBits2);
self.parsedFormatInfo.as_ref().ok_or(Exceptions::FORMAT)
}
pub fn readVersion(&mut self) -> Result<VersionRef> {
if let Some(pv) = self.parsedVersion {
return Ok(pv);
}
let dimension = self.bitMatrix.getHeight();
let provisionalVersion = (dimension - 17) / 4;
if provisionalVersion <= 6 {
return Version::getVersionForNumber(provisionalVersion);
}
let mut versionBits = 0;
let ijMin = dimension - 11;
for j in (0..=5).rev() {
for i in (ijMin..(dimension - 8)).rev() {
versionBits = self.copyBit(i, j, versionBits);
}
}
if let Ok(theParsedVersion) = Version::decodeVersionInformation(versionBits) {
if theParsedVersion.getDimensionForVersion() == dimension {
self.parsedVersion = Some(theParsedVersion);
return Ok(theParsedVersion);
}
}
versionBits = 0;
for i in (0..=5).rev() {
for j in (ijMin..(dimension - 5)).rev() {
versionBits = self.copyBit(i, j, versionBits);
}
}
if let Ok(theParsedVersion) = Version::decodeVersionInformation(versionBits) {
if theParsedVersion.getDimensionForVersion() == dimension {
self.parsedVersion = Some(theParsedVersion);
return Ok(theParsedVersion);
}
}
Err(Exceptions::FORMAT)
}
fn copyBit(&self, i: u32, j: u32, versionBits: u32) -> u32 {
let bit = if self.mirror {
self.bitMatrix.get(j, i)
} else {
self.bitMatrix.get(i, j)
};
if bit {
(versionBits << 1) | 0x1
} else {
versionBits << 1
}
}
pub fn readCodewords(&mut self) -> Result<Vec<u8>> {
let version = self.readVersion()?;
let dataMask: DataMask = self.readFormatInformation()?.getDataMask().try_into()?;
let dimension = self.bitMatrix.getHeight();
dataMask.unmaskBitMatrix(&mut self.bitMatrix, dimension);
let functionPattern = version.buildFunctionPattern()?;
let mut readingUp = true;
let mut result = vec![0u8; version.getTotalCodewords() as usize];
let mut resultOffset = 0;
let mut currentByte = 0;
let mut bitsRead = 0;
let mut j = dimension as i32 - 1;
while j > 0 {
if j == 6 {
j -= 1;
}
for count in 0..dimension {
let i = if readingUp {
dimension - 1 - count
} else {
count
};
for col in 0..2 {
if !functionPattern.get(j as u32 - col, i) {
bitsRead += 1;
currentByte <<= 1;
if self.bitMatrix.get(j as u32 - col, i) {
currentByte |= 1;
}
if bitsRead == 8 {
result[resultOffset] = currentByte;
resultOffset += 1;
bitsRead = 0;
currentByte = 0;
}
}
}
}
readingUp ^= true;
j -= 2;
}
if resultOffset != version.getTotalCodewords() as usize {
return Err(Exceptions::FORMAT);
}
Ok(result)
}
pub fn remask(&mut self) -> Result<()> {
if let Some(pfi) = &self.parsedFormatInfo {
let dataMask: DataMask = pfi.getDataMask().try_into()?;
let dimension = self.bitMatrix.getHeight();
dataMask.unmaskBitMatrix(&mut self.bitMatrix, dimension);
} else {
}
Ok(())
}
pub fn setMirror(&mut self, mirror: bool) {
self.parsedVersion = None;
self.parsedFormatInfo = None;
self.mirror = mirror;
}
pub fn mirror(&mut self) {
for x in 0..self.bitMatrix.getWidth() {
for y in (x + 1)..self.bitMatrix.getHeight() {
if self.bitMatrix.get(x, y) != self.bitMatrix.get(y, x) {
self.bitMatrix.flip_coords(y, x);
self.bitMatrix.flip_coords(x, y);
}
}
}
}
}