rxing 0.8.2

A rust port of the zxing barcode library.
Documentation
/*
 * Copyright 2009 ZXing authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

use rxing_one_d_proc_derive::OneDWriter;

use crate::{
    common::Result,
    oned::{EAN8Reader, UPCEANReader},
    BarcodeFormat,
};

use super::{upc_ean_reader, OneDimensionalCodeWriter, UPCEANWriter};

/**
 * This object renders an EAN8 code as a {@link BitMatrix}.
 *
 * @author aripollak@gmail.com (Ari Pollak)
 */
#[derive(OneDWriter, Default)]
pub struct EAN8Writer;

const CODE_WIDTH: usize = 3 + // start guard
      (7 * 4) + // left bars
      5 + // middle guard
      (7 * 4) + // right bars
      3; // end guard

impl UPCEANWriter for EAN8Writer {}
impl OneDimensionalCodeWriter for EAN8Writer {
    /**
     * @return a byte array of horizontal pixels (false = white, true = black)
     */
    fn encode_oned(&self, contents: &str) -> Result<Vec<bool>> {
        let length = contents.chars().count();
        let reader = EAN8Reader;
        let mut contents = contents.to_owned();
        match length {
            7 => {
                // No check digit present, calculate it and add it
                let check =
                    reader.getStandardUPCEANChecksum(&contents.chars().collect::<Vec<_>>())?;

                contents.push_str(&check.to_string());
            }
            8 => {
                if !EAN8Reader.checkStandardUPCEANChecksum(&contents)? {
                    return Err(Exceptions::illegal_argument_with(
                        "Contents do not pass checksum",
                    ));
                }
            }
            _ => {
                return Err(Exceptions::illegal_argument_with(format!(
                    "Requested contents should be 7 or 8 digits long, but got {length}"
                )))
            }
        }

        Self::checkNumeric(&contents)?;

        let mut result = [false; CODE_WIDTH];
        let mut pos = 0;

        pos += Self::appendPattern(&mut result, pos, &upc_ean_reader::START_END_PATTERN, true)
            as usize;

        for i in 0..=3 {
            // for (int i = 0; i <= 3; i++) {
            let digit = contents
                .chars()
                .nth(i)
                .ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)?
                .to_digit(10)
                .ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)? as usize;
            pos += Self::appendPattern(&mut result, pos, &upc_ean_reader::L_PATTERNS[digit], false)
                as usize;
        }

        pos +=
            Self::appendPattern(&mut result, pos, &upc_ean_reader::MIDDLE_PATTERN, false) as usize;

        for i in 4..=7 {
            // for (int i = 4; i <= 7; i++) {
            let digit = contents
                .chars()
                .nth(i)
                .ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)?
                .to_digit(10)
                .ok_or(Exceptions::INDEX_OUT_OF_BOUNDS)? as usize;
            pos += Self::appendPattern(&mut result, pos, &upc_ean_reader::L_PATTERNS[digit], true)
                as usize;
        }
        Self::appendPattern(&mut result, pos, &upc_ean_reader::START_END_PATTERN, true);

        Ok(result.to_vec())
    }

    fn getSupportedWriteFormats(&self) -> Option<Vec<crate::BarcodeFormat>> {
        Some(vec![BarcodeFormat::EAN_8])
    }

    fn getDefaultMargin(&self) -> u32 {
        Self::DEFAULT_MARGIN
    }
}

/**
 * @author Ari Pollak
 */
#[cfg(test)]
mod EAN8WriterTestCase {
    use crate::{common::bit_matrix_test_case, BarcodeFormat, Writer};

    use super::EAN8Writer;

    #[test]
    fn testEncode() {
        let testStr =
            "0000001010001011010111101111010110111010101001110111001010001001011100101000000";
        let result = EAN8Writer
            .encode(
                "96385074",
                &BarcodeFormat::EAN_8,
                testStr.chars().count() as i32,
                0,
            )
            .expect("ok");
        assert_eq!(testStr, bit_matrix_test_case::matrix_to_string(&result));
    }

    #[test]
    fn testAddChecksumAndEncode() {
        let testStr =
            "0000001010001011010111101111010110111010101001110111001010001001011100101000000";
        let result = EAN8Writer
            .encode(
                "9638507",
                &BarcodeFormat::EAN_8,
                testStr.chars().count() as i32,
                0,
            )
            .expect("ok");
        assert_eq!(testStr, bit_matrix_test_case::matrix_to_string(&result));
    }

    #[test]
    #[should_panic]
    fn testEncodeIllegalCharacters() {
        EAN8Writer
            .encode("96385abc", &BarcodeFormat::EAN_8, 0, 0)
            .expect("ok");
    }
}