Prettify scan result screen, I think
[zxing.git] / csharp / common / PerspectiveTransform.cs
1 /*\r
2 * Copyright 2007 ZXing authors\r
3 *\r
4 * Licensed under the Apache License, Version 2.0 (the "License");\r
5 * you may not use this file except in compliance with the License.\r
6 * You may obtain a copy of the License at\r
7 *\r
8 *      http://www.apache.org/licenses/LICENSE-2.0\r
9 *\r
10 * Unless required by applicable law or agreed to in writing, software\r
11 * distributed under the License is distributed on an "AS IS" BASIS,\r
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\r
13 * See the License for the specific language governing permissions and\r
14 * limitations under the License.\r
15 */\r
16 using System;\r
17 namespace com.google.zxing.common\r
18 {\r
19         \r
20         /// <summary> <p>This class implements a perspective transform in two dimensions. Given four source and four\r
21         /// destination points, it will compute the transformation implied between them. The code is based\r
22         /// directly upon section 3.4.2 of George Wolberg's "Digital Image Warping"; see pages 54-56.</p>\r
23         /// \r
24         /// </summary>\r
25         /// <author>  Sean Owen\r
26         /// </author>\r
27         /// <author>www.Redivivus.in (suraj.supekar@redivivus.in) - Ported from ZXING Java Source \r
28         /// </author>\r
29         public sealed class PerspectiveTransform\r
30         {\r
31                 \r
32                 //UPGRADE_NOTE: Final was removed from the declaration of 'a11 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
33                 //UPGRADE_NOTE: Final was removed from the declaration of 'a12 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
34                 //UPGRADE_NOTE: Final was removed from the declaration of 'a13 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
35                 //UPGRADE_NOTE: Final was removed from the declaration of 'a21 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
36                 //UPGRADE_NOTE: Final was removed from the declaration of 'a22 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
37                 //UPGRADE_NOTE: Final was removed from the declaration of 'a23 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
38                 //UPGRADE_NOTE: Final was removed from the declaration of 'a31 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
39                 //UPGRADE_NOTE: Final was removed from the declaration of 'a32 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
40                 //UPGRADE_NOTE: Final was removed from the declaration of 'a33 '. "ms-help://MS.VSCC.v80/dv_commoner/local/redirect.htm?index='!DefaultContextWindowIndex'&keyword='jlca1003'"\r
41                 private float a11;\r
42                 private float a12;\r
43                 private float a13;\r
44                 private float a21;\r
45                 private float a22;\r
46                 private float a23;\r
47                 private float a31;\r
48                 private float a32;\r
49                 private float a33;\r
50                 \r
51                 private PerspectiveTransform(float a11, float a21, float a31, float a12, float a22, float a32, float a13, float a23, float a33)\r
52                 {\r
53                         this.a11 = a11;\r
54                         this.a12 = a12;\r
55                         this.a13 = a13;\r
56                         this.a21 = a21;\r
57                         this.a22 = a22;\r
58                         this.a23 = a23;\r
59                         this.a31 = a31;\r
60                         this.a32 = a32;\r
61                         this.a33 = a33;\r
62                 }\r
63                 \r
64                 public static PerspectiveTransform quadrilateralToQuadrilateral(float x0, float y0, float x1, float y1, float x2, float y2, float x3, float y3, float x0p, float y0p, float x1p, float y1p, float x2p, float y2p, float x3p, float y3p)\r
65                 {\r
66                         \r
67                         PerspectiveTransform qToS = quadrilateralToSquare(x0, y0, x1, y1, x2, y2, x3, y3);\r
68                         PerspectiveTransform sToQ = squareToQuadrilateral(x0p, y0p, x1p, y1p, x2p, y2p, x3p, y3p);\r
69                         return sToQ.times(qToS);\r
70                 }\r
71                 \r
72                 public void  transformPoints(float[] points)\r
73                 {\r
74                         int max = points.Length;\r
75                         float a11 = this.a11;\r
76                         float a12 = this.a12;\r
77                         float a13 = this.a13;\r
78                         float a21 = this.a21;\r
79                         float a22 = this.a22;\r
80                         float a23 = this.a23;\r
81                         float a31 = this.a31;\r
82                         float a32 = this.a32;\r
83                         float a33 = this.a33;\r
84                         for (int i = 0; i < max; i += 2)\r
85                         {\r
86                                 float x = points[i];\r
87                                 float y = points[i + 1];\r
88                                 float denominator = a13 * x + a23 * y + a33;\r
89                                 points[i] = (a11 * x + a21 * y + a31) / denominator;\r
90                                 points[i + 1] = (a12 * x + a22 * y + a32) / denominator;\r
91                         }\r
92                 }\r
93                 \r
94                 /// <summary>Convenience method, not optimized for performance. </summary>\r
95                 public void  transformPoints(float[] xValues, float[] yValues)\r
96                 {\r
97                         int n = xValues.Length;\r
98                         for (int i = 0; i < n; i++)\r
99                         {\r
100                                 float x = xValues[i];\r
101                                 float y = yValues[i];\r
102                                 float denominator = a13 * x + a23 * y + a33;\r
103                                 xValues[i] = (a11 * x + a21 * y + a31) / denominator;\r
104                                 yValues[i] = (a12 * x + a22 * y + a32) / denominator;\r
105                         }\r
106                 }\r
107                 \r
108                 public static PerspectiveTransform squareToQuadrilateral(float x0, float y0, float x1, float y1, float x2, float y2, float x3, float y3)\r
109                 {\r
110                         float dy2 = y3 - y2;\r
111                         float dy3 = y0 - y1 + y2 - y3;\r
112                         if (dy2 == 0.0f && dy3 == 0.0f)\r
113                         {\r
114                                 return new PerspectiveTransform(x1 - x0, x2 - x1, x0, y1 - y0, y2 - y1, y0, 0.0f, 0.0f, 1.0f);\r
115                         }\r
116                         else\r
117                         {\r
118                                 float dx1 = x1 - x2;\r
119                                 float dx2 = x3 - x2;\r
120                                 float dx3 = x0 - x1 + x2 - x3;\r
121                                 float dy1 = y1 - y2;\r
122                                 float denominator = dx1 * dy2 - dx2 * dy1;\r
123                                 float a13 = (dx3 * dy2 - dx2 * dy3) / denominator;\r
124                                 float a23 = (dx1 * dy3 - dx3 * dy1) / denominator;\r
125                                 return new PerspectiveTransform(x1 - x0 + a13 * x1, x3 - x0 + a23 * x3, x0, y1 - y0 + a13 * y1, y3 - y0 + a23 * y3, y0, a13, a23, 1.0f);\r
126                         }\r
127                 }\r
128                 \r
129                 public static PerspectiveTransform quadrilateralToSquare(float x0, float y0, float x1, float y1, float x2, float y2, float x3, float y3)\r
130                 {\r
131                         // Here, the adjoint serves as the inverse:\r
132                         return squareToQuadrilateral(x0, y0, x1, y1, x2, y2, x3, y3).buildAdjoint();\r
133                 }\r
134                 \r
135                 internal PerspectiveTransform buildAdjoint()\r
136                 {\r
137                         // Adjoint is the transpose of the cofactor matrix:\r
138                         return new PerspectiveTransform(a22 * a33 - a23 * a32, a23 * a31 - a21 * a33, a21 * a32 - a22 * a31, a13 * a32 - a12 * a33, a11 * a33 - a13 * a31, a12 * a31 - a11 * a32, a12 * a23 - a13 * a22, a13 * a21 - a11 * a23, a11 * a22 - a12 * a21);\r
139                 }\r
140                 \r
141                 internal PerspectiveTransform times(PerspectiveTransform other)\r
142                 {\r
143                         return new PerspectiveTransform(a11 * other.a11 + a21 * other.a12 + a31 * other.a13, a11 * other.a21 + a21 * other.a22 + a31 * other.a23, a11 * other.a31 + a21 * other.a32 + a31 * other.a33, a12 * other.a11 + a22 * other.a12 + a32 * other.a13, a12 * other.a21 + a22 * other.a22 + a32 * other.a23, a12 * other.a31 + a22 * other.a32 + a32 * other.a33, a13 * other.a11 + a23 * other.a12 + a33 * other.a13, a13 * other.a21 + a23 * other.a22 + a33 * other.a23, a13 * other.a31 + a23 * other.a32 + a33 * other.a33);\r
144                 }\r
145         }\r
146 }