示例#1
0
        // Initialize "qrCode" according to "numInputBytes", "ecLevel", and "mode". On success, modify
        // "qrCode".
        private static void initQRCode(int numInputBytes, ErrorCorrectionLevel ecLevel, Mode mode, QRCode qrCode)
        {
            try
            {
                qrCode.setECLevel(ecLevel);
                qrCode.setMode(mode);

                // In the following comments, we use numbers of Version 7-H.
                for (int versionNum = 1; versionNum <= 40; versionNum++)
                {
                    Version version = Version.getVersionForNumber(versionNum);
                    // numBytes = 196
                    int numBytes = version.getTotalCodewords();
                    // getNumECBytes = 130
                    Version.ECBlocks ecBlocks = version.getECBlocksForLevel(ecLevel);
                    int numEcBytes            = ecBlocks.getTotalECCodewords();
                    // getNumRSBlocks = 5
                    int numRSBlocks = ecBlocks.getNumBlocks();
                    // getNumDataBytes = 196 - 130 = 66
                    int numDataBytes = numBytes - numEcBytes;
                    // We want to choose the smallest version which can contain data of "numInputBytes" + some
                    // extra bits for the header (mode info and length info). The header can be three bytes
                    // (precisely 4 + 16 bits) at most. Hence we do +3 here.
                    if (numDataBytes >= numInputBytes + 3)
                    {
                        // Yay, we found the proper rs block info!
                        qrCode.setVersion(versionNum);
                        qrCode.setNumTotalBytes(numBytes);
                        qrCode.setNumDataBytes(numDataBytes);
                        qrCode.setNumRSBlocks(numRSBlocks);
                        // getNumECBytes = 196 - 66 = 130
                        qrCode.setNumECBytes(numEcBytes);
                        // matrix width = 21 + 6 * 4 = 45
                        qrCode.setMatrixWidth(version.getDimensionForVersion());
                        return;
                    }
                }
                throw new WriterException("Cannot find proper rs block info (input data too big?)");
            }
            catch (Exception e) {
                throw new WriterException(e.Message);
            }
        }
示例#2
0
//JAVA TO C# CONVERTER WARNING: Method 'throws' clauses are not available in .NET:
//ORIGINAL LINE: private static com.google.zxing.qrcode.decoder.Version chooseVersion(int numInputBits, com.google.zxing.qrcode.decoder.ErrorCorrectionLevel ecLevel) throws com.google.zxing.WriterException
        private static Version chooseVersion(int numInputBits, ErrorCorrectionLevel ecLevel)
        {
            // In the following comments, we use numbers of Version 7-H.
            for (int versionNum = 1; versionNum <= 40; versionNum++)
            {
                Version version = Version.getVersionForNumber(versionNum);
                // numBytes = 196
                int numBytes = version.TotalCodewords;
                // getNumECBytes = 130
                Version.ECBlocks ecBlocks = version.getECBlocksForLevel(ecLevel);
                int numEcBytes            = ecBlocks.TotalECCodewords;
                // getNumDataBytes = 196 - 130 = 66
                int numDataBytes    = numBytes - numEcBytes;
                int totalInputBytes = (numInputBits + 7) / 8;
                if (numDataBytes >= totalInputBytes)
                {
                    return(version);
                }
            }
            throw new WriterException("Data too big");
        }
示例#3
0
        /// <summary> <p>When QR Codes use multiple data blocks, they are actually interleaved.
        /// That is, the first byte of data block 1 to n is written, then the second bytes, and so on. This
        /// method will separate the data into original blocks.</p>
        ///
        /// </summary>
        /// <param name="rawCodewords">bytes as read directly from the QR Code
        /// </param>
        /// <param name="version">version of the QR Code
        /// </param>
        /// <param name="ecLevel">error-correction level of the QR Code
        /// </param>
        /// <returns> {@link DataBlock}s containing original bytes, "de-interleaved" from representation in the
        /// QR Code
        /// </returns>
        internal static DataBlock[] getDataBlocks(sbyte[] rawCodewords, Version version, com.google.zxing.qrcode.decoder.ErrorCorrectionLevel ecLevel)
        {
            if (rawCodewords.Length != version.TotalCodewords)
            {
                throw new System.ArgumentException();
            }

            // Figure out the number and size of data blocks used by this version and
            // error correction level
            Version.ECBlocks ecBlocks = version.getECBlocksForLevel(ecLevel);

            // First count the total number of data blocks
            int totalBlocks = 0;

            Version.ECB[] ecBlockArray = ecBlocks.getECBlocks();
            for (int i = 0; i < ecBlockArray.Length; i++)
            {
                totalBlocks += ecBlockArray[i].Count;
            }

            // Now establish DataBlocks of the appropriate size and number of data codewords
            DataBlock[] result          = new DataBlock[totalBlocks];
            int         numResultBlocks = 0;

            for (int j = 0; j < ecBlockArray.Length; j++)
            {
                Version.ECB ecBlock = ecBlockArray[j];
                for (int i = 0; i < ecBlock.Count; i++)
                {
                    int numDataCodewords  = ecBlock.DataCodewords;
                    int numBlockCodewords = ecBlocks.ECCodewordsPerBlock + numDataCodewords;
                    result[numResultBlocks++] = new DataBlock(numDataCodewords, new sbyte[numBlockCodewords]);
                }
            }

            // All blocks have the same amount of data, except that the last n
            // (where n may be 0) have 1 more byte. Figure out where these start.
            int shorterBlocksTotalCodewords = result[0].codewords.Length;
            int longerBlocksStartAt         = result.Length - 1;

            while (longerBlocksStartAt >= 0)
            {
                int numCodewords = result[longerBlocksStartAt].codewords.Length;
                if (numCodewords == shorterBlocksTotalCodewords)
                {
                    break;
                }
                longerBlocksStartAt--;
            }
            longerBlocksStartAt++;

            int shorterBlocksNumDataCodewords = shorterBlocksTotalCodewords - ecBlocks.ECCodewordsPerBlock;
            // The last elements of result may be 1 element longer;
            // first fill out as many elements as all of them have
            int rawCodewordsOffset = 0;

            for (int i = 0; i < shorterBlocksNumDataCodewords; i++)
            {
                for (int j = 0; j < numResultBlocks; j++)
                {
                    result[j].codewords[i] = rawCodewords[rawCodewordsOffset++];
                }
            }
            // Fill out the last data block in the longer ones
            for (int j = longerBlocksStartAt; j < numResultBlocks; j++)
            {
                result[j].codewords[shorterBlocksNumDataCodewords] = rawCodewords[rawCodewordsOffset++];
            }
            // Now add in error correction blocks
            int max = result[0].codewords.Length;

            for (int i = shorterBlocksNumDataCodewords; i < max; i++)
            {
                for (int j = 0; j < numResultBlocks; j++)
                {
                    int iOffset = j < longerBlocksStartAt?i:i + 1;
                    result[j].codewords[iOffset] = rawCodewords[rawCodewordsOffset++];
                }
            }
            return(result);
        }
示例#4
0
        /// <summary>
        /// <p>When Data Matrix Codes use multiple data blocks, they actually interleave the bytes of each of them.
        /// That is, the first byte of data block 1 to n is written, then the second bytes, and so on. This
        /// method will separate the data into original blocks.</p>
        /// </summary>
        /// <param name="rawCodewords"> bytes as read directly from the Data Matrix Code </param>
        /// <param name="version"> version of the Data Matrix Code </param>
        /// <returns> DataBlocks containing original bytes, "de-interleaved" from representation in the
        ///         Data Matrix Code </returns>
        internal static DataBlock[] getDataBlocks(sbyte[] rawCodewords, Version version)
        {
            // Figure out the number and size of data blocks used by this version
            Version.ECBlocks ecBlocks = version.ECBlocks2;

            // First count the total number of data blocks
            int totalBlocks = 0;

            Version.ECB[] ecBlockArray = ecBlocks.GetECB;
            foreach (Version.ECB ecBlock in ecBlockArray)
            {
                totalBlocks += ecBlock.Count;
            }

            // Now establish DataBlocks of the appropriate size and number of data codewords
            DataBlock[] result          = new DataBlock[totalBlocks];
            int         numResultBlocks = 0;

            foreach (Version.ECB ecBlock in ecBlockArray)
            {
                for (int i = 0; i < ecBlock.Count; i++)
                {
                    int numDataCodewords  = ecBlock.DataCodewords;
                    int numBlockCodewords = ecBlocks.ECCodewords + numDataCodewords;
                    result[numResultBlocks++] = new DataBlock(numDataCodewords, new sbyte[numBlockCodewords]);
                }
            }

            // All blocks have the same amount of data, except that the last n
            // (where n may be 0) have 1 less byte. Figure out where these start.
            // TODO(bbrown): There is only one case where there is a difference for Data Matrix for size 144
            int longerBlocksTotalCodewords = result[0].codewords.Length;
            //int shorterBlocksTotalCodewords = longerBlocksTotalCodewords - 1;

            int longerBlocksNumDataCodewords  = longerBlocksTotalCodewords - ecBlocks.ECCodewords;
            int shorterBlocksNumDataCodewords = longerBlocksNumDataCodewords - 1;
            // The last elements of result may be 1 element shorter for 144 matrix
            // first fill out as many elements as all of them have minus 1
            int rawCodewordsOffset = 0;

            for (int i = 0; i < shorterBlocksNumDataCodewords; i++)
            {
                for (int j = 0; j < numResultBlocks; j++)
                {
                    result[j].codewords[i] = rawCodewords[rawCodewordsOffset++];
                }
            }

            // Fill out the last data block in the longer ones
            bool specialVersion  = version.VersionNumber == 24;
            int  numLongerBlocks = specialVersion ? 8 : numResultBlocks;

            for (int j = 0; j < numLongerBlocks; j++)
            {
                result[j].codewords[longerBlocksNumDataCodewords - 1] = rawCodewords[rawCodewordsOffset++];
            }

            // Now add in error correction blocks
            int max = result[0].codewords.Length;

            for (int i = longerBlocksNumDataCodewords; i < max; i++)
            {
                for (int j = 0; j < numResultBlocks; j++)
                {
                    int iOffset = specialVersion && j > 7 ? i - 1 : i;
                    result[j].codewords[iOffset] = rawCodewords[rawCodewordsOffset++];
                }
            }

            if (rawCodewordsOffset != rawCodewords.Length)
            {
                throw new System.ArgumentException();
            }

            return(result);
        }
示例#5
0
//JAVA TO C# CONVERTER WARNING: Method 'throws' clauses are not available in .NET:
//ORIGINAL LINE: public static QRCode encode(String content, com.google.zxing.qrcode.decoder.ErrorCorrectionLevel ecLevel, java.util.Map<com.google.zxing.EncodeHintType,?> hints) throws com.google.zxing.WriterException
        public static QRCode encode(string content, ErrorCorrectionLevel ecLevel, IDictionary <EncodeHintType, object> hints)
        {
            // Determine what character encoding has been specified by the caller, if any
            //string encoding = hints == null ? null : (string) hints[EncodeHintType.CHARACTER_SET];
            string encoding = null;

            if (hints != null && hints.ContainsKey(EncodeHintType.CHARACTER_SET))
            {
                encoding = (string)hints[EncodeHintType.CHARACTER_SET];
            }

            if (encoding == null)
            {
                encoding = DEFAULT_BYTE_MODE_ENCODING;
            }

            // Pick an encoding mode appropriate for the content. Note that this will not attempt to use
            // multiple modes / segments even if that were more efficient. Twould be nice.
            Mode mode = chooseMode(content, encoding);

            // This will store the header information, like mode and
            // length, as well as "header" segments like an ECI segment.
            BitArray headerBits = new BitArray();

            // Append ECI segment if applicable
            if (mode == Mode.BYTE && !DEFAULT_BYTE_MODE_ENCODING.Equals(encoding))
            {
                CharacterSetECI eci = CharacterSetECI.getCharacterSetECIByName(encoding);
                if (eci != null)
                {
                    appendECI(eci, headerBits);
                }
            }

            // (With ECI in place,) Write the mode marker
            appendModeInfo(mode, headerBits);

            // Collect data within the main segment, separately, to count its size if needed. Don't add it to
            // main payload yet.
            BitArray dataBits = new BitArray();

            appendBytes(content, mode, dataBits, encoding);

            // Hard part: need to know version to know how many bits length takes. But need to know how many
            // bits it takes to know version. First we take a guess at version by assuming version will be
            // the minimum, 1:

            int     provisionalBitsNeeded = headerBits.Size + mode.getCharacterCountBits(Version.getVersionForNumber(1)) + dataBits.Size;
            Version provisionalVersion    = chooseVersion(provisionalBitsNeeded, ecLevel);

            // Use that guess to calculate the right version. I am still not sure this works in 100% of cases.

            int     bitsNeeded = headerBits.Size + mode.getCharacterCountBits(provisionalVersion) + dataBits.Size;
            Version version    = chooseVersion(bitsNeeded, ecLevel);

            BitArray headerAndDataBits = new BitArray();

            headerAndDataBits.appendBitArray(headerBits);
            // Find "length" of main segment and write it
            int numLetters = mode == Mode.BYTE ? dataBits.SizeInBytes : content.Length;

            appendLengthInfo(numLetters, version, mode, headerAndDataBits);
            // Put data together into the overall payload
            headerAndDataBits.appendBitArray(dataBits);

            Version.ECBlocks ecBlocks = version.getECBlocksForLevel(ecLevel);
            int numDataBytes          = version.TotalCodewords - ecBlocks.TotalECCodewords;

            // Terminate the bits properly.
            terminateBits(numDataBytes, headerAndDataBits);

            // Interleave data bits with error correction code.
            BitArray finalBits = interleaveWithECBytes(headerAndDataBits, version.TotalCodewords, numDataBytes, ecBlocks.NumBlocks);

            QRCode qrCode = new QRCode();

            qrCode.ECLevel = ecLevel;
            qrCode.Mode    = mode;
            qrCode.Version = version;

            //  Choose the mask pattern and set to "qrCode".
            int        dimension   = version.DimensionForVersion;
            ByteMatrix matrix      = new ByteMatrix(dimension, dimension);
            int        maskPattern = chooseMaskPattern(finalBits, ecLevel, version, matrix);

            qrCode.MaskPattern = maskPattern;

            // Build the matrix and set it to "qrCode".
            MatrixUtil.buildMatrix(finalBits, ecLevel, version, maskPattern, matrix);
            qrCode.Matrix = matrix;

            return(qrCode);
        }