mirror of
https://gitee.com/akwkevin/aistudio.-wpf.-diagram
synced 2026-04-30 21:23:24 +08:00
196 lines
6.4 KiB
C#
196 lines
6.4 KiB
C#
/*
|
|
* Copyright 2008 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.
|
|
*/
|
|
|
|
using System;
|
|
using System.Text;
|
|
|
|
using ZXing.Common;
|
|
|
|
namespace ZXing.OneD
|
|
{
|
|
/// <summary>
|
|
/// <p>Implements decoding of the UPC-E format.</p>
|
|
/// <p><a href="http://www.barcodeisland.com/upce.phtml">This</a>is a great reference for
|
|
/// UPC-E information.</p>
|
|
/// <author>Sean Owen</author>
|
|
/// </summary>
|
|
public sealed class UPCEReader : UPCEANReader
|
|
{
|
|
/// <summary>
|
|
/// The pattern that marks the middle, and end, of a UPC-E pattern.
|
|
/// There is no "second half" to a UPC-E barcode.
|
|
/// </summary>
|
|
private static readonly int[] MIDDLE_END_PATTERN = { 1, 1, 1, 1, 1, 1 };
|
|
|
|
/// <summary>
|
|
/// See L_AND_G_PATTERNS these values similarly represent patterns of
|
|
/// even-odd parity encodings of digits that imply both the number system (0 or 1)
|
|
/// used, and the check digit.
|
|
/// </summary>
|
|
private static readonly int[][] NUMSYS_AND_CHECK_DIGIT_PATTERNS = {
|
|
new[] { 0x38, 0x34, 0x32, 0x31, 0x2C, 0x26, 0x23, 0x2A, 0x29, 0x25 },
|
|
new[] { 0x07, 0x0B, 0x0D, 0x0E, 0x13, 0x19, 0x1C, 0x15, 0x16, 0x1A }
|
|
};
|
|
|
|
private readonly int[] decodeMiddleCounters;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="UPCEReader"/> class.
|
|
/// </summary>
|
|
public UPCEReader()
|
|
{
|
|
decodeMiddleCounters = new int[4];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Decodes the middle.
|
|
/// </summary>
|
|
/// <param name="row">The row.</param>
|
|
/// <param name="startRange">The start range.</param>
|
|
/// <param name="result">The result.</param>
|
|
/// <returns></returns>
|
|
override internal protected int decodeMiddle(BitArray row, int[] startRange, StringBuilder result)
|
|
{
|
|
int[] counters = decodeMiddleCounters;
|
|
counters[0] = 0;
|
|
counters[1] = 0;
|
|
counters[2] = 0;
|
|
counters[3] = 0;
|
|
int end = row.Size;
|
|
int rowOffset = startRange[1];
|
|
|
|
int lgPatternFound = 0;
|
|
|
|
for (int x = 0; x < 6 && rowOffset < end; x++)
|
|
{
|
|
int bestMatch;
|
|
if (!decodeDigit(row, counters, rowOffset, L_AND_G_PATTERNS, out bestMatch))
|
|
return -1;
|
|
result.Append((char)('0' + bestMatch % 10));
|
|
foreach (int counter in counters)
|
|
{
|
|
rowOffset += counter;
|
|
}
|
|
if (bestMatch >= 10)
|
|
{
|
|
lgPatternFound |= 1 << (5 - x);
|
|
}
|
|
}
|
|
|
|
if (!determineNumSysAndCheckDigit(result, lgPatternFound))
|
|
return -1;
|
|
|
|
return rowOffset;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Decodes the end.
|
|
/// </summary>
|
|
/// <param name="row">The row.</param>
|
|
/// <param name="endStart">The end start.</param>
|
|
/// <returns></returns>
|
|
override protected int[] decodeEnd(BitArray row, int endStart)
|
|
{
|
|
return findGuardPattern(row, endStart, true, MIDDLE_END_PATTERN);
|
|
}
|
|
|
|
/// <summary>
|
|
/// <returns>see checkStandardUPCEANChecksum(String)</returns>
|
|
/// </summary>
|
|
/// <param name="s"></param>
|
|
/// <returns></returns>
|
|
override protected bool checkChecksum(String s)
|
|
{
|
|
return base.checkChecksum(convertUPCEtoUPCA(s));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determines the num sys and check digit.
|
|
/// </summary>
|
|
/// <param name="resultString">The result string.</param>
|
|
/// <param name="lgPatternFound">The lg pattern found.</param>
|
|
/// <returns></returns>
|
|
private static bool determineNumSysAndCheckDigit(StringBuilder resultString, int lgPatternFound)
|
|
{
|
|
|
|
for (int numSys = 0; numSys <= 1; numSys++)
|
|
{
|
|
for (int d = 0; d < 10; d++)
|
|
{
|
|
if (lgPatternFound == NUMSYS_AND_CHECK_DIGIT_PATTERNS[numSys][d])
|
|
{
|
|
resultString.Insert(0, new[] { (char)('0' + numSys) });
|
|
resultString.Append((char)('0' + d));
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the format of this decoder.
|
|
/// <returns>The 1D format.</returns>
|
|
/// </summary>
|
|
override internal BarcodeFormat BarcodeFormat
|
|
{
|
|
get { return BarcodeFormat.UPC_E; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Expands a UPC-E value back into its full, equivalent UPC-A code value.
|
|
///
|
|
/// <param name="upce">UPC-E code as string of digits</param>
|
|
/// <returns>equivalent UPC-A code as string of digits</returns>
|
|
/// </summary>
|
|
public static String convertUPCEtoUPCA(String upce)
|
|
{
|
|
var upceChars = upce.Substring(1, 6);
|
|
StringBuilder result = new StringBuilder(12);
|
|
result.Append(upce[0]);
|
|
char lastChar = upceChars[5];
|
|
switch (lastChar)
|
|
{
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
result.Append(upceChars, 0, 2);
|
|
result.Append(lastChar);
|
|
result.Append("0000");
|
|
result.Append(upceChars, 2, 3);
|
|
break;
|
|
case '3':
|
|
result.Append(upceChars, 0, 3);
|
|
result.Append("00000");
|
|
result.Append(upceChars, 3, 2);
|
|
break;
|
|
case '4':
|
|
result.Append(upceChars, 0, 4);
|
|
result.Append("00000");
|
|
result.Append(upceChars[4]);
|
|
break;
|
|
default:
|
|
result.Append(upceChars, 0, 5);
|
|
result.Append("0000");
|
|
result.Append(lastChar);
|
|
break;
|
|
}
|
|
result.Append(upce[7]);
|
|
return result.ToString();
|
|
}
|
|
}
|
|
}
|