MaskPatternTest.php 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. <?php
  2. /**
  3. * Class MaskPatternTest
  4. *
  5. * @created 21.11.2021
  6. * @author ZXing Authors
  7. * @author smiley <smiley@chillerlan.net>
  8. * @copyright 2021 smiley
  9. * @license Apache-2.0
  10. *
  11. * @codingStandardsIgnoreFile Squiz.Arrays.ArrayDeclaration.NoCommaAfterLast
  12. */
  13. namespace chillerlan\QRCodeTest\Common;
  14. use chillerlan\QRCode\QRCodeException;
  15. use chillerlan\QRCode\Common\MaskPattern;
  16. use PHPUnit\Framework\Attributes\DataProvider;
  17. use PHPUnit\Framework\TestCase;
  18. use Closure;
  19. /**
  20. * @see https://github.com/zxing/zxing/blob/f4f3c2971dc794346d8b6e14752200008cb90716/core/src/test/java/com/google/zxing/qrcode/encoder/MaskUtilTestCase.java
  21. */
  22. final class MaskPatternTest extends TestCase{
  23. // See mask patterns on the page 43 of JISX0510:2004.
  24. public static function maskPatternProvider():array{
  25. return [
  26. 'PATTERN_000' => [MaskPattern::PATTERN_000, [
  27. [1, 0, 1, 0, 1, 0],
  28. [0, 1, 0, 1, 0, 1],
  29. [1, 0, 1, 0, 1, 0],
  30. [0, 1, 0, 1, 0, 1],
  31. [1, 0, 1, 0, 1, 0],
  32. [0, 1, 0, 1, 0, 1],
  33. ]],
  34. 'PATTERN_001' => [MaskPattern::PATTERN_001, [
  35. [1, 1, 1, 1, 1, 1],
  36. [0, 0, 0, 0, 0, 0],
  37. [1, 1, 1, 1, 1, 1],
  38. [0, 0, 0, 0, 0, 0],
  39. [1, 1, 1, 1, 1, 1],
  40. [0, 0, 0, 0, 0, 0],
  41. ]],
  42. 'PATTERN_010' => [MaskPattern::PATTERN_010, [
  43. [1, 0, 0, 1, 0, 0],
  44. [1, 0, 0, 1, 0, 0],
  45. [1, 0, 0, 1, 0, 0],
  46. [1, 0, 0, 1, 0, 0],
  47. [1, 0, 0, 1, 0, 0],
  48. [1, 0, 0, 1, 0, 0],
  49. ]],
  50. 'PATTERN_011' => [MaskPattern::PATTERN_011, [
  51. [1, 0, 0, 1, 0, 0],
  52. [0, 0, 1, 0, 0, 1],
  53. [0, 1, 0, 0, 1, 0],
  54. [1, 0, 0, 1, 0, 0],
  55. [0, 0, 1, 0, 0, 1],
  56. [0, 1, 0, 0, 1, 0],
  57. ]],
  58. 'PATTERN_100' => [MaskPattern::PATTERN_100, [
  59. [1, 1, 1, 0, 0, 0],
  60. [1, 1, 1, 0, 0, 0],
  61. [0, 0, 0, 1, 1, 1],
  62. [0, 0, 0, 1, 1, 1],
  63. [1, 1, 1, 0, 0, 0],
  64. [1, 1, 1, 0, 0, 0],
  65. ]],
  66. 'PATTERN_101' => [MaskPattern::PATTERN_101, [
  67. [1, 1, 1, 1, 1, 1],
  68. [1, 0, 0, 0, 0, 0],
  69. [1, 0, 0, 1, 0, 0],
  70. [1, 0, 1, 0, 1, 0],
  71. [1, 0, 0, 1, 0, 0],
  72. [1, 0, 0, 0, 0, 0],
  73. ]],
  74. 'PATTERN_110' => [MaskPattern::PATTERN_110, [
  75. [1, 1, 1, 1, 1, 1],
  76. [1, 1, 1, 0, 0, 0],
  77. [1, 1, 0, 1, 1, 0],
  78. [1, 0, 1, 0, 1, 0],
  79. [1, 0, 1, 1, 0, 1],
  80. [1, 0, 0, 0, 1, 1],
  81. ]],
  82. 'PATTERN_111' => [MaskPattern::PATTERN_111, [
  83. [1, 0, 1, 0, 1, 0],
  84. [0, 0, 0, 1, 1, 1],
  85. [1, 0, 0, 0, 1, 1],
  86. [0, 1, 0, 1, 0, 1],
  87. [1, 1, 1, 0, 0, 0],
  88. [0, 1, 1, 1, 0, 0],
  89. ]],
  90. ];
  91. }
  92. /**
  93. * Tests if the mask function generates the correct pattern
  94. */
  95. #[DataProvider('maskPatternProvider')]
  96. public function testMask(int $pattern, array $expected):void{
  97. $maskPattern = new MaskPattern($pattern);
  98. $this::assertTrue($this->assertMask($maskPattern->getMask(), $expected));
  99. }
  100. private function assertMask(Closure $mask, array $expected):bool{
  101. for($x = 0; $x < 6; $x++){
  102. for($y = 0; $y < 6; $y++){
  103. if($mask($x, $y) !== ($expected[$y][$x] === 1)){
  104. return false;
  105. }
  106. }
  107. }
  108. return true;
  109. }
  110. /**
  111. * Tests if an exception is thrown on an incorrect mask pattern
  112. */
  113. public function testInvalidMaskPatternException():void{
  114. $this->expectException(QRCodeException::class);
  115. $this->expectExceptionMessage('invalid mask pattern');
  116. /** @phan-suppress-next-line PhanNoopNew */
  117. new MaskPattern(42);
  118. }
  119. public function testPenaltyRule1():void{
  120. // horizontal
  121. $this::assertSame(0, MaskPattern::testRule1([[0, 0, 0, 0]], 1, 4));
  122. $this::assertSame(3, MaskPattern::testRule1([[0, 0, 0, 0, 0, 1]], 1, 6));
  123. $this::assertSame(4, MaskPattern::testRule1([[0, 0, 0, 0, 0, 0]], 1, 6));
  124. // vertical
  125. $this::assertSame(0, MaskPattern::testRule1([[0], [0], [0], [0]], 4, 1));
  126. $this::assertSame(3, MaskPattern::testRule1([[0], [0], [0], [0], [0], [1]], 6, 1));
  127. $this::assertSame(4, MaskPattern::testRule1([[0], [0], [0], [0], [0], [0]], 6, 1));
  128. }
  129. public function testPenaltyRule2():void{
  130. $this::assertSame(0, MaskPattern::testRule2([[0]], 1, 1));
  131. $this::assertSame(0, MaskPattern::testRule2([[0, 0], [0, 1]], 2, 2));
  132. $this::assertSame(3, MaskPattern::testRule2([[0, 0], [0, 0]], 2, 2));
  133. $this::assertSame(12, MaskPattern::testRule2([[0, 0, 0], [0, 0, 0], [0, 0, 0]], 3, 3));
  134. }
  135. public function testPenaltyRule3():void{
  136. // horizontal
  137. $this::assertSame(40, MaskPattern::testRule3([[0, 0, 0, 0, 1, 0, 1, 1, 1, 0, 1]], 1, 11));
  138. $this::assertSame(40, MaskPattern::testRule3([[1, 0, 1, 1, 1, 0, 1, 0, 0, 0, 0]], 1, 11));
  139. $this::assertSame(0, MaskPattern::testRule3([[1, 0, 1, 1, 1, 0, 1]], 1, 7));
  140. // vertical
  141. $this::assertSame(40, MaskPattern::testRule3([[0], [0], [0], [0], [1], [0], [1], [1], [1], [0], [1]], 11, 1));
  142. $this::assertSame(40, MaskPattern::testRule3([[1], [0], [1], [1], [1], [0], [1], [0], [0], [0], [0]], 11, 1));
  143. $this::assertSame(0, MaskPattern::testRule3([[1], [0], [1], [1], [1], [0], [1]], 7, 1));
  144. }
  145. public function testPenaltyRule4():void{
  146. $this::assertSame(100, MaskPattern::testRule4([[0]], 1, 1));
  147. $this::assertSame(0, MaskPattern::testRule4([[0, 1]], 1, 2));
  148. $this::assertSame(30, MaskPattern::testRule4([[0, 1, 1, 1, 1, 0]], 1, 6));
  149. }
  150. }