1 /*
2  * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved.
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * This code is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 only, as
7  * published by the Free Software Foundation.  Oracle designates this
8  * particular file as subject to the "Classpath" exception as provided
9  * by Oracle in the LICENSE file that accompanied this code.
10  *
11  * This code is distributed in the hope that it will be useful, but WITHOUT
12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14  * version 2 for more details (a copy is included in the LICENSE file that
15  * accompanied this code).
16  *
17  * You should have received a copy of the GNU General Public License version
18  * 2 along with this work; if not, write to the Free Software Foundation,
19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20  *
21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22  * or visit www.oracle.com if you need additional information or have any
23  * questions.
24  */

25
26 package java.awt;
27
28 import java.awt.image.Raster;
29 import sun.awt.image.IntegerComponentRaster;
30 import java.awt.image.ColorModel;
31 import java.awt.image.DirectColorModel;
32 import java.awt.geom.Point2D;
33 import java.awt.geom.AffineTransform;
34 import java.awt.geom.NoninvertibleTransformException;
35 import java.lang.ref.WeakReference;
36
37 class GradientPaintContext implements PaintContext {
38     static ColorModel xrgbmodel =
39         new DirectColorModel(24, 0x00ff0000, 0x0000ff00, 0x000000ff);
40     static ColorModel xbgrmodel =
41         new DirectColorModel(24, 0x000000ff, 0x0000ff00, 0x00ff0000);
42
43     static ColorModel cachedModel;
44     static WeakReference<Raster> cached;
45
46     static synchronized Raster getCachedRaster(ColorModel cm, int w, int h) {
47         if (cm == cachedModel) {
48             if (cached != null) {
49                 Raster ras = cached.get();
50                 if (ras != null &&
51                     ras.getWidth() >= w &&
52                     ras.getHeight() >= h)
53                 {
54                     cached = null;
55                     return ras;
56                 }
57             }
58         }
59         return cm.createCompatibleWritableRaster(w, h);
60     }
61
62     static synchronized void putCachedRaster(ColorModel cm, Raster ras) {
63         if (cached != null) {
64             Raster cras = cached.get();
65             if (cras != null) {
66                 int cw = cras.getWidth();
67                 int ch = cras.getHeight();
68                 int iw = ras.getWidth();
69                 int ih = ras.getHeight();
70                 if (cw >= iw && ch >= ih) {
71                     return;
72                 }
73                 if (cw * ch >= iw * ih) {
74                     return;
75                 }
76             }
77         }
78         cachedModel = cm;
79         cached = new WeakReference<>(ras);
80     }
81
82     double x1;
83     double y1;
84     double dx;
85     double dy;
86     boolean cyclic;
87     int interp[];
88     Raster saved;
89     ColorModel model;
90
91     public GradientPaintContext(ColorModel cm,
92                                 Point2D p1, Point2D p2, AffineTransform xform,
93                                 Color c1, Color c2, boolean cyclic) {
94         // First calculate the distance moved in user space when
95         // we move a single unit along the X & Y axes in device space.
96         Point2D xvec = new Point2D.Double(1, 0);
97         Point2D yvec = new Point2D.Double(0, 1);
98         try {
99             AffineTransform inverse = xform.createInverse();
100             inverse.deltaTransform(xvec, xvec);
101             inverse.deltaTransform(yvec, yvec);
102         } catch (NoninvertibleTransformException e) {
103             xvec.setLocation(0, 0);
104             yvec.setLocation(0, 0);
105         }
106
107         // Now calculate the (square of the) user space distance
108         // between the anchor points. This value equals:
109         //     (UserVec . UserVec)
110         double udx = p2.getX() - p1.getX();
111         double udy = p2.getY() - p1.getY();
112         double ulenSq = udx * udx + udy * udy;
113
114         if (ulenSq <= Double.MIN_VALUE) {
115             dx = 0;
116             dy = 0;
117         } else {
118             // Now calculate the proportional distance moved along the
119             // vector from p1 to p2 when we move a unit along X & Y in
120             // device space.
121             //
122             // The length of the projection of the Device Axis Vector is
123             // its dot product with the Unit User Vector:
124             //     (DevAxisVec . (UserVec / Len(UserVec))
125             //
126             // The "proportional" length is that length divided again
127             // by the length of the User Vector:
128             //     (DevAxisVec . (UserVec / Len(UserVec))) / Len(UserVec)
129             // which simplifies to:
130             //     ((DevAxisVec . UserVec) / Len(UserVec)) / Len(UserVec)
131             // which simplifies to:
132             //     (DevAxisVec . UserVec) / LenSquared(UserVec)
133             dx = (xvec.getX() * udx + xvec.getY() * udy) / ulenSq;
134             dy = (yvec.getX() * udx + yvec.getY() * udy) / ulenSq;
135
136             if (cyclic) {
137                 dx = dx % 1.0;
138                 dy = dy % 1.0;
139             } else {
140                 // We are acyclic
141                 if (dx < 0) {
142                     // If we are using the acyclic form below, we need
143                     // dx to be non-negative for simplicity of scanning
144                     // across the scan lines for the transition points.
145                     // To ensure that constraint, we negate the dx/dy
146                     // values and swap the points and colors.
147                     Point2D p = p1; p1 = p2; p2 = p;
148                     Color c = c1; c1 = c2; c2 = c;
149                     dx = -dx;
150                     dy = -dy;
151                 }
152             }
153         }
154
155         Point2D dp1 = xform.transform(p1, null);
156         this.x1 = dp1.getX();
157         this.y1 = dp1.getY();
158
159         this.cyclic = cyclic;
160         int rgb1 = c1.getRGB();
161         int rgb2 = c2.getRGB();
162         int a1 = (rgb1 >> 24) & 0xff;
163         int r1 = (rgb1 >> 16) & 0xff;
164         int g1 = (rgb1 >>  8) & 0xff;
165         int b1 = (rgb1      ) & 0xff;
166         int da = ((rgb2 >> 24) & 0xff) - a1;
167         int dr = ((rgb2 >> 16) & 0xff) - r1;
168         int dg = ((rgb2 >>  8) & 0xff) - g1;
169         int db = ((rgb2      ) & 0xff) - b1;
170         if (a1 == 0xff && da == 0) {
171             model = xrgbmodel;
172             if (cm instanceof DirectColorModel) {
173                 DirectColorModel dcm = (DirectColorModel) cm;
174                 int tmp = dcm.getAlphaMask();
175                 if ((tmp == 0 || tmp == 0xff) &&
176                     dcm.getRedMask() == 0xff &&
177                     dcm.getGreenMask() == 0xff00 &&
178                     dcm.getBlueMask() == 0xff0000)
179                 {
180                     model = xbgrmodel;
181                     tmp = r1; r1 = b1; b1 = tmp;
182                     tmp = dr; dr = db; db = tmp;
183                 }
184             }
185         } else {
186             model = ColorModel.getRGBdefault();
187         }
188         interp = new int[cyclic ? 513 : 257];
189         for (int i = 0; i <= 256; i++) {
190             float rel = i / 256.0f;
191             int rgb =
192                 (((int) (a1 + da * rel)) << 24) |
193                 (((int) (r1 + dr * rel)) << 16) |
194                 (((int) (g1 + dg * rel)) <<  8) |
195                 (((int) (b1 + db * rel))      );
196             interp[i] = rgb;
197             if (cyclic) {
198                 interp[512 - i] = rgb;
199             }
200         }
201     }
202
203     /**
204      * Release the resources allocated for the operation.
205      */

206     public void dispose() {
207         if (saved != null) {
208             putCachedRaster(model, saved);
209             saved = null;
210         }
211     }
212
213     /**
214      * Return the ColorModel of the output.
215      */

216     public ColorModel getColorModel() {
217         return model;
218     }
219
220     /**
221      * Return a Raster containing the colors generated for the graphics
222      * operation.
223      * @param x,y,w,h The area in device space for which colors are
224      * generated.
225      */

226     public Raster getRaster(int x, int y, int w, int h) {
227         double rowrel = (x - x1) * dx + (y - y1) * dy;
228
229         Raster rast = saved;
230         if (rast == null || rast.getWidth() < w || rast.getHeight() < h) {
231             rast = getCachedRaster(model, w, h);
232             saved = rast;
233         }
234         IntegerComponentRaster irast = (IntegerComponentRaster) rast;
235         int off = irast.getDataOffset(0);
236         int adjust = irast.getScanlineStride() - w;
237         int[] pixels = irast.getDataStorage();
238
239         if (cyclic) {
240             cycleFillRaster(pixels, off, adjust, w, h, rowrel, dx, dy);
241         } else {
242             clipFillRaster(pixels, off, adjust, w, h, rowrel, dx, dy);
243         }
244
245         irast.markDirty();
246
247         return rast;
248     }
249
250     void cycleFillRaster(int[] pixels, int off, int adjust, int w, int h,
251                          double rowrel, double dx, double dy) {
252         rowrel = rowrel % 2.0;
253         int irowrel = ((int) (rowrel * (1 << 30))) << 1;
254         int idx = (int) (-dx * (1 << 31));
255         int idy = (int) (-dy * (1 << 31));
256         while (--h >= 0) {
257             int icolrel = irowrel;
258             for (int j = w; j > 0; j--) {
259                 pixels[off++] = interp[icolrel >>> 23];
260                 icolrel += idx;
261             }
262
263             off += adjust;
264             irowrel += idy;
265         }
266     }
267
268     void clipFillRaster(int[] pixels, int off, int adjust, int w, int h,
269                         double rowrel, double dx, double dy) {
270         while (--h >= 0) {
271             double colrel = rowrel;
272             int j = w;
273             if (colrel <= 0.0) {
274                 int rgb = interp[0];
275                 do {
276                     pixels[off++] = rgb;
277                     colrel += dx;
278                 } while (--j > 0 && colrel <= 0.0);
279             }
280             while (colrel < 1.0 && --j >= 0) {
281                 pixels[off++] = interp[(int) (colrel * 256)];
282                 colrel += dx;
283             }
284             if (j > 0) {
285                 int rgb = interp[256];
286                 do {
287                     pixels[off++] = rgb;
288                 } while (--j > 0);
289             }
290
291             off += adjust;
292             rowrel += dy;
293         }
294     }
295 }
296