/*
 * Copyright 2013 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.
 */

namespace ZXing.PDF417.Internal
{
   /// <summary>
   /// 
   /// </summary>
   /// <author>Guenther Grau</author>
   /// <author>creatale GmbH ([email protected])</author>
   public static class PDF417CodewordDecoder
   {
      /// <summary>
      /// The ratios table
      /// </summary>
      private static readonly float[][] RATIOS_TABLE; // = new float[PDF417Common.SYMBOL_TABLE.Length][PDF417Common.BARS_IN_MODULE];

      /// <summary>
      /// Initializes the <see cref="ZXing.PDF417.Internal.PDF417CodewordDecoder"/> class & Pre-computes the symbol ratio table.
      /// </summary>
      static PDF417CodewordDecoder()
      {
         // Jagged arrays in Java assign the memory automatically, but C# has no equivalent. (Jon Skeet says so!)
         // http://stackoverflow.com/a/5313879/266252
         RATIOS_TABLE = new float[PDF417Common.SYMBOL_TABLE.Length][];
         for (int s = 0; s < RATIOS_TABLE.Length; s++)
         {
            RATIOS_TABLE[s] = new float[PDF417Common.BARS_IN_MODULE];
         }

         // Pre-computes the symbol ratio table.
         for (int i = 0; i < PDF417Common.SYMBOL_TABLE.Length; i++)
         {
            int currentSymbol = PDF417Common.SYMBOL_TABLE[i];
            int currentBit = currentSymbol & 0x1;
            for (int j = 0; j < PDF417Common.BARS_IN_MODULE; j++)
            {
               float size = 0.0f;
               while ((currentSymbol & 0x1) == currentBit)
               {
                  size += 1.0f;
                  currentSymbol >>= 1;
               }
               currentBit = currentSymbol & 0x1;
               RATIOS_TABLE[i][PDF417Common.BARS_IN_MODULE - j - 1] = size/PDF417Common.MODULES_IN_CODEWORD;
            }
         }
      }

      /// <summary>
      /// Gets the decoded value.
      /// </summary>
      /// <returns>The decoded value.</returns>
      /// <param name="moduleBitCount">Module bit count.</param>
      public static int getDecodedValue(int[] moduleBitCount)
      {
         int decodedValue = getDecodedCodewordValue(sampleBitCounts(moduleBitCount));
         if (decodedValue != PDF417Common.INVALID_CODEWORD)
         {
            return decodedValue;
         }
         return getClosestDecodedValue(moduleBitCount);
      }

      /// <summary>
      /// Samples the bit counts.
      /// </summary>
      /// <returns>The bit counts.</returns>
      /// <param name="moduleBitCount">Module bit count.</param>
      private static int[] sampleBitCounts(int[] moduleBitCount)
      {
         float bitCountSum = ZXing.Common.Detector.MathUtils.sum(moduleBitCount);
         int[] result = new int[PDF417Common.BARS_IN_MODULE];
         int bitCountIndex = 0;
         int sumPreviousBits = 0;
         for (int i = 0; i < PDF417Common.MODULES_IN_CODEWORD; i++)
         {
            float sampleIndex =
               bitCountSum/(2*PDF417Common.MODULES_IN_CODEWORD) +
               (i*bitCountSum)/PDF417Common.MODULES_IN_CODEWORD;
            if (sumPreviousBits + moduleBitCount[bitCountIndex] <= sampleIndex)
            {
               sumPreviousBits += moduleBitCount[bitCountIndex];
               bitCountIndex++;
            }
            result[bitCountIndex]++;
         }
         return result;
      }

      /// <summary>
      /// Gets the decoded codeword value.
      /// </summary>
      /// <returns>The decoded codeword value.</returns>
      /// <param name="moduleBitCount">Module bit count.</param>
      private static int getDecodedCodewordValue(int[] moduleBitCount)
      {
         int decodedValue = getBitValue(moduleBitCount);
         return PDF417Common.getCodeword(decodedValue) == PDF417Common.INVALID_CODEWORD ? PDF417Common.INVALID_CODEWORD : decodedValue;
      }

      /// <summary>
      /// Gets the bit value.
      /// </summary>
      /// <returns>The bit value.</returns>
      /// <param name="moduleBitCount">Module bit count.</param>
      private static int getBitValue(int[] moduleBitCount)
      {
         ulong result = 0;
         for (ulong i = 0; i < (ulong) moduleBitCount.Length; i++)
         {
            for (int bit = 0; bit < moduleBitCount[i]; bit++)
            {
               result = (result << 1) | (i%2ul == 0ul ? 1ul : 0ul); // C# was warning about using the bit-wise 'OR' here with a mix of int/longs.
            }
         }
         return (int) result;
      }

      /// <summary>
      /// Gets the closest decoded value.
      /// </summary>
      /// <returns>The closest decoded value.</returns>
      /// <param name="moduleBitCount">Module bit count.</param>
      private static int getClosestDecodedValue(int[] moduleBitCount)
      {
         int bitCountSum = ZXing.Common.Detector.MathUtils.sum(moduleBitCount);
         float[] bitCountRatios = new float[PDF417Common.BARS_IN_MODULE];
         if (bitCountSum > 1)
         {
            for (int i = 0; i < bitCountRatios.Length; i++)
            {
               bitCountRatios[i] = moduleBitCount[i] / (float)bitCountSum;
            }
         }
         float bestMatchError = float.MaxValue;
         int bestMatch = PDF417Common.INVALID_CODEWORD;
         for (int j = 0; j < RATIOS_TABLE.Length; j++)
         {
            float error = 0.0f;
            float[] ratioTableRow = RATIOS_TABLE[j];
            for (int k = 0; k < PDF417Common.BARS_IN_MODULE; k++)
            {
               float diff = ratioTableRow[k] - bitCountRatios[k];
               error += diff*diff;
               if (error >= bestMatchError)
               {
                  break;
               }
            }
            if (error < bestMatchError)
            {
               bestMatchError = error;
               bestMatch = PDF417Common.SYMBOL_TABLE[j];
            }
         }
         return bestMatch;
      }
   }
}