MaskPatternTest.php 4.8 KB

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