/android/LGame-Android-0.2.6S/org/loon/framework/android/game/core/graphics/geom/Crossings.java

http://loon-simple.googlecode.com/ · Java · 502 lines · 405 code · 29 blank · 68 comment · 127 complexity · b7d2ce873924c24687b6093743830648 MD5 · raw file

  1. /*
  2. * Copyright 1998-2003 Sun Microsystems, Inc. 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. Sun designates this
  8. * particular file as subject to the "Classpath" exception as provided
  9. * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
  22. * CA 95054 USA or visit www.sun.com if you need additional information or
  23. * have any questions.
  24. */
  25. package org.loon.framework.android.game.core.graphics.geom;
  26. import java.util.Vector;
  27. import java.util.Enumeration;
  28. public abstract class Crossings {
  29. //public static final boolean debug = false;
  30. int limit = 0;
  31. double yranges[] = new double[10];
  32. double xlo, ylo, xhi, yhi;
  33. public Crossings(double xlo, double ylo, double xhi, double yhi) {
  34. this.xlo = xlo;
  35. this.ylo = ylo;
  36. this.xhi = xhi;
  37. this.yhi = yhi;
  38. }
  39. public final double getXLo() {
  40. return xlo;
  41. }
  42. public final double getYLo() {
  43. return ylo;
  44. }
  45. public final double getXHi() {
  46. return xhi;
  47. }
  48. public final double getYHi() {
  49. return yhi;
  50. }
  51. public abstract void record(double ystart, double yend, int direction);
  52. /*
  53. public void print() {
  54. System.out.println("Crossings [");
  55. System.out.println(" bounds = [" + ylo + ", " + yhi + "]");
  56. for (int i = 0; i < limit; i += 2) {
  57. System.out
  58. .println(" [" + yranges[i] + ", " + yranges[i + 1] + "]");
  59. }
  60. System.out.println("]");
  61. }*/
  62. public final boolean isEmpty() {
  63. return (limit == 0);
  64. }
  65. public abstract boolean covers(double ystart, double yend);
  66. public static Crossings findCrossings(Vector<?> curves, double xlo,
  67. double ylo, double xhi, double yhi) {
  68. Crossings cross = new EvenOdd(xlo, ylo, xhi, yhi);
  69. Enumeration<?> enum_ = curves.elements();
  70. while (enum_.hasMoreElements()) {
  71. Curve c = (Curve) enum_.nextElement();
  72. if (c.accumulateCrossings(cross)) {
  73. return null;
  74. }
  75. }
  76. /*if (debug) {
  77. cross.print();
  78. }*/
  79. return cross;
  80. }
  81. public static Crossings findCrossings(PathIterator pi, double xlo,
  82. double ylo, double xhi, double yhi) {
  83. Crossings cross;
  84. if (pi.getWindingRule() == PathIterator.WIND_EVEN_ODD) {
  85. cross = new EvenOdd(xlo, ylo, xhi, yhi);
  86. } else {
  87. cross = new NonZero(xlo, ylo, xhi, yhi);
  88. }
  89. // coords array is big enough for holding:
  90. // coordinates returned from currentSegment (6)
  91. // OR
  92. // two subdivided quadratic curves (2+4+4=10)
  93. // AND
  94. // 0-1 horizontal splitting parameters
  95. // OR
  96. // 2 parametric equation derivative coefficients
  97. // OR
  98. // three subdivided cubic curves (2+6+6+6=20)
  99. // AND
  100. // 0-2 horizontal splitting parameters
  101. // OR
  102. // 3 parametric equation derivative coefficients
  103. double coords[] = new double[23];
  104. double movx = 0;
  105. double movy = 0;
  106. double curx = 0;
  107. double cury = 0;
  108. double newx, newy;
  109. while (!pi.isDone()) {
  110. int type = pi.currentSegment(coords);
  111. switch (type) {
  112. case PathIterator.SEG_MOVETO:
  113. if (movy != cury
  114. && cross.accumulateLine(curx, cury, movx, movy)) {
  115. return null;
  116. }
  117. movx = curx = coords[0];
  118. movy = cury = coords[1];
  119. break;
  120. case PathIterator.SEG_LINETO:
  121. newx = coords[0];
  122. newy = coords[1];
  123. if (cross.accumulateLine(curx, cury, newx, newy)) {
  124. return null;
  125. }
  126. curx = newx;
  127. cury = newy;
  128. break;
  129. case PathIterator.SEG_QUADTO:
  130. newx = coords[2];
  131. newy = coords[3];
  132. if (cross.accumulateQuad(curx, cury, coords)) {
  133. return null;
  134. }
  135. curx = newx;
  136. cury = newy;
  137. break;
  138. case PathIterator.SEG_CUBICTO:
  139. newx = coords[4];
  140. newy = coords[5];
  141. if (cross.accumulateCubic(curx, cury, coords)) {
  142. return null;
  143. }
  144. curx = newx;
  145. cury = newy;
  146. break;
  147. case PathIterator.SEG_CLOSE:
  148. if (movy != cury
  149. && cross.accumulateLine(curx, cury, movx, movy)) {
  150. return null;
  151. }
  152. curx = movx;
  153. cury = movy;
  154. break;
  155. }
  156. pi.next();
  157. }
  158. if (movy != cury) {
  159. if (cross.accumulateLine(curx, cury, movx, movy)) {
  160. return null;
  161. }
  162. }
  163. /*if (debug) {
  164. cross.print();
  165. }*/
  166. return cross;
  167. }
  168. public boolean accumulateLine(double x0, double y0, double x1, double y1) {
  169. if (y0 <= y1) {
  170. return accumulateLine(x0, y0, x1, y1, 1);
  171. } else {
  172. return accumulateLine(x1, y1, x0, y0, -1);
  173. }
  174. }
  175. public boolean accumulateLine(double x0, double y0, double x1, double y1,
  176. int direction) {
  177. if (yhi <= y0 || ylo >= y1) {
  178. return false;
  179. }
  180. if (x0 >= xhi && x1 >= xhi) {
  181. return false;
  182. }
  183. if (y0 == y1) {
  184. return (x0 >= xlo || x1 >= xlo);
  185. }
  186. double xstart, ystart, xend, yend;
  187. double dx = (x1 - x0);
  188. double dy = (y1 - y0);
  189. if (y0 < ylo) {
  190. xstart = x0 + (ylo - y0) * dx / dy;
  191. ystart = ylo;
  192. } else {
  193. xstart = x0;
  194. ystart = y0;
  195. }
  196. if (yhi < y1) {
  197. xend = x0 + (yhi - y0) * dx / dy;
  198. yend = yhi;
  199. } else {
  200. xend = x1;
  201. yend = y1;
  202. }
  203. if (xstart >= xhi && xend >= xhi) {
  204. return false;
  205. }
  206. if (xstart > xlo || xend > xlo) {
  207. return true;
  208. }
  209. record(ystart, yend, direction);
  210. return false;
  211. }
  212. private Vector<Curve> tmp = new Vector<Curve>();
  213. public boolean accumulateQuad(double x0, double y0, double coords[]) {
  214. if (y0 < ylo && coords[1] < ylo && coords[3] < ylo) {
  215. return false;
  216. }
  217. if (y0 > yhi && coords[1] > yhi && coords[3] > yhi) {
  218. return false;
  219. }
  220. if (x0 > xhi && coords[0] > xhi && coords[2] > xhi) {
  221. return false;
  222. }
  223. if (x0 < xlo && coords[0] < xlo && coords[2] < xlo) {
  224. if (y0 < coords[3]) {
  225. record(Math.max(y0, ylo), Math.min(coords[3], yhi), 1);
  226. } else if (y0 > coords[3]) {
  227. record(Math.max(coords[3], ylo), Math.min(y0, yhi), -1);
  228. }
  229. return false;
  230. }
  231. Curve.insertQuad(tmp, x0, y0, coords);
  232. Enumeration<Curve> enum_ = tmp.elements();
  233. while (enum_.hasMoreElements()) {
  234. Curve c = (Curve) enum_.nextElement();
  235. if (c.accumulateCrossings(this)) {
  236. return true;
  237. }
  238. }
  239. tmp.clear();
  240. return false;
  241. }
  242. public boolean accumulateCubic(double x0, double y0, double coords[]) {
  243. if (y0 < ylo && coords[1] < ylo && coords[3] < ylo && coords[5] < ylo) {
  244. return false;
  245. }
  246. if (y0 > yhi && coords[1] > yhi && coords[3] > yhi && coords[5] > yhi) {
  247. return false;
  248. }
  249. if (x0 > xhi && coords[0] > xhi && coords[2] > xhi && coords[4] > xhi) {
  250. return false;
  251. }
  252. if (x0 < xlo && coords[0] < xlo && coords[2] < xlo && coords[4] < xlo) {
  253. if (y0 <= coords[5]) {
  254. record(Math.max(y0, ylo), Math.min(coords[5], yhi), 1);
  255. } else {
  256. record(Math.max(coords[5], ylo), Math.min(y0, yhi), -1);
  257. }
  258. return false;
  259. }
  260. Curve.insertCubic(tmp, x0, y0, coords);
  261. Enumeration<Curve> enum_ = tmp.elements();
  262. while (enum_.hasMoreElements()) {
  263. Curve c = (Curve) enum_.nextElement();
  264. if (c.accumulateCrossings(this)) {
  265. return true;
  266. }
  267. }
  268. tmp.clear();
  269. return false;
  270. }
  271. public final static class EvenOdd extends Crossings {
  272. public EvenOdd(double xlo, double ylo, double xhi, double yhi) {
  273. super(xlo, ylo, xhi, yhi);
  274. }
  275. public final boolean covers(double ystart, double yend) {
  276. return (limit == 2 && yranges[0] <= ystart && yranges[1] >= yend);
  277. }
  278. public void record(double ystart, double yend, int direction) {
  279. if (ystart >= yend) {
  280. return;
  281. }
  282. int from = 0;
  283. // Quickly jump over all pairs that are completely "above"
  284. while (from < limit && ystart > yranges[from + 1]) {
  285. from += 2;
  286. }
  287. int to = from;
  288. while (from < limit) {
  289. double yrlo = yranges[from++];
  290. double yrhi = yranges[from++];
  291. if (yend < yrlo) {
  292. // Quickly handle insertion of the new range
  293. yranges[to++] = ystart;
  294. yranges[to++] = yend;
  295. ystart = yrlo;
  296. yend = yrhi;
  297. continue;
  298. }
  299. // The ranges overlap - sort, collapse, insert, iterate
  300. double yll, ylh, yhl, yhh;
  301. if (ystart < yrlo) {
  302. yll = ystart;
  303. ylh = yrlo;
  304. } else {
  305. yll = yrlo;
  306. ylh = ystart;
  307. }
  308. if (yend < yrhi) {
  309. yhl = yend;
  310. yhh = yrhi;
  311. } else {
  312. yhl = yrhi;
  313. yhh = yend;
  314. }
  315. if (ylh == yhl) {
  316. ystart = yll;
  317. yend = yhh;
  318. } else {
  319. if (ylh > yhl) {
  320. ystart = yhl;
  321. yhl = ylh;
  322. ylh = ystart;
  323. }
  324. if (yll != ylh) {
  325. yranges[to++] = yll;
  326. yranges[to++] = ylh;
  327. }
  328. ystart = yhl;
  329. yend = yhh;
  330. }
  331. if (ystart >= yend) {
  332. break;
  333. }
  334. }
  335. if (to < from && from < limit) {
  336. System.arraycopy(yranges, from, yranges, to, limit - from);
  337. }
  338. to += (limit - from);
  339. if (ystart < yend) {
  340. if (to >= yranges.length) {
  341. double newranges[] = new double[to + 10];
  342. System.arraycopy(yranges, 0, newranges, 0, to);
  343. yranges = newranges;
  344. }
  345. yranges[to++] = ystart;
  346. yranges[to++] = yend;
  347. }
  348. limit = to;
  349. }
  350. }
  351. public final static class NonZero extends Crossings {
  352. private int crosscounts[];
  353. public NonZero(double xlo, double ylo, double xhi, double yhi) {
  354. super(xlo, ylo, xhi, yhi);
  355. crosscounts = new int[yranges.length / 2];
  356. }
  357. public final boolean covers(double ystart, double yend) {
  358. int i = 0;
  359. while (i < limit) {
  360. double ylo = yranges[i++];
  361. double yhi = yranges[i++];
  362. if (ystart >= yhi) {
  363. continue;
  364. }
  365. if (ystart < ylo) {
  366. return false;
  367. }
  368. if (yend <= yhi) {
  369. return true;
  370. }
  371. ystart = yhi;
  372. }
  373. return (ystart >= yend);
  374. }
  375. public void remove(int cur) {
  376. limit -= 2;
  377. int rem = limit - cur;
  378. if (rem > 0) {
  379. System.arraycopy(yranges, cur + 2, yranges, cur, rem);
  380. System.arraycopy(crosscounts, cur / 2 + 1, crosscounts,
  381. cur / 2, rem / 2);
  382. }
  383. }
  384. public void insert(int cur, double lo, double hi, int dir) {
  385. int rem = limit - cur;
  386. double oldranges[] = yranges;
  387. int oldcounts[] = crosscounts;
  388. if (limit >= yranges.length) {
  389. yranges = new double[limit + 10];
  390. System.arraycopy(oldranges, 0, yranges, 0, cur);
  391. crosscounts = new int[(limit + 10) / 2];
  392. System.arraycopy(oldcounts, 0, crosscounts, 0, cur / 2);
  393. }
  394. if (rem > 0) {
  395. System.arraycopy(oldranges, cur, yranges, cur + 2, rem);
  396. System.arraycopy(oldcounts, cur / 2, crosscounts, cur / 2 + 1,
  397. rem / 2);
  398. }
  399. yranges[cur + 0] = lo;
  400. yranges[cur + 1] = hi;
  401. crosscounts[cur / 2] = dir;
  402. limit += 2;
  403. }
  404. public void record(double ystart, double yend, int direction) {
  405. if (ystart >= yend) {
  406. return;
  407. }
  408. int cur = 0;
  409. // Quickly jump over all pairs that are completely "above"
  410. while (cur < limit && ystart > yranges[cur + 1]) {
  411. cur += 2;
  412. }
  413. if (cur < limit) {
  414. int rdir = crosscounts[cur / 2];
  415. double yrlo = yranges[cur + 0];
  416. double yrhi = yranges[cur + 1];
  417. if (yrhi == ystart && rdir == direction) {
  418. // Remove the range from the list and collapse it
  419. // into the range being inserted. Note that the
  420. // new combined range may overlap the following range
  421. // so we must not simply combine the ranges in place
  422. // unless we are at the last range.
  423. if (cur + 2 == limit) {
  424. yranges[cur + 1] = yend;
  425. return;
  426. }
  427. remove(cur);
  428. ystart = yrlo;
  429. rdir = crosscounts[cur / 2];
  430. yrlo = yranges[cur + 0];
  431. yrhi = yranges[cur + 1];
  432. }
  433. if (yend < yrlo) {
  434. // Just insert the new range at the current location
  435. insert(cur, ystart, yend, direction);
  436. return;
  437. }
  438. if (yend == yrlo && rdir == direction) {
  439. // Just prepend the new range to the current one
  440. yranges[cur] = ystart;
  441. return;
  442. }
  443. // The ranges must overlap - (yend > yrlo && yrhi > ystart)
  444. if (ystart < yrlo) {
  445. insert(cur, ystart, yrlo, direction);
  446. cur += 2;
  447. ystart = yrlo;
  448. } else if (yrlo < ystart) {
  449. insert(cur, yrlo, ystart, rdir);
  450. cur += 2;
  451. yrlo = ystart;
  452. }
  453. // assert(yrlo == ystart);
  454. int newdir = rdir + direction;
  455. double newend = Math.min(yend, yrhi);
  456. if (newdir == 0) {
  457. remove(cur);
  458. } else {
  459. crosscounts[cur / 2] = newdir;
  460. yranges[cur++] = ystart;
  461. yranges[cur++] = newend;
  462. }
  463. ystart = yrlo = newend;
  464. if (yrlo < yrhi) {
  465. insert(cur, yrlo, yrhi, rdir);
  466. }
  467. }
  468. if (ystart < yend) {
  469. insert(cur, ystart, yend, direction);
  470. }
  471. }
  472. }
  473. }