1 // Copyright 2014 PDFium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com 6 // Original code is licensed as follows: 7 /* 8 * Copyright 2008 ZXing authors 9 * 10 * Licensed under the Apache License, Version 2.0 (the "License"); 11 * you may not use this file except in compliance with the License. 12 * You may obtain a copy of the License at 13 * 14 * http://www.apache.org/licenses/LICENSE-2.0 15 * 16 * Unless required by applicable law or agreed to in writing, software 17 * distributed under the License is distributed on an "AS IS" BASIS, 18 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 19 * See the License for the specific language governing permissions and 20 * limitations under the License. 21 */ 22 23 #include "fxbarcode/common/BC_CommonByteMatrix.h" 24 #include "fxbarcode/qrcode/BC_QRCoder.h" 25 #include "fxbarcode/qrcode/BC_QRCoderErrorCorrectionLevel.h" 26 #include "fxbarcode/qrcode/BC_QRCoderMaskUtil.h" 27 #include "fxbarcode/utils.h" 28 29 CBC_QRCoderMaskUtil::CBC_QRCoderMaskUtil() {} 30 CBC_QRCoderMaskUtil::~CBC_QRCoderMaskUtil() {} 31 int32_t CBC_QRCoderMaskUtil::ApplyMaskPenaltyRule1( 32 CBC_CommonByteMatrix* matrix) { 33 return ApplyMaskPenaltyRule1Internal(matrix, true) + 34 ApplyMaskPenaltyRule1Internal(matrix, false); 35 } 36 37 int32_t CBC_QRCoderMaskUtil::ApplyMaskPenaltyRule2( 38 CBC_CommonByteMatrix* matrix) { 39 int32_t penalty = 0; 40 uint8_t* array = matrix->GetArray(); 41 int32_t width = matrix->GetWidth(); 42 int32_t height = matrix->GetHeight(); 43 for (int32_t y = 0; y < height - 1; y++) { 44 for (int32_t x = 0; x < width - 1; x++) { 45 int32_t value = array[y * width + x]; 46 if (value == array[y * width + x + 1] && 47 value == array[(y + 1) * width + x] && 48 value == array[(y + 1) * width + x + 1]) { 49 penalty++; 50 } 51 } 52 } 53 return 3 * penalty; 54 } 55 56 int32_t CBC_QRCoderMaskUtil::ApplyMaskPenaltyRule3( 57 CBC_CommonByteMatrix* matrix) { 58 int32_t penalty = 0; 59 uint8_t* array = matrix->GetArray(); 60 int32_t width = matrix->GetWidth(); 61 int32_t height = matrix->GetHeight(); 62 for (int32_t y = 0; y < height; ++y) { 63 for (int32_t x = 0; x < width; ++x) { 64 if (x == 0 && 65 ((y >= 0 && y <= 6) || (y >= height - 7 && y <= height - 1))) { 66 continue; 67 } 68 if (x == width - 7 && (y >= 0 && y <= 6)) { 69 continue; 70 } 71 if (y == 0 && 72 ((x >= 0 && x <= 6) || (x >= width - 7 && x <= width - 1))) { 73 continue; 74 } 75 if (y == height - 7 && (x >= 0 && x <= 6)) { 76 continue; 77 } 78 if (x + 6 < width && array[y * width + x] == 1 && 79 array[y * width + x + 1] == 0 && array[y * width + x + 2] == 1 && 80 array[y * width + x + 3] == 1 && array[y * width + x + 4] == 1 && 81 array[y * width + x + 5] == 0 && array[y * width + x + 6] == 1 && 82 ((x + 10 < width && array[y * width + x + 7] == 0 && 83 array[y * width + x + 8] == 0 && array[y * width + x + 9] == 0 && 84 array[y * width + x + 10] == 0) || 85 (x - 4 >= 0 && array[y * width + x - 1] == 0 && 86 array[y * width + x - 2] == 0 && array[y * width + x - 3] == 0 && 87 array[y * width + x - 4] == 0))) { 88 penalty += 40; 89 } 90 if (y + 6 < height && array[y * width + x] == 1 && 91 array[(y + 1) * width + x] == 0 && array[(y + 2) * width + x] == 1 && 92 array[(y + 3) * width + x] == 1 && array[(y + 4) * width + x] == 1 && 93 array[(y + 5) * width + x] == 0 && array[(y + 6) * width + x] == 1 && 94 ((y + 10 < height && array[(y + 7) * width + x] == 0 && 95 array[(y + 8) * width + x] == 0 && 96 array[(y + 9) * width + x] == 0 && 97 array[(y + 10) * width + x] == 0) || 98 (y - 4 >= 0 && array[(y - 1) * width + x] == 0 && 99 array[(y - 2) * width + x] == 0 && 100 array[(y - 3) * width + x] == 0 && 101 array[(y - 4) * width + x] == 0))) { 102 penalty += 40; 103 } 104 } 105 } 106 return penalty; 107 } 108 int32_t CBC_QRCoderMaskUtil::ApplyMaskPenaltyRule4( 109 CBC_CommonByteMatrix* matrix) { 110 int32_t numDarkCells = 0; 111 uint8_t* array = matrix->GetArray(); 112 int32_t width = matrix->GetWidth(); 113 int32_t height = matrix->GetHeight(); 114 for (int32_t y = 0; y < height; ++y) { 115 for (int32_t x = 0; x < width; ++x) { 116 if (array[y * width + x] == 1) { 117 numDarkCells += 1; 118 } 119 } 120 } 121 int32_t numTotalCells = matrix->GetHeight() * matrix->GetWidth(); 122 double darkRatio = (double)numDarkCells / numTotalCells; 123 return abs((int32_t)(darkRatio * 100 - 50) / 5) * 5 * 10; 124 } 125 bool CBC_QRCoderMaskUtil::GetDataMaskBit(int32_t maskPattern, 126 int32_t x, 127 int32_t y, 128 int32_t& e) { 129 if (!CBC_QRCoder::IsValidMaskPattern(maskPattern)) { 130 e = (BCExceptionInvalidateMaskPattern); 131 return false; 132 } 133 int32_t intermediate = 0, temp = 0; 134 switch (maskPattern) { 135 case 0: 136 intermediate = (y + x) & 0x1; 137 break; 138 case 1: 139 intermediate = y & 0x1; 140 break; 141 case 2: 142 intermediate = x % 3; 143 break; 144 case 3: 145 intermediate = (y + x) % 3; 146 break; 147 case 4: 148 intermediate = ((y >> 1) + (x / 3)) & 0x1; 149 break; 150 case 5: 151 temp = y * x; 152 intermediate = (temp & 0x1) + (temp % 3); 153 break; 154 case 6: 155 temp = y * x; 156 intermediate = (((temp & 0x1) + (temp % 3)) & 0x1); 157 break; 158 case 7: 159 temp = y * x; 160 intermediate = (((temp % 3) + ((y + x) & 0x1)) & 0x1); 161 break; 162 default: { 163 e = BCExceptionInvalidateMaskPattern; 164 return false; 165 } 166 } 167 return intermediate == 0; 168 } 169 int32_t CBC_QRCoderMaskUtil::ApplyMaskPenaltyRule1Internal( 170 CBC_CommonByteMatrix* matrix, 171 bool isHorizontal) { 172 int32_t penalty = 0; 173 int32_t numSameBitCells = 0; 174 int32_t prevBit = -1; 175 int32_t width = matrix->GetWidth(); 176 int32_t height = matrix->GetHeight(); 177 int32_t iLimit = isHorizontal ? height : width; 178 int32_t jLimit = isHorizontal ? width : height; 179 uint8_t* array = matrix->GetArray(); 180 for (int32_t i = 0; i < iLimit; ++i) { 181 for (int32_t j = 0; j < jLimit; ++j) { 182 int32_t bit = isHorizontal ? array[i * width + j] : array[j * width + i]; 183 if (bit == prevBit) { 184 numSameBitCells += 1; 185 if (numSameBitCells == 5) { 186 penalty += 3; 187 } else if (numSameBitCells > 5) { 188 penalty += 1; 189 } 190 } else { 191 numSameBitCells = 1; 192 prevBit = bit; 193 } 194 } 195 numSameBitCells = 0; 196 } 197 return penalty; 198 } 199