Skip to content

Instantly share code, notes, and snippets.

@nraynaud
Created October 14, 2013 09:05
Show Gist options
  • Save nraynaud/6972980 to your computer and use it in GitHub Desktop.
Save nraynaud/6972980 to your computer and use it in GitHub Desktop.
Testing clipper algo change.
//instrumented by nraynaud for testing purpose
/*******************************************************************************
* *
* Author : Angus Johnson *
* Version : 5.0.2 *
* Date : 30 December 2012 *
* Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2012 *
* *
* License: *
* Use, modification & distribution is subject to Boost Software License Ver 1. *
* http://www.boost.org/LICENSE_1_0.txt *
* *
* Attributions: *
* The code in this library is an extension of Bala Vatti's clipping algorithm: *
* "A generic solution to polygon clipping" *
* Communications of the ACM, Vol 35, Issue 7 (July 1992) pp 56-63. *
* http://portal.acm.org/citation.cfm?id=129906 *
* *
* Computer graphics and geometric modeling: implementation and algorithms *
* By Max K. Agoston *
* Springer; 1 edition (January 4, 2005) *
* http://books.google.com/books?q=vatti+clipping+agoston *
* *
* See also: *
* "Polygon Offsetting by Computing Winding Numbers" *
* Paper no. DETC2005-85513 pp. 565-575 *
* ASME 2005 International Design Engineering Technical Conferences *
* and Computers and Information in Engineering Conference (IDETC/CIE2005) *
* September 24�28, 2005 , Long Beach, California, USA *
* http://www.me.berkeley.edu/~mcmains/pubs/DAC05OffsetPolygon.pdf *
* *
*******************************************************************************/
/*******************************************************************************
* *
* Author : Timo *
* Version : 5.0.2.2 *
* Date : 11 September 2013 *
* *
* This is a translation of the C# Clipper library to Javascript. *
* Int128 struct of C# is implemented using JSBN of Tom Wu. *
* Because Javascript lacks support for 64-bit integers, the space *
* is a little more restricted than in C# version. *
* *
* C# version has support for coordinate space: *
* +-4611686018427387903 ( sqrt(2^127 -1)/2 ) *
* while Javascript version has support for space: *
* +-4503599627370495 ( sqrt(2^106 -1)/2 ) *
* *
* Tom Wu's JSBN proved to be the fastest big integer library: *
* http://jsperf.com/big-integer-library-test *
* *
* This class can be made simpler when (if ever) 64-bit integer support comes. *
* *
*******************************************************************************/
/*******************************************************************************
* *
* Basic JavaScript BN library - subset useful for RSA encryption. *
* http://www-cs-students.stanford.edu/~tjw/jsbn/ *
* Copyright (c) 2005 Tom Wu *
* All Rights Reserved. *
* See "LICENSE" for details: *
* http://www-cs-students.stanford.edu/~tjw/jsbn/LICENSE *
* *
*******************************************************************************/
(function () {
// "use strict";
// Browser test to speedup performance critical functions
var nav = navigator.userAgent.toString().toLowerCase();
var browser = {};
if (nav.indexOf("chrome") != -1 && nav.indexOf("chromium") == -1) browser.chrome = 1; else browser.chrome = 0;
if (nav.indexOf("chromium") != -1) browser.chromium = 1; else browser.chromium = 0;
if (nav.indexOf("safari") != -1 && nav.indexOf("chrome") == -1 && nav.indexOf("chromium") == -1) browser.safari = 1; else browser.safari = 0;
if (nav.indexOf("firefox") != -1) browser.firefox = 1; else browser.firefox = 0;
if (nav.indexOf("firefox/17") != -1) browser.firefox17 = 1; else browser.firefox17 = 0;
if (nav.indexOf("firefox/15") != -1) browser.firefox15 = 1; else browser.firefox15 = 0;
if (nav.indexOf("firefox/3") != -1) browser.firefox3 = 1; else browser.firefox3 = 0;
if (nav.indexOf("opera") != -1) browser.opera = 1; else browser.opera = 0;
if (nav.indexOf("msie 10") != -1) browser.msie10 = 1; else browser.msie10 = 0;
if (nav.indexOf("msie 9") != -1) browser.msie9 = 1; else browser.msie9 = 0;
if (nav.indexOf("msie 8") != -1) browser.msie8 = 1; else browser.msie8 = 0;
if (nav.indexOf("msie 7") != -1) browser.msie7 = 1; else browser.msie7 = 0;
if (nav.indexOf("msie ") != -1) browser.msie = 1; else browser.msie = 0;
var ClipperLib = {};
ClipperLib.biginteger_used = null;
// Bits per digit
var dbits;
// JavaScript engine analysis
var canary = 0xdeadbeefcafe;
var j_lm = ((canary & 0xffffff) == 0xefcafe);
// (public) Constructor
function Int128(a, b, c) {
// This test variable can be removed,
// but at least for performance tests it is useful piece of knowledge
// This is the only ClipperLib related variable in Int128 library
ClipperLib.biginteger_used = 1;
if (a != null) if ("number" == typeof a) {
this.fromString(Math.floor(a)
.toString(), 10); //this.fromNumber(a,b,c);
}
else if (b == null && "string" != typeof a) this.fromString(a, 256);
else {
if (a.indexOf(".") != -1) a = a.substring(0, a.indexOf("."));
this.fromString(a, b);
}
}
// return new, unset Int128
function nbi() {
return new Int128(null);
}
// am: Compute w_j += (x*this_i), propagate carries,
// c is initial carry, returns final carry.
// c < 3*dvalue, x < 2*dvalue, this_i < dvalue
// We need to select the fastest one that works in this environment.
// am1: use a single mult and divide to get the high bits,
// max digit bits should be 26 because
// max internal value = 2*dvalue^2-2*dvalue (< 2^53)
function am1(i, x, w, j, c, n) {
while (--n >= 0) {
var v = x * this[i++] + w[j] + c;
c = Math.floor(v / 0x4000000);
w[j++] = v & 0x3ffffff;
}
return c;
}
// am2 avoids a big mult-and-extract completely.
// Max digit bits should be <= 30 because we do bitwise ops
// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)
function am2(i, x, w, j, c, n) {
var xl = x & 0x7fff,
xh = x >> 15;
while (--n >= 0) {
var l = this[i] & 0x7fff;
var h = this[i++] >> 15;
var m = xh * l + h * xl;
l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);
c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);
w[j++] = l & 0x3fffffff;
}
return c;
}
// Alternately, set max digit bits to 28 since some
// browsers slow down when dealing with 32-bit numbers.
function am3(i, x, w, j, c, n) {
var xl = x & 0x3fff,
xh = x >> 14;
while (--n >= 0) {
var l = this[i] & 0x3fff;
var h = this[i++] >> 14;
var m = xh * l + h * xl;
l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;
c = (l >> 28) + (m >> 14) + xh * h;
w[j++] = l & 0xfffffff;
}
return c;
}
if (j_lm && (navigator.appName == "Microsoft Internet Explorer")) {
Int128.prototype.am = am2;
dbits = 30;
}
else if (j_lm && (navigator.appName != "Netscape")) {
Int128.prototype.am = am1;
dbits = 26;
}
else { // Mozilla/Netscape seems to prefer am3
Int128.prototype.am = am3;
dbits = 28;
}
Int128.prototype.DB = dbits;
Int128.prototype.DM = ((1 << dbits) - 1);
Int128.prototype.DV = (1 << dbits);
var BI_FP = 52;
Int128.prototype.FV = Math.pow(2, BI_FP);
Int128.prototype.F1 = BI_FP - dbits;
Int128.prototype.F2 = 2 * dbits - BI_FP;
// Digit conversions
var BI_RM = "0123456789abcdefghijklmnopqrstuvwxyz";
var BI_RC = [];
var rr, vv;
rr = "0".charCodeAt(0);
for (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;
rr = "a".charCodeAt(0);
for (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
rr = "A".charCodeAt(0);
for (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
function int2char(n) {
return BI_RM.charAt(n);
}
function intAt(s, i) {
var c = BI_RC[s.charCodeAt(i)];
return (c == null) ? -1 : c;
}
// (protected) copy this to r
function bnpCopyTo(r) {
for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];
r.t = this.t;
r.s = this.s;
}
// (protected) set from integer value x, -DV <= x < DV
function bnpFromInt(x) {
this.t = 1;
this.s = (x < 0) ? -1 : 0;
if (x > 0) this[0] = x;
else if (x < -1) this[0] = x + this.DV;
else this.t = 0;
}
// return bigint initialized to value
function nbv(i) {
var r = nbi();
r.fromInt(i);
return r;
}
// (protected) set from string and radix
function bnpFromString(s, b) {
var k;
if (b == 16) k = 4;
else if (b == 8) k = 3;
else if (b == 256) k = 8; // byte array
else if (b == 2) k = 1;
else if (b == 32) k = 5;
else if (b == 4) k = 2;
else {
this.fromRadix(s, b);
return;
}
this.t = 0;
this.s = 0;
var i = s.length,
mi = false,
sh = 0;
while (--i >= 0) {
var x = (k == 8) ? s[i] & 0xff : intAt(s, i);
if (x < 0) {
if (s.charAt(i) == "-") mi = true;
continue;
}
mi = false;
if (sh == 0) this[this.t++] = x;
else if (sh + k > this.DB) {
this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;
this[this.t++] = (x >> (this.DB - sh));
}
else this[this.t - 1] |= x << sh;
sh += k;
if (sh >= this.DB) sh -= this.DB;
}
if (k == 8 && (s[0] & 0x80) != 0) {
this.s = -1;
if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;
}
this.clamp();
if (mi) Int128.ZERO.subTo(this, this);
}
// (protected) clamp off excess high words
function bnpClamp() {
var c = this.s & this.DM;
while (this.t > 0 && this[this.t - 1] == c)--this.t;
}
// (public) return string representation in given radix
function bnToString(b) {
if (this.s < 0) return "-" + this.negate()
.toString(b);
var k;
if (b == 16) k = 4;
else if (b == 8) k = 3;
else if (b == 2) k = 1;
else if (b == 32) k = 5;
else if (b == 4) k = 2;
else return this.toRadix(b);
var km = (1 << k) - 1,
d, m = false,
r = "",
i = this.t;
var p = this.DB - (i * this.DB) % k;
if (i-- > 0) {
if (p < this.DB && (d = this[i] >> p) > 0) {
m = true;
r = int2char(d);
}
while (i >= 0) {
if (p < k) {
d = (this[i] & ((1 << p) - 1)) << (k - p);
d |= this[--i] >> (p += this.DB - k);
}
else {
d = (this[i] >> (p -= k)) & km;
if (p <= 0) {
p += this.DB;
--i;
}
}
if (d > 0) m = true;
if (m) r += int2char(d);
}
}
return m ? r : "0";
}
// (public) -this
function bnNegate() {
var r = nbi();
Int128.ZERO.subTo(this, r);
return r;
}
// (public) |this|
function bnAbs() {
return (this.s < 0) ? this.negate() : this;
}
// (public) return + if this > a, - if this < a, 0 if equal
function bnCompareTo(a) {
var r = this.s - a.s;
if (r != 0) return r;
var i = this.t;
r = i - a.t;
if (r != 0) return (this.s < 0) ? -r : r;
while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;
return 0;
}
// returns bit length of the integer x
function nbits(x) {
var r = 1,
t;
if ((t = x >>> 16) != 0) {
x = t;
r += 16;
}
if ((t = x >> 8) != 0) {
x = t;
r += 8;
}
if ((t = x >> 4) != 0) {
x = t;
r += 4;
}
if ((t = x >> 2) != 0) {
x = t;
r += 2;
}
if ((t = x >> 1) != 0) {
x = t;
r += 1;
}
return r;
}
// (public) return the number of bits in "this"
function bnBitLength() {
if (this.t <= 0) return 0;
return this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM));
}
// (protected) r = this << n*DB
function bnpDLShiftTo(n, r) {
var i;
for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];
for (i = n - 1; i >= 0; --i) r[i] = 0;
r.t = this.t + n;
r.s = this.s;
}
// (protected) r = this >> n*DB
function bnpDRShiftTo(n, r) {
for (var i = n; i < this.t; ++i) r[i - n] = this[i];
r.t = Math.max(this.t - n, 0);
r.s = this.s;
}
// (protected) r = this << n
function bnpLShiftTo(n, r) {
var bs = n % this.DB;
var cbs = this.DB - bs;
var bm = (1 << cbs) - 1;
var ds = Math.floor(n / this.DB),
c = (this.s << bs) & this.DM,
i;
for (i = this.t - 1; i >= 0; --i) {
r[i + ds + 1] = (this[i] >> cbs) | c;
c = (this[i] & bm) << bs;
}
for (i = ds - 1; i >= 0; --i) r[i] = 0;
r[ds] = c;
r.t = this.t + ds + 1;
r.s = this.s;
r.clamp();
}
// (protected) r = this >> n
function bnpRShiftTo(n, r) {
r.s = this.s;
var ds = Math.floor(n / this.DB);
if (ds >= this.t) {
r.t = 0;
return;
}
var bs = n % this.DB;
var cbs = this.DB - bs;
var bm = (1 << bs) - 1;
r[0] = this[ds] >> bs;
for (var i = ds + 1; i < this.t; ++i) {
r[i - ds - 1] |= (this[i] & bm) << cbs;
r[i - ds] = this[i] >> bs;
}
if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;
r.t = this.t - ds;
r.clamp();
}
// (protected) r = this - a
function bnpSubTo(a, r) {
var i = 0,
c = 0,
m = Math.min(a.t, this.t);
while (i < m) {
c += this[i] - a[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
if (a.t < this.t) {
c -= a.s;
while (i < this.t) {
c += this[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
c += this.s;
}
else {
c += this.s;
while (i < a.t) {
c -= a[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
c -= a.s;
}
r.s = (c < 0) ? -1 : 0;
if (c < -1) r[i++] = this.DV + c;
else if (c > 0) r[i++] = c;
r.t = i;
r.clamp();
}
// (protected) r = this * a, r != this,a (HAC 14.12)
// "this" should be the larger one if appropriate.
function bnpMultiplyTo(a, r) {
var x = this.abs(),
y = a.abs();
var i = x.t;
r.t = i + y.t;
while (--i >= 0) r[i] = 0;
for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);
r.s = 0;
r.clamp();
if (this.s != a.s) Int128.ZERO.subTo(r, r);
}
// (protected) r = this^2, r != this (HAC 14.16)
function bnpSquareTo(r) {
var x = this.abs();
var i = r.t = 2 * x.t;
while (--i >= 0) r[i] = 0;
for (i = 0; i < x.t - 1; ++i) {
var c = x.am(i, x[i], r, 2 * i, 0, 1);
if ((r[i + x.t] += x.am(i + 1, 2 * x[i], r, 2 * i + 1, c, x.t - i - 1)) >= x.DV) {
r[i + x.t] -= x.DV;
r[i + x.t + 1] = 1;
}
}
if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);
r.s = 0;
r.clamp();
}
// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)
// r != q, this != m. q or r may be null.
function bnpDivRemTo(m, q, r) {
var pm = m.abs();
if (pm.t <= 0) return;
var pt = this.abs();
if (pt.t < pm.t) {
if (q != null) q.fromInt(0);
if (r != null) this.copyTo(r);
return;
}
if (r == null) r = nbi();
var y = nbi(),
ts = this.s,
ms = m.s;
var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus
if (nsh > 0) {
pm.lShiftTo(nsh, y);
pt.lShiftTo(nsh, r);
}
else {
pm.copyTo(y);
pt.copyTo(r);
}
var ys = y.t;
var y0 = y[ys - 1];
if (y0 == 0) return;
var yt = y0 * (1 << this.F1) + ((ys > 1) ? y[ys - 2] >> this.F2 : 0);
var d1 = this.FV / yt,
d2 = (1 << this.F1) / yt,
e = 1 << this.F2;
var i = r.t,
j = i - ys,
t = (q == null) ? nbi() : q;
y.dlShiftTo(j, t);
if (r.compareTo(t) >= 0) {
r[r.t++] = 1;
r.subTo(t, r);
}
Int128.ONE.dlShiftTo(ys, t);
t.subTo(y, y); // "negative" y so we can replace sub with am later
while (y.t < ys) y[y.t++] = 0;
while (--j >= 0) {
// Estimate quotient digit
var qd = (r[--i] == y0) ? this.DM : Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);
if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) { // Try it out
y.dlShiftTo(j, t);
r.subTo(t, r);
while (r[i] < --qd) r.subTo(t, r);
}
}
if (q != null) {
r.drShiftTo(ys, q);
if (ts != ms) Int128.ZERO.subTo(q, q);
}
r.t = ys;
r.clamp();
if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder
if (ts < 0) Int128.ZERO.subTo(r, r);
}
// (public) this mod a
function bnMod(a) {
var r = nbi();
this.abs()
.divRemTo(a, null, r);
if (this.s < 0 && r.compareTo(Int128.ZERO) > 0) a.subTo(r, r);
return r;
}
// Modular reduction using "classic" algorithm
function Classic(m) {
this.m = m;
}
function cConvert(x) {
if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);
else return x;
}
function cRevert(x) {
return x;
}
function cReduce(x) {
x.divRemTo(this.m, null, x);
}
function cMulTo(x, y, r) {
x.multiplyTo(y, r);
this.reduce(r);
}
function cSqrTo(x, r) {
x.squareTo(r);
this.reduce(r);
}
Classic.prototype.convert = cConvert;
Classic.prototype.revert = cRevert;
Classic.prototype.reduce = cReduce;
Classic.prototype.mulTo = cMulTo;
Classic.prototype.sqrTo = cSqrTo;
// (protected) return "-1/this % 2^DB"; useful for Mont. reduction
// justification:
// xy == 1 (mod m)
// xy = 1+km
// xy(2-xy) = (1+km)(1-km)
// x[y(2-xy)] = 1-k^2m^2
// x[y(2-xy)] == 1 (mod m^2)
// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2
// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.
// JS multiply "overflows" differently from C/C++, so care is needed here.
function bnpInvDigit() {
if (this.t < 1) return 0;
var x = this[0];
if ((x & 1) == 0) return 0;
var y = x & 3; // y == 1/x mod 2^2
y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4
y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8
y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16
// last step - calculate inverse mod DV directly;
// assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints
y = (y * (2 - x * y % this.DV)) % this.DV; // y == 1/x mod 2^dbits
// we really want the negative inverse, and -DV < y < DV
return (y > 0) ? this.DV - y : -y;
}
// Montgomery reduction
function Montgomery(m) {
this.m = m;
this.mp = m.invDigit();
this.mpl = this.mp & 0x7fff;
this.mph = this.mp >> 15;
this.um = (1 << (m.DB - 15)) - 1;
this.mt2 = 2 * m.t;
}
// xR mod m
function montConvert(x) {
var r = nbi();
x.abs()
.dlShiftTo(this.m.t, r);
r.divRemTo(this.m, null, r);
if (x.s < 0 && r.compareTo(Int128.ZERO) > 0) this.m.subTo(r, r);
return r;
}
// x/R mod m
function montRevert(x) {
var r = nbi();
x.copyTo(r);
this.reduce(r);
return r;
}
// x = x/R mod m (HAC 14.32)
function montReduce(x) {
while (x.t <= this.mt2) // pad x so am has enough room later
x[x.t++] = 0;
for (var i = 0; i < this.m.t; ++i) {
// faster way of calculating u0 = x[i]*mp mod DV
var j = x[i] & 0x7fff;
var u0 = (j * this.mpl + (((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) & x.DM;
// use am to combine the multiply-shift-add into one call
j = i + this.m.t;
x[j] += this.m.am(0, u0, x, i, 0, this.m.t);
// propagate carry
while (x[j] >= x.DV) {
x[j] -= x.DV;
x[++j]++;
}
}
x.clamp();
x.drShiftTo(this.m.t, x);
if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);
}
// r = "x^2/R mod m"; x != r
function montSqrTo(x, r) {
x.squareTo(r);
this.reduce(r);
}
// r = "xy/R mod m"; x,y != r
function montMulTo(x, y, r) {
x.multiplyTo(y, r);
this.reduce(r);
}
Montgomery.prototype.convert = montConvert;
Montgomery.prototype.revert = montRevert;
Montgomery.prototype.reduce = montReduce;
Montgomery.prototype.mulTo = montMulTo;
Montgomery.prototype.sqrTo = montSqrTo;
// (protected) true iff this is even
function bnpIsEven() {
return ((this.t > 0) ? (this[0] & 1) : this.s) == 0;
}
// (protected) this^e, e < 2^32, doing sqr and mul with "r" (HAC 14.79)
function bnpExp(e, z) {
if (e > 0xffffffff || e < 1) return Int128.ONE;
var r = nbi(),
r2 = nbi(),
g = z.convert(this),
i = nbits(e) - 1;
g.copyTo(r);
while (--i >= 0) {
z.sqrTo(r, r2);
if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);
else {
var t = r;
r = r2;
r2 = t;
}
}
return z.revert(r);
}
// (public) this^e % m, 0 <= e < 2^32
function bnModPowInt(e, m) {
var z;
if (e < 256 || m.isEven()) z = new Classic(m);
else z = new Montgomery(m);
return this.exp(e, z);
}
// protected
Int128.prototype.copyTo = bnpCopyTo;
Int128.prototype.fromInt = bnpFromInt;
Int128.prototype.fromString = bnpFromString;
Int128.prototype.clamp = bnpClamp;
Int128.prototype.dlShiftTo = bnpDLShiftTo;
Int128.prototype.drShiftTo = bnpDRShiftTo;
Int128.prototype.lShiftTo = bnpLShiftTo;
Int128.prototype.rShiftTo = bnpRShiftTo;
Int128.prototype.subTo = bnpSubTo;
Int128.prototype.multiplyTo = bnpMultiplyTo;
Int128.prototype.squareTo = bnpSquareTo;
Int128.prototype.divRemTo = bnpDivRemTo;
Int128.prototype.invDigit = bnpInvDigit;
Int128.prototype.isEven = bnpIsEven;
Int128.prototype.exp = bnpExp;
// public
Int128.prototype.toString = bnToString;
Int128.prototype.negate = bnNegate;
Int128.prototype.abs = bnAbs;
Int128.prototype.compareTo = bnCompareTo;
Int128.prototype.bitLength = bnBitLength;
Int128.prototype.mod = bnMod;
Int128.prototype.modPowInt = bnModPowInt;
// "constants"
Int128.ZERO = nbv(0);
Int128.ONE = nbv(1);
// Copyright (c) 2005-2009 Tom Wu
// All Rights Reserved.
// See "LICENSE" for details.
// Extended JavaScript BN functions, required for RSA private ops.
// Version 1.1: new Int128("0", 10) returns "proper" zero
// Version 1.2: square() API, isProbablePrime fix
// (public)
function bnClone() {
var r = nbi();
this.copyTo(r);
return r;
}
// (public) return value as integer
function bnIntValue() {
if (this.s < 0) {
if (this.t == 1) return this[0] - this.DV;
else if (this.t == 0) return -1;
}
else if (this.t == 1) return this[0];
else if (this.t == 0) return 0;
// assumes 16 < DB < 32
return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];
}
// (public) return value as byte
function bnByteValue() {
return (this.t == 0) ? this.s : (this[0] << 24) >> 24;
}
// (public) return value as short (assumes DB>=16)
function bnShortValue() {
return (this.t == 0) ? this.s : (this[0] << 16) >> 16;
}
// (protected) return x s.t. r^x < DV
function bnpChunkSize(r) {
return Math.floor(Math.LN2 * this.DB / Math.log(r));
}
// (public) 0 if this == 0, 1 if this > 0
function bnSigNum() {
if (this.s < 0) return -1;
else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;
else return 1;
}
// (protected) convert to radix string
function bnpToRadix(b) {
if (b == null) b = 10;
if (this.signum() == 0 || b < 2 || b > 36) return "0";
var cs = this.chunkSize(b);
var a = Math.pow(b, cs);
var d = nbv(a),
y = nbi(),
z = nbi(),
r = "";
this.divRemTo(d, y, z);
while (y.signum() > 0) {
r = (a + z.intValue())
.toString(b)
.substr(1) + r;
y.divRemTo(d, y, z);
}
return z.intValue()
.toString(b) + r;
}
// (protected) convert from radix string
function bnpFromRadix(s, b) {
this.fromInt(0);
if (b == null) b = 10;
var cs = this.chunkSize(b);
var d = Math.pow(b, cs),
mi = false,
j = 0,
w = 0;
for (var i = 0; i < s.length; ++i) {
var x = intAt(s, i);
if (x < 0) {
if (s.charAt(i) == "-" && this.signum() == 0) mi = true;
continue;
}
w = b * w + x;
if (++j >= cs) {
this.dMultiply(d);
this.dAddOffset(w, 0);
j = 0;
w = 0;
}
}
if (j > 0) {
this.dMultiply(Math.pow(b, j));
this.dAddOffset(w, 0);
}
if (mi) Int128.ZERO.subTo(this, this);
}
// (protected) alternate constructor
function bnpFromNumber(a, b, c) {
if ("number" == typeof b) {
// new Int128(int,int,RNG)
if (a < 2) this.fromInt(1);
else {
this.fromNumber(a, c);
if (!this.testBit(a - 1)) // force MSB set
this.bitwiseTo(Int128.ONE.shiftLeft(a - 1), op_or, this);
if (this.isEven()) this.dAddOffset(1, 0); // force odd
while (!this.isProbablePrime(b)) {
this.dAddOffset(2, 0);
if (this.bitLength() > a) this.subTo(Int128.ONE.shiftLeft(a - 1), this);
}
}
}
else {
// new Int128(int,RNG)
var x = [],
t = a & 7;
x.length = (a >> 3) + 1;
b.nextBytes(x);
if (t > 0) x[0] &= ((1 << t) - 1);
else x[0] = 0;
this.fromString(x, 256);
}
}
// (public) convert to bigendian byte array
function bnToByteArray() {
var i = this.t,
r = [];
r[0] = this.s;
var p = this.DB - (i * this.DB) % 8,
d, k = 0;
if (i-- > 0) {
if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p) r[k++] = d | (this.s << (this.DB - p));
while (i >= 0) {
if (p < 8) {
d = (this[i] & ((1 << p) - 1)) << (8 - p);
d |= this[--i] >> (p += this.DB - 8);
}
else {
d = (this[i] >> (p -= 8)) & 0xff;
if (p <= 0) {
p += this.DB;
--i;
}
}
if ((d & 0x80) != 0) d |= -256;
if (k == 0 && (this.s & 0x80) != (d & 0x80))++k;
if (k > 0 || d != this.s) r[k++] = d;
}
}
return r;
}
function bnEquals(a) {
return (this.compareTo(a) == 0);
}
function bnMin(a) {
return (this.compareTo(a) < 0) ? this : a;
}
function bnMax(a) {
return (this.compareTo(a) > 0) ? this : a;
}
// (protected) r = this op a (bitwise)
function bnpBitwiseTo(a, op, r) {
var i, f, m = Math.min(a.t, this.t);
for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);
if (a.t < this.t) {
f = a.s & this.DM;
for (i = m; i < this.t; ++i) r[i] = op(this[i], f);
r.t = this.t;
}
else {
f = this.s & this.DM;
for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);
r.t = a.t;
}
r.s = op(this.s, a.s);
r.clamp();
}
// (public) this & a
function op_and(x, y) {
return x & y;
}
function bnAnd(a) {
var r = nbi();
this.bitwiseTo(a, op_and, r);
return r;
}
// (public) this | a
function op_or(x, y) {
return x | y;
}
function bnOr(a) {
var r = nbi();
this.bitwiseTo(a, op_or, r);
return r;
}
// (public) this ^ a
function op_xor(x, y) {
return x ^ y;
}
function bnXor(a) {
var r = nbi();
this.bitwiseTo(a, op_xor, r);
return r;
}
// (public) this & ~a
function op_andnot(x, y) {
return x & ~y;
}
function bnAndNot(a) {
var r = nbi();
this.bitwiseTo(a, op_andnot, r);
return r;
}
// (public) ~this
function bnNot() {
var r = nbi();
for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];
r.t = this.t;
r.s = ~this.s;
return r;
}
// (public) this << n
function bnShiftLeft(n) {
var r = nbi();
if (n < 0) this.rShiftTo(-n, r);
else this.lShiftTo(n, r);
return r;
}
// (public) this >> n
function bnShiftRight(n) {
var r = nbi();
if (n < 0) this.lShiftTo(-n, r);
else this.rShiftTo(n, r);
return r;
}
// return index of lowest 1-bit in x, x < 2^31
function lbit(x) {
if (x == 0) return -1;
var r = 0;
if ((x & 0xffff) == 0) {
x >>= 16;
r += 16;
}
if ((x & 0xff) == 0) {
x >>= 8;
r += 8;
}
if ((x & 0xf) == 0) {
x >>= 4;
r += 4;
}
if ((x & 3) == 0) {
x >>= 2;
r += 2;
}
if ((x & 1) == 0)++r;
return r;
}
// (public) returns index of lowest 1-bit (or -1 if none)
function bnGetLowestSetBit() {
for (var i = 0; i < this.t; ++i)
if (this[i] != 0) return i * this.DB + lbit(this[i]);
if (this.s < 0) return this.t * this.DB;
return -1;
}
// return number of 1 bits in x
function cbit(x) {
var r = 0;
while (x != 0) {
x &= x - 1;
++r;
}
return r;
}
// (public) return number of set bits
function bnBitCount() {
var r = 0,
x = this.s & this.DM;
for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);
return r;
}
// (public) true iff nth bit is set
function bnTestBit(n) {
var j = Math.floor(n / this.DB);
if (j >= this.t) return (this.s != 0);
return ((this[j] & (1 << (n % this.DB))) != 0);
}
// (protected) this op (1<<n)
function bnpChangeBit(n, op) {
var r = Int128.ONE.shiftLeft(n);
this.bitwiseTo(r, op, r);
return r;
}
// (public) this | (1<<n)
function bnSetBit(n) {
return this.changeBit(n, op_or);
}
// (public) this & ~(1<<n)
function bnClearBit(n) {
return this.changeBit(n, op_andnot);
}
// (public) this ^ (1<<n)
function bnFlipBit(n) {
return this.changeBit(n, op_xor);
}
// (protected) r = this + a
function bnpAddTo(a, r) {
var i = 0,
c = 0,
m = Math.min(a.t, this.t);
while (i < m) {
c += this[i] + a[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
if (a.t < this.t) {
c += a.s;
while (i < this.t) {
c += this[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
c += this.s;
}
else {
c += this.s;
while (i < a.t) {
c += a[i];
r[i++] = c & this.DM;
c >>= this.DB;
}
c += a.s;
}
r.s = (c < 0) ? -1 : 0;
if (c > 0) r[i++] = c;
else if (c < -1) r[i++] = this.DV + c;
r.t = i;
r.clamp();
}
// (public) this + a
function bnAdd(a) {
var r = nbi();
this.addTo(a, r);
return r;
}
// (public) this - a
function bnSubtract(a) {
var r = nbi();
this.subTo(a, r);
return r;
}
// (public) this * a
function bnMultiply(a) {
var r = nbi();
this.multiplyTo(a, r);
return r;
}
// (public) this^2
function bnSquare() {
var r = nbi();
this.squareTo(r);
return r;
}
// (public) this / a
function bnDivide(a) {
var r = nbi();
this.divRemTo(a, r, null);
return r;
}
// (public) this % a
function bnRemainder(a) {
var r = nbi();
this.divRemTo(a, null, r);
return r;
}
// (public) [this/a,this%a]
function bnDivideAndRemainder(a) {
var q = nbi(),
r = nbi();
this.divRemTo(a, q, r);
return new Array(q, r);
}
// (protected) this *= n, this >= 0, 1 < n < DV
function bnpDMultiply(n) {
this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);
++this.t;
this.clamp();
}
// (protected) this += n << w words, this >= 0
function bnpDAddOffset(n, w) {
if (n == 0) return;
while (this.t <= w) this[this.t++] = 0;
this[w] += n;
while (this[w] >= this.DV) {
this[w] -= this.DV;
if (++w >= this.t) this[this.t++] = 0;
++this[w];
}
}
// A "null" reducer
function NullExp() {
}
function nNop(x) {
return x;
}
function nMulTo(x, y, r) {
x.multiplyTo(y, r);
}
function nSqrTo(x, r) {
x.squareTo(r);
}
NullExp.prototype.convert = nNop;
NullExp.prototype.revert = nNop;
NullExp.prototype.mulTo = nMulTo;
NullExp.prototype.sqrTo = nSqrTo;
// (public) this^e
function bnPow(e) {
return this.exp(e, new NullExp());
}
// (protected) r = lower n words of "this * a", a.t <= n
// "this" should be the larger one if appropriate.
function bnpMultiplyLowerTo(a, n, r) {
var i = Math.min(this.t + a.t, n);
r.s = 0; // assumes a,this >= 0
r.t = i;
while (i > 0) r[--i] = 0;
var j;
for (j = r.t - this.t; i < j; ++i) r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);
for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);
r.clamp();
}
// (protected) r = "this * a" without lower n words, n > 0
// "this" should be the larger one if appropriate.
function bnpMultiplyUpperTo(a, n, r) {
--n;
var i = r.t = this.t + a.t - n;
r.s = 0; // assumes a,this >= 0
while (--i >= 0) r[i] = 0;
for (i = Math.max(n - this.t, 0); i < a.t; ++i)
r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);
r.clamp();
r.drShiftTo(1, r);
}
// Barrett modular reduction
function Barrett(m) {
// setup Barrett
this.r2 = nbi();
this.q3 = nbi();
Int128.ONE.dlShiftTo(2 * m.t, this.r2);
this.mu = this.r2.divide(m);
this.m = m;
}
function barrettConvert(x) {
if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);
else if (x.compareTo(this.m) < 0) return x;
else {
var r = nbi();
x.copyTo(r);
this.reduce(r);
return r;
}
}
function barrettRevert(x) {
return x;
}
// x = x mod m (HAC 14.42)
function barrettReduce(x) {
x.drShiftTo(this.m.t - 1, this.r2);
if (x.t > this.m.t + 1) {
x.t = this.m.t + 1;
x.clamp();
}
this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);
this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);
while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);
x.subTo(this.r2, x);
while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);
}
// r = x^2 mod m; x != r
function barrettSqrTo(x, r) {
x.squareTo(r);
this.reduce(r);
}
// r = x*y mod m; x,y != r
function barrettMulTo(x, y, r) {
x.multiplyTo(y, r);
this.reduce(r);
}
Barrett.prototype.convert = barrettConvert;
Barrett.prototype.revert = barrettRevert;
Barrett.prototype.reduce = barrettReduce;
Barrett.prototype.mulTo = barrettMulTo;
Barrett.prototype.sqrTo = barrettSqrTo;
// (public) this^e % m (HAC 14.85)
function bnModPow(e, m) {
var i = e.bitLength(),
k, r = nbv(1),
z;
if (i <= 0) return r;
else if (i < 18) k = 1;
else if (i < 48) k = 3;
else if (i < 144) k = 4;
else if (i < 768) k = 5;
else k = 6;
if (i < 8) z = new Classic(m);
else if (m.isEven()) z = new Barrett(m);
else z = new Montgomery(m);
// precomputation
var g = [],
n = 3,
k1 = k - 1,
km = (1 << k) - 1;
g[1] = z.convert(this);
if (k > 1) {
var g2 = nbi();
z.sqrTo(g[1], g2);
while (n <= km) {
g[n] = nbi();
z.mulTo(g2, g[n - 2], g[n]);
n += 2;
}
}
var j = e.t - 1,
w, is1 = true,
r2 = nbi(),
t;
i = nbits(e[j]) - 1;
while (j >= 0) {
if (i >= k1) w = (e[j] >> (i - k1)) & km;
else {
w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);
if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);
}
n = k;
while ((w & 1) == 0) {
w >>= 1;
--n;
}
if ((i -= n) < 0) {
i += this.DB;
--j;
}
if (is1) { // ret == 1, don't bother squaring or multiplying it
g[w].copyTo(r);
is1 = false;
}
else {
while (n > 1) {
z.sqrTo(r, r2);
z.sqrTo(r2, r);
n -= 2;
}
if (n > 0) z.sqrTo(r, r2);
else {
t = r;
r = r2;
r2 = t;
}
z.mulTo(r2, g[w], r);
}
while (j >= 0 && (e[j] & (1 << i)) == 0) {
z.sqrTo(r, r2);
t = r;
r = r2;
r2 = t;
if (--i < 0) {
i = this.DB - 1;
--j;
}
}
}
return z.revert(r);
}
// (public) gcd(this,a) (HAC 14.54)
function bnGCD(a) {
var x = (this.s < 0) ? this.negate() : this.clone();
var y = (a.s < 0) ? a.negate() : a.clone();
if (x.compareTo(y) < 0) {
var t = x;
x = y;
y = t;
}
var i = x.getLowestSetBit(),
g = y.getLowestSetBit();
if (g < 0) return x;
if (i < g) g = i;
if (g > 0) {
x.rShiftTo(g, x);
y.rShiftTo(g, y);
}
while (x.signum() > 0) {
if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);
if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);
if (x.compareTo(y) >= 0) {
x.subTo(y, x);
x.rShiftTo(1, x);
}
else {
y.subTo(x, y);
y.rShiftTo(1, y);
}
}
if (g > 0) y.lShiftTo(g, y);
return y;
}
// (protected) this % n, n < 2^26
function bnpModInt(n) {
if (n <= 0) return 0;
var d = this.DV % n,
r = (this.s < 0) ? n - 1 : 0;
if (this.t > 0) if (d == 0) r = this[0] % n;
else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;
return r;
}
// (public) 1/this % m (HAC 14.61)
function bnModInverse(m) {
var ac = m.isEven();
if ((this.isEven() && ac) || m.signum() == 0) return Int128.ZERO;
var u = m.clone(),
v = this.clone();
var a = nbv(1),
b = nbv(0),
c = nbv(0),
d = nbv(1);
while (u.signum() != 0) {
while (u.isEven()) {
u.rShiftTo(1, u);
if (ac) {
if (!a.isEven() || !b.isEven()) {
a.addTo(this, a);
b.subTo(m, b);
}
a.rShiftTo(1, a);
}
else if (!b.isEven()) b.subTo(m, b);
b.rShiftTo(1, b);
}
while (v.isEven()) {
v.rShiftTo(1, v);
if (ac) {
if (!c.isEven() || !d.isEven()) {
c.addTo(this, c);
d.subTo(m, d);
}
c.rShiftTo(1, c);
}
else if (!d.isEven()) d.subTo(m, d);
d.rShiftTo(1, d);
}
if (u.compareTo(v) >= 0) {
u.subTo(v, u);
if (ac) a.subTo(c, a);
b.subTo(d, b);
}
else {
v.subTo(u, v);
if (ac) c.subTo(a, c);
d.subTo(b, d);
}
}
if (v.compareTo(Int128.ONE) != 0) return Int128.ZERO;
if (d.compareTo(m) >= 0) return d.subtract(m);
if (d.signum() < 0) d.addTo(m, d);
else return d;
if (d.signum() < 0) return d.add(m);
else return d;
}
var lowprimes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227, 229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359, 367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499, 503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647, 653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811, 821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971, 977, 983, 991, 997];
var lplim = (1 << 26) / lowprimes[lowprimes.length - 1];
// (public) test primality with certainty >= 1-.5^t
function bnIsProbablePrime(t) {
var i, x = this.abs();
if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {
for (i = 0; i < lowprimes.length; ++i)
if (x[0] == lowprimes[i]) return true;
return false;
}
if (x.isEven()) return false;
i = 1;
while (i < lowprimes.length) {
var m = lowprimes[i],
j = i + 1;
while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];
m = x.modInt(m);
while (i < j) if (m % lowprimes[i++] == 0) return false;
}
return x.millerRabin(t);
}
// (protected) true if probably prime (HAC 4.24, Miller-Rabin)
function bnpMillerRabin(t) {
var n1 = this.subtract(Int128.ONE);
var k = n1.getLowestSetBit();
if (k <= 0) return false;
var r = n1.shiftRight(k);
t = (t + 1) >> 1;
if (t > lowprimes.length) t = lowprimes.length;
var a = nbi();
for (var i = 0; i < t; ++i) {
//Pick bases at random, instead of starting at 2
a.fromInt(lowprimes[Math.floor(Math.random() * lowprimes.length)]);
var y = a.modPow(r, this);
if (y.compareTo(Int128.ONE) != 0 && y.compareTo(n1) != 0) {
var j = 1;
while (j++ < k && y.compareTo(n1) != 0) {
y = y.modPowInt(2, this);
if (y.compareTo(Int128.ONE) == 0) return false;
}
if (y.compareTo(n1) != 0) return false;
}
}
return true;
}
// protected
Int128.prototype.chunkSize = bnpChunkSize;
Int128.prototype.toRadix = bnpToRadix;
Int128.prototype.fromRadix = bnpFromRadix;
Int128.prototype.fromNumber = bnpFromNumber;
Int128.prototype.bitwiseTo = bnpBitwiseTo;
Int128.prototype.changeBit = bnpChangeBit;
Int128.prototype.addTo = bnpAddTo;
Int128.prototype.dMultiply = bnpDMultiply;
Int128.prototype.dAddOffset = bnpDAddOffset;
Int128.prototype.multiplyLowerTo = bnpMultiplyLowerTo;
Int128.prototype.multiplyUpperTo = bnpMultiplyUpperTo;
Int128.prototype.modInt = bnpModInt;
Int128.prototype.millerRabin = bnpMillerRabin;
// public
Int128.prototype.clone = bnClone;
Int128.prototype.intValue = bnIntValue;
Int128.prototype.byteValue = bnByteValue;
Int128.prototype.shortValue = bnShortValue;
Int128.prototype.signum = bnSigNum;
Int128.prototype.toByteArray = bnToByteArray;
Int128.prototype.equals = bnEquals;
Int128.prototype.min = bnMin;
Int128.prototype.max = bnMax;
Int128.prototype.and = bnAnd;
Int128.prototype.or = bnOr;
Int128.prototype.xor = bnXor;
Int128.prototype.andNot = bnAndNot;
Int128.prototype.not = bnNot;
Int128.prototype.shiftLeft = bnShiftLeft;
Int128.prototype.shiftRight = bnShiftRight;
Int128.prototype.getLowestSetBit = bnGetLowestSetBit;
Int128.prototype.bitCount = bnBitCount;
Int128.prototype.testBit = bnTestBit;
Int128.prototype.setBit = bnSetBit;
Int128.prototype.clearBit = bnClearBit;
Int128.prototype.flipBit = bnFlipBit;
Int128.prototype.add = bnAdd;
Int128.prototype.subtract = bnSubtract;
Int128.prototype.multiply = bnMultiply;
Int128.prototype.divide = bnDivide;
Int128.prototype.remainder = bnRemainder;
Int128.prototype.divideAndRemainder = bnDivideAndRemainder;
Int128.prototype.modPow = bnModPow;
Int128.prototype.modInverse = bnModInverse;
Int128.prototype.pow = bnPow;
Int128.prototype.gcd = bnGCD;
Int128.prototype.isProbablePrime = bnIsProbablePrime;
// JSBN-specific extension
Int128.prototype.square = bnSquare;
// end of Int128 section
/*
// Uncomment the following two lines if you want to use Int128 outside ClipperLib
if (typeof(document) !== "undefined") window.Int128 = Int128;
else self.Int128 = Int128;
*/
// Here starts the actual Clipper library:
ClipperLib.Math_Abs_Int64 = ClipperLib.Math_Abs_Int32 = ClipperLib.Math_Abs_Double = function (a) {
return Math.abs(a);
};
ClipperLib.Math_Max_Int32_Int32 = function (a, b) {
return Math.max(a, b);
};
/*
-----------------------------------
cast_32 speedtest: http://jsperf.com/truncate-float-to-integer/2
-----------------------------------
*/
if (browser.msie || browser.opera || browser.safari) ClipperLib.Cast_Int32 = function (a) {
return a | 0;
};
else ClipperLib.Cast_Int32 = function (a) { // eg. browser.chrome || browser.chromium || browser.firefox
return ~~a;
};
/*
--------------------------
cast_64 speedtests: http://jsperf.com/truncate-float-to-integer
Chrome: bitwise_not_floor
Firefox17: toInteger (typeof test)
IE9: bitwise_or_floor
IE7 and IE8: to_parseint
Chromium: to_floor_or_ceil
Firefox3: to_floor_or_ceil
Firefox15: to_floor_or_ceil
Opera: to_floor_or_ceil
Safari: to_floor_or_ceil
--------------------------
*/
if (browser.chrome) ClipperLib.Cast_Int64 = function (a) {
if (a < -2147483648 || a > 2147483647)
return a < 0 ? Math.ceil(a) : Math.floor(a);
else return ~~a;
};
else if (browser.firefox && typeof(Number.toInteger) == "function") ClipperLib.Cast_Int64 = function (a) {
return Number.toInteger(a);
};
else if (browser.msie7 || browser.msie8) ClipperLib.Cast_Int64 = function (a) {
return parseInt(a, 10);
};
else if (browser.msie) ClipperLib.Cast_Int64 = function (a) {
if (a < -2147483648 || a > 2147483647)
return a < 0 ? Math.ceil(a) : Math.floor(a);
return a | 0;
};
// eg. browser.chromium || browser.firefox || browser.opera || browser.safari
else ClipperLib.Cast_Int64 = function (a) {
return a < 0 ? Math.ceil(a) : Math.floor(a);
};
ClipperLib.Clear = function (a) {
a.length = 0;
};
ClipperLib.MaxSteps = 222; // How many steps at maximum in arc in BuildArc() function
ClipperLib.PI = 3.141592653589793;
ClipperLib.PI2 = 2 * 3.141592653589793;
ClipperLib.IntPoint = function () {
var a = arguments;
if (a.length == 1) {
this.X = a[0].X;
this.Y = a[0].Y;
}
if (a.length == 2) {
this.X = a[0];
this.Y = a[1];
}
};
ClipperLib.IntRect = function () {
var a = arguments;
if (a.length == 4) // function (l, t, r, b)
{
var l = a[0],
t = a[1],
r = a[2],
b = a[3];
this.left = l;
this.top = t;
this.right = r;
this.bottom = b;
}
else {
this.left = 0;
this.top = 0;
this.right = 0;
this.bottom = 0;
}
};
ClipperLib.Polygon = function () {
return [];
};
ClipperLib.Polygons = function () {
return []; // Was previously [[]], but caused problems when pushed
};
ClipperLib.ExPolygons = function () {
var a = [];
a.exPolygons = true; // this is needed to make "overloading" possible in Execute
return a;
}
ClipperLib.ExPolygon = function () {
this.outer = null;
this.holes = null;
};
ClipperLib.ClipType = {
ctIntersection: 0,
ctUnion: 1,
ctDifference: 2,
ctXor: 3
};
ClipperLib.PolyType = {
ptSubject: 0,
ptClip: 1
};
ClipperLib.PolyFillType = {
pftEvenOdd: 0,
pftNonZero: 1,
pftPositive: 2,
pftNegative: 3
};
ClipperLib.JoinType = {
jtSquare: 0,
jtRound: 1,
jtMiter: 2
};
ClipperLib.EdgeSide = {
esLeft: 1,
esRight: 2
};
ClipperLib.Protects = {
ipNone: 0,
ipLeft: 1,
ipRight: 2,
ipBoth: 3
};
ClipperLib.Direction = {
dRightToLeft: 0,
dLeftToRight: 1
};
ClipperLib.TEdge = function () {
this.xbot = 0;
this.ybot = 0;
this.xcurr = 0;
this.ycurr = 0;
this.xtop = 0;
this.ytop = 0;
this.dx = 0;
this.deltaX = 0;
this.deltaY = 0;
this.tmpX = 0;
this.polyType = ClipperLib.PolyType.ptSubject;
this.side = null; //= ClipperLib.EdgeSide.esNeither;
this.windDelta = 0;
this.windCnt = 0;
this.windCnt2 = 0;
this.outIdx = 0;
this.next = null;
this.prev = null;
this.nextInLML = null;
this.nextInAEL = null;
this.prevInAEL = null;
this.nextInSEL = null;
this.prevInSEL = null;
};
ClipperLib.IntersectNode = function () {
this.edge1 = null;
this.edge2 = null;
this.pt = null;
this.next = null;
};
ClipperLib.LocalMinima = function () {
this.Y = 0;
this.leftBound = null;
this.rightBound = null;
this.next = null;
};
ClipperLib.Scanbeam = function () {
this.Y = 0;
this.next = null;
};
ClipperLib.OutRec = function () {
this.idx = 0;
this.isHole = false;
this.FirstLeft = null;
this.AppendLink = null;
this.pts = null;
this.bottomPt = null;
};
ClipperLib.OutPt = function () {
this.idx = 0;
this.pt = null;
this.next = null;
this.prev = null;
};
ClipperLib.JoinRec = function () {
this.pt1a = null;
this.pt1b = null;
this.poly1Idx = 0;
this.pt2a = null;
this.pt2b = null;
this.poly2Idx = 0;
};
ClipperLib.HorzJoinRec = function () {
this.edge = null;
this.savedIdx = 0;
};
ClipperLib.ClipperBase = function () {
this.m_MinimaList = null;
this.m_CurrentLM = null;
this.m_edges = [
[]
]; // 2-dimensional array
this.m_UseFullRange = false;
};
// Ranges are in original C# too high for Javascript (in current state 2012 December):
// protected const double horizontal = -3.4E+38;
// internal const Int64 loRange = 0x3FFFFFFF; // = 1073741823 = sqrt(2^63 -1)/2
// internal const Int64 hiRange = 0x3FFFFFFFFFFFFFFFL; // = 4611686018427387903 = sqrt(2^127 -1)/2
// So had to adjust them to more suitable:
ClipperLib.ClipperBase.horizontal = -9007199254740992; //-2^53
ClipperLib.ClipperBase.loRange = 47453132; // sqrt(2^53 -1)/2
ClipperLib.ClipperBase.hiRange = 4503599627370495; // sqrt(2^106 -1)/2
// If JS some day supports truly 64-bit integers, then these ranges can be as in C#
// and biginteger library can be more simpler (as then 128bit can be represented as two 64bit numbers)
ClipperLib.ClipperBase.PointsEqual = function (pt1, pt2) {
return (pt1.X == pt2.X && pt1.Y == pt2.Y);
};
ClipperLib.ClipperBase.prototype.PointIsVertex = function (pt, pp) {
var pp2 = pp;
do {
if (ClipperLib.ClipperBase.PointsEqual(pp2.pt, pt)) return true;
pp2 = pp2.next;
}
while (pp2 != pp);
return false;
};
ClipperLib.ClipperBase.prototype.PointInPolygon = function (pt, pp, UseFulllongRange) {
var pp2 = pp;
var result = false;
if (UseFulllongRange) {
do {
if ((((pp2.pt.Y <= pt.Y) && (pt.Y < pp2.prev.pt.Y)) || ((pp2.prev.pt.Y <= pt.Y) && (pt.Y < pp2.pt.Y))) && new Int128(pt.X - pp2.pt.X)
.compareTo(
new Int128(pp2.prev.pt.X - pp2.pt.X)
.multiply(new Int128(pt.Y - pp2.pt.Y))
.divide(
new Int128(pp2.prev.pt.Y - pp2.pt.Y))) < 0) result = !result;
pp2 = pp2.next;
}
while (pp2 != pp);
}
else {
do {
if ((((pp2.pt.Y <= pt.Y) && (pt.Y < pp2.prev.pt.Y)) || ((pp2.prev.pt.Y <= pt.Y) && (pt.Y < pp2.pt.Y))) && (pt.X - pp2.pt.X < (pp2.prev.pt.X - pp2.pt.X) * (pt.Y - pp2.pt.Y) / (pp2.prev.pt.Y - pp2.pt.Y))) result = !result;
pp2 = pp2.next;
}
while (pp2 != pp);
}
return result;
};
ClipperLib.ClipperBase.prototype.SlopesEqual = ClipperLib.ClipperBase.SlopesEqual = function () {
var a = arguments;
var e1, e2, pt1, pt2, pt3, pt4, UseFullRange;
if (a.length == 3) // function (e1, e2, UseFullRange)
{
e1 = a[0], e2 = a[1], UseFullRange = a[2];
if (UseFullRange) return new Int128(e1.deltaY)
.multiply(new Int128(e2.deltaX))
.toString() == new Int128(e1.deltaX)
.multiply(new Int128(e2.deltaY))
.toString();
else return (e1.deltaY) * (e2.deltaX) == (e1.deltaX) * (e2.deltaY);
}
else if (a.length == 4) // function (pt1, pt2, pt3, UseFullRange)
{
pt1 = a[0], pt2 = a[1], pt3 = a[2], UseFullRange = a[3];
if (UseFullRange) return new Int128(pt1.Y - pt2.Y)
.multiply(new Int128(pt2.X - pt3.X))
.toString() == new Int128(pt1.X - pt2.X)
.multiply(new Int128(pt2.Y - pt3.Y))
.toString();
else return (pt1.Y - pt2.Y) * (pt2.X - pt3.X) - (pt1.X - pt2.X) * (pt2.Y - pt3.Y) == 0;
}
else if (a.length == 5) // function (pt1, pt2, pt3, pt4, UseFullRange)
{
pt1 = a[0], pt2 = a[1], pt3 = a[2], pt4 = a[3], UseFullRange = a[4];
if (UseFullRange) return new Int128(pt1.Y - pt2.Y)
.multiply(new Int128(pt3.X - pt4.X))
.toString() == new Int128(pt1.X - pt2.X)
.multiply(new Int128(pt3.Y - pt4.Y))
.toString();
else return (pt1.Y - pt2.Y) * (pt3.X - pt4.X) - (pt1.X - pt2.X) * (pt3.Y - pt4.Y) == 0;
}
};
ClipperLib.ClipperBase.prototype.Clear = function () {
this.DisposeLocalMinimaList();
for (var i = 0; i < this.m_edges.length; ++i) {
for (var j = 0; j < this.m_edges[i].length; ++j)
this.m_edges[i][j] = null;
ClipperLib.Clear(this.m_edges[i]);
}
ClipperLib.Clear(this.m_edges);
this.m_UseFullRange = false;
};
ClipperLib.ClipperBase.prototype.DisposeLocalMinimaList = function () {
while (this.m_MinimaList != null) {
var tmpLm = this.m_MinimaList.next;
this.m_MinimaList = null;
this.m_MinimaList = tmpLm;
}
this.m_CurrentLM = null;
};
ClipperLib.ClipperBase.prototype.AddPolygons = function (ppg, polyType) {
var result = false;
var res = false;
if (!(ppg instanceof Array)) return result;
for (var i = 0; i < ppg.length; ++i) {
res = this.AddPolygon(ppg[i], polyType, true);
if (res && res != "exceed") result = true;
else if (res == "exceed") break;
}
if (res == "exceed") ClipperLib.Error("Coordinate exceeds range bounds in AddPolygons().");
return result;
};
ClipperLib.ClipperBase.prototype.AddPolygon = function (pg, polyType, multiple) {
if (!(pg instanceof Array)) return false;
var len = pg.length;
if (len < 3) return false;
var p = new ClipperLib.Polygon();
p.push(new ClipperLib.IntPoint(pg[0].X, pg[0].Y));
var j = 0;
var i;
var exceed = false;
for (i = 1; i < len; ++i) {
var maxVal;
if (this.m_UseFullRange) maxVal = ClipperLib.ClipperBase.hiRange;
else maxVal = ClipperLib.ClipperBase.loRange;
if (ClipperLib.Math_Abs_Int64(pg[i].X) > maxVal || ClipperLib.Math_Abs_Int64(pg[i].Y) > maxVal) {
if (ClipperLib.Math_Abs_Int64(pg[i].X) > ClipperLib.ClipperBase.hiRange || ClipperLib.Math_Abs_Int64(pg[i].Y) > ClipperLib.ClipperBase.hiRange) {
if (typeof(multiple) != "undefined") return "exceed";
exceed = true;
break;
}
maxVal = ClipperLib.ClipperBase.hiRange;
this.m_UseFullRange = true;
}
if (ClipperLib.ClipperBase.PointsEqual(p[j], pg[i])) continue;
else if (j > 0 && this.SlopesEqual(p[j - 1], p[j], pg[i], this.m_UseFullRange)) {
if (ClipperLib.ClipperBase.PointsEqual(p[j - 1], pg[i])) j--;
}
else j++;
if (j < p.length) p[j] = pg[i];
else p.push(new ClipperLib.IntPoint(pg[i].X, pg[i].Y));
}
if (exceed && typeof(multiple) == "undefined")
ClipperLib.Error("Coordinate exceeds range bounds in AddPolygon()");
if (j < 2) return false;
len = j + 1;
while (len > 2) {
if (ClipperLib.ClipperBase.PointsEqual(p[j], p[0])) j--;
else if (ClipperLib.ClipperBase.PointsEqual(p[0], p[1]) || this.SlopesEqual(p[j], p[0], p[1], this.m_UseFullRange)) p[0] = p[j--];
else if (this.SlopesEqual(p[j - 1], p[j], p[0], this.m_UseFullRange)) j--;
else if (this.SlopesEqual(p[0], p[1], p[2], this.m_UseFullRange)) {
for (i = 2; i <= j; ++i)
p[i - 1] = p[i];
j--;
}
else break;
len--;
}
if (len < 3) return false;
var edges = [];
for (i = 0; i < len; i++)
edges.push(new ClipperLib.TEdge());
this.m_edges.push(edges);
edges[0].xcurr = p[0].X;
edges[0].ycurr = p[0].Y;
this.InitEdge(edges[len - 1], edges[0], edges[len - 2], p[len - 1], polyType);
for (i = len - 2; i > 0; --i)
this.InitEdge(edges[i], edges[i + 1], edges[i - 1], p[i], polyType);
this.InitEdge(edges[0], edges[1], edges[len - 1], p[0], polyType);
var e = edges[0];
var eHighest = e;
do {
e.xcurr = e.xbot;
e.ycurr = e.ybot;
if (e.ytop < eHighest.ytop) eHighest = e;
e = e.next;
}
while (e != edges[0]);
if (eHighest.windDelta > 0) eHighest = eHighest.next;
if (eHighest.dx == ClipperLib.ClipperBase.horizontal) eHighest = eHighest.next;
e = eHighest;
do {
e = this.AddBoundsToLML(e);
}
while (e != eHighest);
return true;
};
ClipperLib.ClipperBase.prototype.InitEdge = function (e, eNext, ePrev, pt, polyType) {
e.next = eNext;
e.prev = ePrev;
e.xcurr = pt.X;
e.ycurr = pt.Y;
if (e.ycurr >= e.next.ycurr) {
e.xbot = e.xcurr;
e.ybot = e.ycurr;
e.xtop = e.next.xcurr;
e.ytop = e.next.ycurr;
e.windDelta = 1;
}
else {
e.xtop = e.xcurr;
e.ytop = e.ycurr;
e.xbot = e.next.xcurr;
e.ybot = e.next.ycurr;
e.windDelta = -1;
}
this.SetDx(e);
e.polyType = polyType;
e.outIdx = -1;
};
ClipperLib.ClipperBase.prototype.SetDx = function (e) {
e.deltaX = (e.xtop - e.xbot);
e.deltaY = (e.ytop - e.ybot);
if (e.deltaY == 0) e.dx = ClipperLib.ClipperBase.horizontal;
else e.dx = (e.deltaX) / (e.deltaY);
};
ClipperLib.ClipperBase.prototype.AddBoundsToLML = function (e) {
e.nextInLML = null;
e = e.next;
for (; ;) {
if (e.dx == ClipperLib.ClipperBase.horizontal) {
if (e.next.ytop < e.ytop && e.next.xbot > e.prev.xbot) break;
if (e.xtop != e.prev.xbot) this.SwapX(e);
e.nextInLML = e.prev;
}
else if (e.ycurr == e.prev.ycurr) break;
else e.nextInLML = e.prev;
e = e.next;
}
var newLm = new ClipperLib.LocalMinima();
newLm.next = null;
newLm.Y = e.prev.ybot;
if (e.dx == ClipperLib.ClipperBase.horizontal) {
if (e.xbot != e.prev.xbot) this.SwapX(e);
newLm.leftBound = e.prev;
newLm.rightBound = e;
}
else if (e.dx < e.prev.dx) {
newLm.leftBound = e.prev;
newLm.rightBound = e;
}
else {
newLm.leftBound = e;
newLm.rightBound = e.prev;
}
newLm.leftBound.side = ClipperLib.EdgeSide.esLeft;
newLm.rightBound.side = ClipperLib.EdgeSide.esRight;
this.InsertLocalMinima(newLm);
for (; ;) {
if (e.next.ytop == e.ytop && e.next.dx != ClipperLib.ClipperBase.horizontal) break;
e.nextInLML = e.next;
e = e.next;
if (e.dx == ClipperLib.ClipperBase.horizontal && e.xbot != e.prev.xtop) this.SwapX(e);
}
return e.next;
};
ClipperLib.ClipperBase.prototype.InsertLocalMinima = function (newLm) {
if (this.m_MinimaList == null) {
this.m_MinimaList = newLm;
}
else if (newLm.Y >= this.m_MinimaList.Y) {
newLm.next = this.m_MinimaList;
this.m_MinimaList = newLm;
}
else {
var tmpLm = this.m_MinimaList;
while (tmpLm.next != null && (newLm.Y < tmpLm.next.Y))
tmpLm = tmpLm.next;
newLm.next = tmpLm.next;
tmpLm.next = newLm;
}
};
ClipperLib.ClipperBase.prototype.PopLocalMinima = function () {
if (this.m_CurrentLM == null) return;
this.m_CurrentLM = this.m_CurrentLM.next;
};
ClipperLib.ClipperBase.prototype.SwapX = function (e) {
e.xcurr = e.xtop;
e.xtop = e.xbot;
e.xbot = e.xcurr;
};
ClipperLib.ClipperBase.prototype.Reset = function () {
this.m_CurrentLM = this.m_MinimaList;
var lm = this.m_MinimaList;
while (lm != null) {
var e = lm.leftBound;
while (e != null) {
e.xcurr = e.xbot;
e.ycurr = e.ybot;
e.side = ClipperLib.EdgeSide.esLeft;
e.outIdx = -1;
e = e.nextInLML;
}
e = lm.rightBound;
while (e != null) {
e.xcurr = e.xbot;
e.ycurr = e.ybot;
e.side = ClipperLib.EdgeSide.esRight;
e.outIdx = -1;
e = e.nextInLML;
}
lm = lm.next;
}
return;
};
ClipperLib.ClipperBase.prototype.GetBounds = function () {
var result = new ClipperLib.IntRect();
var lm = this.m_MinimaList;
if (lm == null) return result;
result.left = lm.leftBound.xbot;
result.top = lm.leftBound.ybot;
result.right = lm.leftBound.xbot;
result.bottom = lm.leftBound.ybot;
while (lm != null) {
if (lm.leftBound.ybot > result.bottom) result.bottom = lm.leftBound.ybot;
var e = lm.leftBound;
for (; ;) {
var bottomE = e;
while (e.nextInLML != null) {
if (e.xbot < result.left) result.left = e.xbot;
if (e.xbot > result.right) result.right = e.xbot;
e = e.nextInLML;
}
if (e.xbot < result.left) result.left = e.xbot;
if (e.xbot > result.right) result.right = e.xbot;
if (e.xtop < result.left) result.left = e.xtop;
if (e.xtop > result.right) result.right = e.xtop;
if (e.ytop < result.top) result.top = e.ytop;
if (bottomE == lm.leftBound) e = lm.rightBound;
else break;
}
lm = lm.next;
}
return result;
};
ClipperLib.Clipper = function () {
this.m_PolyOuts = null;
this.m_ClipType = ClipperLib.ClipType.ctIntersection;
this.m_Scanbeam = null;
this.m_ActiveEdges = null;
this.m_SortedEdges = null;
this.m_IntersectNodes = null;
this.m_ExecuteLocked = false;
this.m_ClipFillType = ClipperLib.PolyFillType.pftEvenOdd;
this.m_SubjFillType = ClipperLib.PolyFillType.pftEvenOdd;
this.m_Joins = null;
this.m_HorizJoins = null;
this.m_ReverseOutput = false;
this.m_UsingExPolygons = false;
ClipperLib.ClipperBase.call(this);
this.m_Scanbeam = null;
this.m_ActiveEdges = null;
this.m_SortedEdges = null;
this.m_IntersectNodes = null;
this.m_ExecuteLocked = false;
this.m_PolyOuts = [];
this.m_Joins = [];
this.m_HorizJoins = [];
this.m_ReverseOutput = false;
this.m_UsingExPolygons = false;
};
ClipperLib.Clipper.prototype.Clear = function () {
if (this.m_edges.length == 0) return;
this.DisposeAllPolyPts();
ClipperLib.ClipperBase.prototype.Clear.call(this);
};
ClipperLib.Clipper.prototype.DisposeScanbeamList = function () {
while (this.m_Scanbeam != null) {
var sb2 = this.m_Scanbeam.next;
this.m_Scanbeam = null;
this.m_Scanbeam = sb2;
}
};
ClipperLib.Clipper.prototype.Reset = function () {
ClipperLib.ClipperBase.prototype.Reset.call(this);
this.m_Scanbeam = null;
this.m_ActiveEdges = null;
this.m_SortedEdges = null;
this.DisposeAllPolyPts();
var lm = this.m_MinimaList;
while (lm != null) {
this.InsertScanbeam(lm.Y);
this.InsertScanbeam(lm.leftBound.ytop);
lm = lm.next;
}
};
ClipperLib.Clipper.prototype.get_ReverseSolution = function () {
return this.m_ReverseOutput;
};
ClipperLib.Clipper.prototype.set_ReverseSolution = function (value) {
this.m_ReverseOutput = value;
};
ClipperLib.Clipper.prototype.InsertScanbeam = function (Y) {
var newSb;
if (this.m_Scanbeam == null) {
this.m_Scanbeam = new ClipperLib.Scanbeam();
this.m_Scanbeam.next = null;
this.m_Scanbeam.Y = Y;
}
else if (Y > this.m_Scanbeam.Y) {
newSb = new ClipperLib.Scanbeam();
newSb.Y = Y;
newSb.next = this.m_Scanbeam;
this.m_Scanbeam = newSb;
}
else {
var sb2 = this.m_Scanbeam;
while (sb2.next != null && (Y <= sb2.next.Y))
sb2 = sb2.next;
if (Y == sb2.Y) return;
newSb = new ClipperLib.Scanbeam();
newSb.Y = Y;
newSb.next = sb2.next;
sb2.next = newSb;
}
};
ClipperLib.Clipper.prototype.Execute = function (clipType, solution, subjFillType, clipFillType) {
var succeeded;
if (arguments.length == 2) {
subjFillType = ClipperLib.PolyFillType.pftEvenOdd;
clipFillType = ClipperLib.PolyFillType.pftEvenOdd;
}
if (typeof(solution.exPolygons) == "undefined") // hacky way to test if solution is not exPolygons
{
if (this.m_ExecuteLocked) return false;
this.m_ExecuteLocked = true;
ClipperLib.Clear(solution);
this.m_SubjFillType = subjFillType;
this.m_ClipFillType = clipFillType;
this.m_ClipType = clipType;
this.m_UsingExPolygons = false;
succeeded = this.ExecuteInternal();
if (succeeded) {
this.BuildResult(solution);
}
this.m_ExecuteLocked = false;
return succeeded;
}
else {
if (this.m_ExecuteLocked) return false;
this.m_ExecuteLocked = true;
ClipperLib.Clear(solution);
this.m_SubjFillType = subjFillType;
this.m_ClipFillType = clipFillType;
this.m_ClipType = clipType;
this.m_UsingExPolygons = true;
succeeded = this.ExecuteInternal();
if (succeeded) {
this.BuildResultEx(solution);
}
this.m_ExecuteLocked = false;
return succeeded;
}
};
ClipperLib.Clipper.prototype.PolySort = function (or1, or2) {
if (or1 == or2) return 0;
else if (or1.pts == null || or2.pts == null) {
if ((or1.pts == null) != (or2.pts == null)) {
return or1.pts == null ? 1 : -1;
}
else return 0;
}
var i1, i2;
if (or1.isHole) i1 = or1.FirstLeft.idx;
else i1 = or1.idx;
if (or2.isHole) i2 = or2.FirstLeft.idx;
else i2 = or2.idx;
var result = i1 - i2;
if (result == 0 && (or1.isHole != or2.isHole)) {
return or1.isHole ? 1 : -1;
}
return result;
};
ClipperLib.Clipper.prototype.FindAppendLinkEnd = function (outRec) {
while (outRec.AppendLink != null)
outRec = outRec.AppendLink;
return outRec;
};
ClipperLib.Clipper.prototype.FixHoleLinkage = function (outRec) {
var tmp;
if (outRec.bottomPt != null) tmp = this.m_PolyOuts[outRec.bottomPt.idx].FirstLeft;
else tmp = outRec.FirstLeft;
if (outRec == tmp) ClipperLib.Error("HoleLinkage error");
if (tmp != null) {
if (tmp.AppendLink != null) tmp = this.FindAppendLinkEnd(tmp);
if (tmp == outRec) tmp = null;
else if (tmp.isHole) {
this.FixHoleLinkage(tmp);
tmp = tmp.FirstLeft;
}
}
outRec.FirstLeft = tmp;
if (tmp == null) outRec.isHole = false;
outRec.AppendLink = null;
};
ClipperLib.Clipper.prototype.ExecuteInternal = function () {
var succeeded;
try {
this.Reset();
if (this.m_CurrentLM == null) return true;
var botY = this.PopScanbeam();
do {
this.InsertLocalMinimaIntoAEL(botY);
ClipperLib.Clear(this.m_HorizJoins);
this.ProcessHorizontals();
var topY = this.PopScanbeam();
succeeded = this.ProcessIntersections(botY, topY);
if (!succeeded) break;
this.ProcessEdgesAtTopOfScanbeam(topY);
botY = topY;
}
while (this.m_Scanbeam != null);
}
catch ($$e1) {
succeeded = false;
}
if (succeeded) {
var outRec;
for (var i = 0; i < this.m_PolyOuts.length; i++) {
outRec = this.m_PolyOuts[i];
if (outRec.pts == null) continue;
this.FixupOutPolygon(outRec);
if (outRec.pts == null) continue;
if (outRec.isHole && this.m_UsingExPolygons) this.FixHoleLinkage(outRec);
if ((outRec.isHole ^ this.m_ReverseOutput) == (this.Area(outRec, this.m_UseFullRange) > 0))
this.ReversePolyPtLinks(outRec.pts);
}
this.JoinCommonEdges();
if (this.m_UsingExPolygons) this.m_PolyOuts.sort(this.PolySort);
}
ClipperLib.Clear(this.m_Joins);
ClipperLib.Clear(this.m_HorizJoins);
return succeeded;
};
ClipperLib.Clipper.prototype.PopScanbeam = function () {
var Y = this.m_Scanbeam.Y;
var sb2 = this.m_Scanbeam;
this.m_Scanbeam = this.m_Scanbeam.next;
sb2 = null;
return Y;
};
ClipperLib.Clipper.prototype.DisposeAllPolyPts = function () {
for (var i = 0; i < this.m_PolyOuts.length; ++i)
this.DisposeOutRec(i);
ClipperLib.Clear(this.m_PolyOuts);
};
ClipperLib.Clipper.prototype.DisposeOutRec = function (index) {
var outRec = this.m_PolyOuts[index];
if (outRec.pts != null) this.DisposeOutPts(outRec.pts);
outRec = null;
this.m_PolyOuts[index] = null;
};
ClipperLib.Clipper.prototype.DisposeOutPts = function (pp) {
if (pp == null) return;
var tmpPp = null;
pp.prev.next = null;
while (pp != null) {
tmpPp = pp;
pp = pp.next;
tmpPp = null;
}
};
ClipperLib.Clipper.prototype.AddJoin = function (e1, e2, e1OutIdx, e2OutIdx) {
var jr = new ClipperLib.JoinRec();
if (e1OutIdx >= 0) jr.poly1Idx = e1OutIdx;
else jr.poly1Idx = e1.outIdx;
jr.pt1a = new ClipperLib.IntPoint(e1.xcurr, e1.ycurr);
jr.pt1b = new ClipperLib.IntPoint(e1.xtop, e1.ytop);
if (e2OutIdx >= 0) jr.poly2Idx = e2OutIdx;
else jr.poly2Idx = e2.outIdx;
jr.pt2a = new ClipperLib.IntPoint(e2.xcurr, e2.ycurr);
jr.pt2b = new ClipperLib.IntPoint(e2.xtop, e2.ytop);
this.m_Joins.push(jr);
};
ClipperLib.Clipper.prototype.AddHorzJoin = function (e, idx) {
var hj = new ClipperLib.HorzJoinRec();
hj.edge = e;
hj.savedIdx = idx;
this.m_HorizJoins.push(hj);
};
ClipperLib.Clipper.prototype.InsertLocalMinimaIntoAEL = function (botY) {
var pt, pt2;
while (this.m_CurrentLM != null && (this.m_CurrentLM.Y == botY)) {
var lb = this.m_CurrentLM.leftBound;
var rb = this.m_CurrentLM.rightBound;
this.InsertEdgeIntoAEL(lb);
this.InsertScanbeam(lb.ytop);
this.InsertEdgeIntoAEL(rb);
if (this.IsEvenOddFillType(lb)) {
lb.windDelta = 1;
rb.windDelta = 1;
}
else {
rb.windDelta = -lb.windDelta;
}
this.SetWindingCount(lb);
rb.windCnt = lb.windCnt;
rb.windCnt2 = lb.windCnt2;
if (rb.dx == ClipperLib.ClipperBase.horizontal) {
this.AddEdgeToSEL(rb);
this.InsertScanbeam(rb.nextInLML.ytop);
}
else this.InsertScanbeam(rb.ytop);
if (this.IsContributing(lb)) this.AddLocalMinPoly(lb, rb, new ClipperLib.IntPoint(lb.xcurr, this.m_CurrentLM.Y));
if (rb.outIdx >= 0) {
if (rb.dx == ClipperLib.ClipperBase.horizontal) {
for (var i = 0; i < this.m_HorizJoins.length; i++) {
pt = new ClipperLib.IntPoint(), pt2 = new ClipperLib.IntPoint();
var hj = this.m_HorizJoins[i];
if ((function () {
pt = {
Value: pt
};
pt2 = {
Value: pt2
};
var $res = this.GetOverlapSegment(new ClipperLib.IntPoint(hj.edge.xbot, hj.edge.ybot),
new ClipperLib.IntPoint(hj.edge.xtop, hj.edge.ytop),
new ClipperLib.IntPoint(rb.xbot, rb.ybot),
new ClipperLib.IntPoint(rb.xtop, rb.ytop),
pt, pt2);
pt = pt.Value;
pt2 = pt2.Value;
return $res;
})
.call(this)) this.AddJoin(hj.edge, rb, hj.savedIdx, -1);
}
}
}
if (lb.nextInAEL != rb) {
if (rb.outIdx >= 0 && rb.prevInAEL.outIdx >= 0 && this.SlopesEqual(rb.prevInAEL, rb, this.m_UseFullRange)) this.AddJoin(rb, rb.prevInAEL, -1, -1);
var e = lb.nextInAEL;
pt = new ClipperLib.IntPoint(lb.xcurr, lb.ycurr);
while (e != rb) {
if (e == null) ClipperLib.Error("InsertLocalMinimaIntoAEL: missing rightbound!");
this.IntersectEdges(rb, e, pt, ClipperLib.Protects.ipNone);
e = e.nextInAEL;
}
}
this.PopLocalMinima();
}
};
ClipperLib.Clipper.prototype.InsertEdgeIntoAEL = function (edge) {
edge.prevInAEL = null;
edge.nextInAEL = null;
if (this.m_ActiveEdges == null) {
this.m_ActiveEdges = edge;
}
else if (this.E2InsertsBeforeE1(this.m_ActiveEdges, edge)) {
edge.nextInAEL = this.m_ActiveEdges;
this.m_ActiveEdges.prevInAEL = edge;
this.m_ActiveEdges = edge;
}
else {
var e = this.m_ActiveEdges;
while (e.nextInAEL != null && !this.E2InsertsBeforeE1(e.nextInAEL, edge))
e = e.nextInAEL;
edge.nextInAEL = e.nextInAEL;
if (e.nextInAEL != null) e.nextInAEL.prevInAEL = edge;
edge.prevInAEL = e;
e.nextInAEL = edge;
}
};
ClipperLib.Clipper.prototype.E2InsertsBeforeE1 = function (e1, e2) {
return e2.xcurr == e1.xcurr ? e2.dx > e1.dx : e2.xcurr < e1.xcurr;
};
ClipperLib.Clipper.prototype.IsEvenOddFillType = function (edge) {
if (edge.polyType == ClipperLib.PolyType.ptSubject) return this.m_SubjFillType == ClipperLib.PolyFillType.pftEvenOdd;
else return this.m_ClipFillType == ClipperLib.PolyFillType.pftEvenOdd;
};
ClipperLib.Clipper.prototype.IsEvenOddAltFillType = function (edge) {
if (edge.polyType == ClipperLib.PolyType.ptSubject) return this.m_ClipFillType == ClipperLib.PolyFillType.pftEvenOdd;
else return this.m_SubjFillType == ClipperLib.PolyFillType.pftEvenOdd;
};
ClipperLib.Clipper.prototype.IsContributing = function (edge) {
var pft, pft2;
if (edge.polyType == ClipperLib.PolyType.ptSubject) {
pft = this.m_SubjFillType;
pft2 = this.m_ClipFillType;
}
else {
pft = this.m_ClipFillType;
pft2 = this.m_SubjFillType;
}
switch (pft) {
case ClipperLib.PolyFillType.pftEvenOdd:
case ClipperLib.PolyFillType.pftNonZero:
if (ClipperLib.Math_Abs_Int32(edge.windCnt) != 1) return false;
break;
case ClipperLib.PolyFillType.pftPositive:
if (edge.windCnt != 1) return false;
break;
default:
if (edge.windCnt != -1) return false;
break;
}
switch (this.m_ClipType) {
case ClipperLib.ClipType.ctIntersection:
switch (pft2) {
case ClipperLib.PolyFillType.pftEvenOdd:
case ClipperLib.PolyFillType.pftNonZero:
return (edge.windCnt2 != 0);
case ClipperLib.PolyFillType.pftPositive:
return (edge.windCnt2 > 0);
default:
return (edge.windCnt2 < 0);
}
break;
case ClipperLib.ClipType.ctUnion:
switch (pft2) {
case ClipperLib.PolyFillType.pftEvenOdd:
case ClipperLib.PolyFillType.pftNonZero:
return (edge.windCnt2 == 0);
case ClipperLib.PolyFillType.pftPositive:
return (edge.windCnt2 <= 0);
default:
return (edge.windCnt2 >= 0);
}
break;
case ClipperLib.ClipType.ctDifference:
if (edge.polyType == ClipperLib.PolyType.ptSubject) switch (pft2) {
case ClipperLib.PolyFillType.pftEvenOdd:
case ClipperLib.PolyFillType.pftNonZero:
return (edge.windCnt2 == 0);
case ClipperLib.PolyFillType.pftPositive:
return (edge.windCnt2 <= 0);
default:
return (edge.windCnt2 >= 0);
}
else switch (pft2) {
case ClipperLib.PolyFillType.pftEvenOdd:
case ClipperLib.PolyFillType.pftNonZero:
return (edge.windCnt2 != 0);
case ClipperLib.PolyFillType.pftPositive:
return (edge.windCnt2 > 0);
default:
return (edge.windCnt2 < 0);
}
}
return true;
};
ClipperLib.Clipper.prototype.SetWindingCount = function (edge) {
var e = edge.prevInAEL;
while (e != null && e.polyType != edge.polyType)
e = e.prevInAEL;
if (e == null) {
edge.windCnt = edge.windDelta;
edge.windCnt2 = 0;
e = this.m_ActiveEdges;
}
else if (this.IsEvenOddFillType(edge)) {
edge.windCnt = 1;
edge.windCnt2 = e.windCnt2;
e = e.nextInAEL;
}
else {
if (e.windCnt * e.windDelta < 0) {
if (ClipperLib.Math_Abs_Int32(e.windCnt) > 1) {
if (e.windDelta * edge.windDelta < 0) edge.windCnt = e.windCnt;
else edge.windCnt = e.windCnt + edge.windDelta;
}
else edge.windCnt = e.windCnt + e.windDelta + edge.windDelta;
}
else {
if (ClipperLib.Math_Abs_Int32(e.windCnt) > 1 && e.windDelta * edge.windDelta < 0) edge.windCnt = e.windCnt;
else if (e.windCnt + edge.windDelta == 0) edge.windCnt = e.windCnt;
else edge.windCnt = e.windCnt + edge.windDelta;
}
edge.windCnt2 = e.windCnt2;
e = e.nextInAEL;
}
if (this.IsEvenOddAltFillType(edge)) {
while (e != edge) {
edge.windCnt2 = (edge.windCnt2 == 0) ? 1 : 0;
e = e.nextInAEL;
}
}
else {
while (e != edge) {
edge.windCnt2 += e.windDelta;
e = e.nextInAEL;
}
}
};
ClipperLib.Clipper.prototype.AddEdgeToSEL = function (edge) {
if (this.m_SortedEdges == null) {
this.m_SortedEdges = edge;
edge.prevInSEL = null;
edge.nextInSEL = null;
}
else {
edge.nextInSEL = this.m_SortedEdges;
edge.prevInSEL = null;
this.m_SortedEdges.prevInSEL = edge;
this.m_SortedEdges = edge;
}
};
ClipperLib.Clipper.prototype.CopyAELToSEL = function () {
var e = this.m_ActiveEdges;
this.m_SortedEdges = e;
if (this.m_ActiveEdges == null) return;
this.m_SortedEdges.prevInSEL = null;
e = e.nextInAEL;
while (e != null) {
e.prevInSEL = e.prevInAEL;
e.prevInSEL.nextInSEL = e;
e.nextInSEL = null;
e = e.nextInAEL;
}
};
ClipperLib.Clipper.prototype.SwapPositionsInAEL = function (edge1, edge2) {
var next, prev;
if (edge1.nextInAEL == null && edge1.prevInAEL == null) return;
if (edge2.nextInAEL == null && edge2.prevInAEL == null) return;
if (edge1.nextInAEL == edge2) {
next = edge2.nextInAEL;
if (next != null) next.prevInAEL = edge1;
prev = edge1.prevInAEL;
if (prev != null) prev.nextInAEL = edge2;
edge2.prevInAEL = prev;
edge2.nextInAEL = edge1;
edge1.prevInAEL = edge2;
edge1.nextInAEL = next;
}
else if (edge2.nextInAEL == edge1) {
next = edge1.nextInAEL;
if (next != null) next.prevInAEL = edge2;
prev = edge2.prevInAEL;
if (prev != null) prev.nextInAEL = edge1;
edge1.prevInAEL = prev;
edge1.nextInAEL = edge2;
edge2.prevInAEL = edge1;
edge2.nextInAEL = next;
}
else {
next = edge1.nextInAEL;
prev = edge1.prevInAEL;
edge1.nextInAEL = edge2.nextInAEL;
if (edge1.nextInAEL != null) edge1.nextInAEL.prevInAEL = edge1;
edge1.prevInAEL = edge2.prevInAEL;
if (edge1.prevInAEL != null) edge1.prevInAEL.nextInAEL = edge1;
edge2.nextInAEL = next;
if (edge2.nextInAEL != null) edge2.nextInAEL.prevInAEL = edge2;
edge2.prevInAEL = prev;
if (edge2.prevInAEL != null) edge2.prevInAEL.nextInAEL = edge2;
}
if (edge1.prevInAEL == null) this.m_ActiveEdges = edge1;
else if (edge2.prevInAEL == null) this.m_ActiveEdges = edge2;
};
ClipperLib.Clipper.prototype.SwapPositionsInSEL = function (edge1, edge2) {
var next, prev;
if (edge1.nextInSEL == null && edge1.prevInSEL == null) return;
if (edge2.nextInSEL == null && edge2.prevInSEL == null) return;
if (edge1.nextInSEL == edge2) {
next = edge2.nextInSEL;
if (next != null) next.prevInSEL = edge1;
prev = edge1.prevInSEL;
if (prev != null) prev.nextInSEL = edge2;
edge2.prevInSEL = prev;
edge2.nextInSEL = edge1;
edge1.prevInSEL = edge2;
edge1.nextInSEL = next;
}
else if (edge2.nextInSEL == edge1) {
next = edge1.nextInSEL;
if (next != null) next.prevInSEL = edge2;
prev = edge2.prevInSEL;
if (prev != null) prev.nextInSEL = edge1;
edge1.prevInSEL = prev;
edge1.nextInSEL = edge2;
edge2.prevInSEL = edge1;
edge2.nextInSEL = next;
}
else {
next = edge1.nextInSEL;
prev = edge1.prevInSEL;
edge1.nextInSEL = edge2.nextInSEL;
if (edge1.nextInSEL != null) edge1.nextInSEL.prevInSEL = edge1;
edge1.prevInSEL = edge2.prevInSEL;
if (edge1.prevInSEL != null) edge1.prevInSEL.nextInSEL = edge1;
edge2.nextInSEL = next;
if (edge2.nextInSEL != null) edge2.nextInSEL.prevInSEL = edge2;
edge2.prevInSEL = prev;
if (edge2.prevInSEL != null) edge2.prevInSEL.nextInSEL = edge2;
}
if (edge1.prevInSEL == null) this.m_SortedEdges = edge1;
else if (edge2.prevInSEL == null) this.m_SortedEdges = edge2;
};
ClipperLib.Clipper.prototype.AddLocalMaxPoly = function (e1, e2, pt) {
this.AddOutPt(e1, pt);
if (e1.outIdx == e2.outIdx) {
e1.outIdx = -1;
e2.outIdx = -1;
}
else if (e1.outIdx < e2.outIdx) this.AppendPolygon(e1, e2);
else this.AppendPolygon(e2, e1);
};
ClipperLib.Clipper.prototype.AddLocalMinPoly = function (e1, e2, pt) {
var e, prevE;
if (e2.dx == ClipperLib.ClipperBase.horizontal || (e1.dx > e2.dx)) {
this.AddOutPt(e1, pt);
e2.outIdx = e1.outIdx;
e1.side = ClipperLib.EdgeSide.esLeft;
e2.side = ClipperLib.EdgeSide.esRight;
e = e1;
if (e.prevInAEL == e2) prevE = e2.prevInAEL;
else prevE = e.prevInAEL;
}
else {
this.AddOutPt(e2, pt);
e1.outIdx = e2.outIdx;
e1.side = ClipperLib.EdgeSide.esRight;
e2.side = ClipperLib.EdgeSide.esLeft;
e = e2;
if (e.prevInAEL == e1) prevE = e1.prevInAEL;
else prevE = e.prevInAEL;
}
if (prevE != null && prevE.outIdx >= 0 && (ClipperLib.Clipper.TopX(prevE, pt.Y) == ClipperLib.Clipper.TopX(e, pt.Y)) && this.SlopesEqual(e, prevE, this.m_UseFullRange)) this.AddJoin(e, prevE, -1, -1);
};
ClipperLib.Clipper.prototype.CreateOutRec = function () {
var result = new ClipperLib.OutRec();
result.idx = -1;
result.isHole = false;
result.FirstLeft = null;
result.AppendLink = null;
result.pts = null;
result.bottomPt = null;
return result;
};
ClipperLib.Clipper.prototype.AddOutPt = function (e, pt) {
var outRec, op;
var ToFront = (e.side == ClipperLib.EdgeSide.esLeft);
if (e.outIdx < 0) {
outRec = this.CreateOutRec();
this.m_PolyOuts.push(outRec);
outRec.idx = this.m_PolyOuts.length - 1;
e.outIdx = outRec.idx;
op = new ClipperLib.OutPt();
outRec.pts = op;
outRec.bottomPt = op;
op.pt = pt;
op.idx = outRec.idx;
op.next = op;
op.prev = op;
this.SetHoleState(e, outRec);
}
else {
outRec = this.m_PolyOuts[e.outIdx];
op = outRec.pts;
var op2;
if (ToFront && ClipperLib.ClipperBase.PointsEqual(pt, op.pt) || (!ToFront && ClipperLib.ClipperBase.PointsEqual(pt, op.prev.pt))) return;
op2 = new ClipperLib.OutPt();
op2.pt = pt;
op2.idx = outRec.idx;
if (op2.pt.Y == outRec.bottomPt.pt.Y && op2.pt.X < outRec.bottomPt.pt.X) outRec.bottomPt = op2;
op2.next = op;
op2.prev = op.prev;
op2.prev.next = op2;
op.prev = op2;
if (ToFront) outRec.pts = op2;
}
};
ClipperLib.Clipper.prototype.SwapPoints = function (pt1, pt2) {
var tmp = pt1.Value;
pt1.Value = pt2.Value;
pt2.Value = tmp;
};
ClipperLib.Clipper.prototype.GetOverlapSegment = function (pt1a, pt1b, pt2a, pt2b, pt1, pt2) {
if (ClipperLib.Math_Abs_Int64(pt1a.X - pt1b.X) > ClipperLib.Math_Abs_Int64(pt1a.Y - pt1b.Y)) {
if (pt1a.X > pt1b.X)
(function () {
pt1a = {
Value: pt1a
};
pt1b = {
Value: pt1b
};
var $res = this.SwapPoints(pt1a, pt1b);
pt1a = pt1a.Value;
pt1b = pt1b.Value;
return $res;
})
.call(this);
if (pt2a.X > pt2b.X)
(function () {
pt2a = {
Value: pt2a
};
pt2b = {
Value: pt2b
};
var $res = this.SwapPoints(pt2a, pt2b);
pt2a = pt2a.Value;
pt2b = pt2b.Value;
return $res;
})
.call(this);
if (pt1a.X > pt2a.X) pt1.Value = pt1a;
else pt1.Value = pt2a;
if (pt1b.X < pt2b.X) pt2.Value = pt1b;
else pt2.Value = pt2b;
return pt1.Value.X < pt2.Value.X;
}
else {
if (pt1a.Y < pt1b.Y)
(function () {
pt1a = {
Value: pt1a
};
pt1b = {
Value: pt1b
};
var $res = this.SwapPoints(pt1a, pt1b);
pt1a = pt1a.Value;
pt1b = pt1b.Value;
return $res;
})
.call(this);
if (pt2a.Y < pt2b.Y)
(function () {
pt2a = {
Value: pt2a
};
pt2b = {
Value: pt2b
};
var $res = this.SwapPoints(pt2a, pt2b);
pt2a = pt2a.Value;
pt2b = pt2b.Value;
return $res;
})
.call(this);
if (pt1a.Y < pt2a.Y) pt1.Value = pt1a;
else pt1.Value = pt2a;
if (pt1b.Y > pt2b.Y) pt2.Value = pt1b;
else pt2.Value = pt2b;
return pt1.Value.Y > pt2.Value.Y;
}
};
ClipperLib.Clipper.prototype.FindSegment = function (pp, UseFullInt64Range, pt1, pt2) {
if (pp.Value == null) return false;
var pp2 = pp.Value;
var pt1a = new ClipperLib.IntPoint(pt1.Value);
var pt2a = new ClipperLib.IntPoint(pt2.Value);
do {
// Timo's comment: for some reason calling SlopesEqual() below uses big integers
// So although coordinates are low (eg. 900), big integers are sometimes used.
// => Fixed according to changes in original Clipper ver 5.1.2 (25 February 2013)
if (this.SlopesEqual(pt1a, pt2a, pp.Value.pt, pp.Value.prev.pt, UseFullInt64Range) && this.SlopesEqual(pt1a, pt2a, pp.Value.pt, UseFullInt64Range) && this.GetOverlapSegment(pt1a, pt2a, pp.Value.pt, pp.Value.prev.pt, pt1, pt2)) return true;
pp.Value = pp.Value.next;
}
while (pp.Value != pp2);
return false;
};
ClipperLib.Clipper.prototype.Pt3IsBetweenPt1AndPt2 = function (pt1, pt2, pt3) {
if (ClipperLib.ClipperBase.PointsEqual(pt1, pt3) || ClipperLib.ClipperBase.PointsEqual(pt2, pt3)) return true;
else if (pt1.X != pt2.X) return (pt1.X < pt3.X) == (pt3.X < pt2.X);
else return (pt1.Y < pt3.Y) == (pt3.Y < pt2.Y);
};
ClipperLib.Clipper.prototype.InsertPolyPtBetween = function (p1, p2, pt) {
var result = new ClipperLib.OutPt();
result.pt = pt;
if (p2 == p1.next) {
p1.next = result;
p2.prev = result;
result.next = p2;
result.prev = p1;
}
else {
p2.next = result;
p1.prev = result;
result.next = p1;
result.prev = p2;
}
return result;
};
ClipperLib.Clipper.prototype.SetHoleState = function (e, outRec) {
var isHole = false;
var e2 = e.prevInAEL;
while (e2 != null) {
if (e2.outIdx >= 0) {
isHole = !isHole;
if (outRec.FirstLeft == null) outRec.FirstLeft = this.m_PolyOuts[e2.outIdx];
}
e2 = e2.prevInAEL;
}
if (isHole) outRec.isHole = true;
};
ClipperLib.Clipper.prototype.GetDx = function (pt1, pt2) {
if (pt1.Y == pt2.Y) return ClipperLib.ClipperBase.horizontal;
else return (pt2.X - pt1.X) / (pt2.Y - pt1.Y);
};
ClipperLib.Clipper.prototype.FirstIsBottomPt = function (btmPt1, btmPt2) {
var p = btmPt1.prev;
while (ClipperLib.ClipperBase.PointsEqual(p.pt, btmPt1.pt) && (p != btmPt1))
p = p.prev;
var dx1p = ClipperLib.Math_Abs_Double(this.GetDx(btmPt1.pt, p.pt));
p = btmPt1.next;
while (ClipperLib.ClipperBase.PointsEqual(p.pt, btmPt1.pt) && (p != btmPt1))
p = p.next;
var dx1n = ClipperLib.Math_Abs_Double(this.GetDx(btmPt1.pt, p.pt));
p = btmPt2.prev;
while (ClipperLib.ClipperBase.PointsEqual(p.pt, btmPt2.pt) && (p != btmPt2))
p = p.prev;
var dx2p = ClipperLib.Math_Abs_Double(this.GetDx(btmPt2.pt, p.pt));
p = btmPt2.next;
while (ClipperLib.ClipperBase.PointsEqual(p.pt, btmPt2.pt) && (p != btmPt2))
p = p.next;
var dx2n = ClipperLib.Math_Abs_Double(this.GetDx(btmPt2.pt, p.pt));
return (dx1p >= dx2p && dx1p >= dx2n) || (dx1n >= dx2p && dx1n >= dx2n);
};
ClipperLib.Clipper.prototype.GetBottomPt = function (pp) {
var dups = null;
var p = pp.next;
while (p != pp) {
if (p.pt.Y > pp.pt.Y) {
pp = p;
dups = null;
}
else if (p.pt.Y == pp.pt.Y && p.pt.X <= pp.pt.X) {
if (p.pt.X < pp.pt.X) {
dups = null;
pp = p;
}
else {
if (p.next != pp && p.prev != pp) dups = p;
}
}
p = p.next;
}
if (dups != null) {
while (dups != p) {
if (!this.FirstIsBottomPt(p, dups)) pp = dups;
dups = dups.next;
while (!ClipperLib.ClipperBase.PointsEqual(dups.pt, pp.pt))
dups = dups.next;
}
}
return pp;
};
ClipperLib.Clipper.prototype.GetLowermostRec = function (outRec1, outRec2) {
var bPt1 = outRec1.bottomPt;
var bPt2 = outRec2.bottomPt;
if (bPt1.pt.Y > bPt2.pt.Y) return outRec1;
else if (bPt1.pt.Y < bPt2.pt.Y) return outRec2;
else if (bPt1.pt.X < bPt2.pt.X) return outRec1;
else if (bPt1.pt.X > bPt2.pt.X) return outRec2;
else if (bPt1.next == bPt1) return outRec2;
else if (bPt2.next == bPt2) return outRec1;
else if (this.FirstIsBottomPt(bPt1, bPt2)) return outRec1;
else return outRec2;
};
ClipperLib.Clipper.prototype.Param1RightOfParam2 = function (outRec1, outRec2) {
do {
outRec1 = outRec1.FirstLeft;
if (outRec1 == outRec2) return true;
}
while (outRec1 != null);
return false;
};
ClipperLib.Clipper.prototype.AppendPolygon = function (e1, e2) {
var outRec1 = this.m_PolyOuts[e1.outIdx];
var outRec2 = this.m_PolyOuts[e2.outIdx];
var holeStateRec;
if (this.Param1RightOfParam2(outRec1, outRec2)) holeStateRec = outRec2;
else if (this.Param1RightOfParam2(outRec2, outRec1)) holeStateRec = outRec1;
else holeStateRec = this.GetLowermostRec(outRec1, outRec2);
var p1_lft = outRec1.pts;
var p1_rt = p1_lft.prev;
var p2_lft = outRec2.pts;
var p2_rt = p2_lft.prev;
var side;
var i;
if (e1.side == ClipperLib.EdgeSide.esLeft) {
if (e2.side == ClipperLib.EdgeSide.esLeft) {
this.ReversePolyPtLinks(p2_lft);
p2_lft.next = p1_lft;
p1_lft.prev = p2_lft;
p1_rt.next = p2_rt;
p2_rt.prev = p1_rt;
outRec1.pts = p2_rt;
}
else {
p2_rt.next = p1_lft;
p1_lft.prev = p2_rt;
p2_lft.prev = p1_rt;
p1_rt.next = p2_lft;
outRec1.pts = p2_lft;
}
side = ClipperLib.EdgeSide.esLeft;
}
else {
if (e2.side == ClipperLib.EdgeSide.esRight) {
this.ReversePolyPtLinks(p2_lft);
p1_rt.next = p2_rt;
p2_rt.prev = p1_rt;
p2_lft.next = p1_lft;
p1_lft.prev = p2_lft;
}
else {
p1_rt.next = p2_lft;
p2_lft.prev = p1_rt;
p1_lft.prev = p2_rt;
p2_rt.next = p1_lft;
}
side = ClipperLib.EdgeSide.esRight;
}
if (holeStateRec == outRec2) {
outRec1.bottomPt = outRec2.bottomPt;
outRec1.bottomPt.idx = outRec1.idx;
if (outRec2.FirstLeft != outRec1) outRec1.FirstLeft = outRec2.FirstLeft;
outRec1.isHole = outRec2.isHole;
}
outRec2.pts = null;
outRec2.bottomPt = null;
outRec2.AppendLink = outRec1;
var OKIdx = e1.outIdx;
var ObsoleteIdx = e2.outIdx;
e1.outIdx = -1;
e2.outIdx = -1;
var e = this.m_ActiveEdges;
while (e != null) {
if (e.outIdx == ObsoleteIdx) {
e.outIdx = OKIdx;
e.side = side;
break;
}
e = e.nextInAEL;
}
for (i = 0; i < this.m_Joins.length; ++i) {
if (this.m_Joins[i].poly1Idx == ObsoleteIdx) this.m_Joins[i].poly1Idx = OKIdx;
if (this.m_Joins[i].poly2Idx == ObsoleteIdx) this.m_Joins[i].poly2Idx = OKIdx;
}
for (i = 0; i < this.m_HorizJoins.length; ++i) {
if (this.m_HorizJoins[i].savedIdx == ObsoleteIdx) this.m_HorizJoins[i].savedIdx = OKIdx;
}
};
ClipperLib.Clipper.prototype.ReversePolyPtLinks = function (pp) {
if (pp == null) return;
var pp1;
var pp2;
pp1 = pp;
do {
pp2 = pp1.next;
pp1.next = pp1.prev;
pp1.prev = pp2;
pp1 = pp2;
}
while (pp1 != pp);
};
ClipperLib.Clipper.SwapSides = function (edge1, edge2) {
var side = edge1.side;
edge1.side = edge2.side;
edge2.side = side;
};
ClipperLib.Clipper.SwapPolyIndexes = function (edge1, edge2) {
var outIdx = edge1.outIdx;
edge1.outIdx = edge2.outIdx;
edge2.outIdx = outIdx;
};
ClipperLib.Clipper.prototype.DoEdge1 = function (edge1, edge2, pt) {
this.AddOutPt(edge1, pt);
ClipperLib.Clipper.SwapSides(edge1, edge2);
ClipperLib.Clipper.SwapPolyIndexes(edge1, edge2);
};
ClipperLib.Clipper.prototype.DoEdge2 = function (edge1, edge2, pt) {
this.AddOutPt(edge2, pt);
ClipperLib.Clipper.SwapSides(edge1, edge2);
ClipperLib.Clipper.SwapPolyIndexes(edge1, edge2);
};
ClipperLib.Clipper.prototype.DoBothEdges = function (edge1, edge2, pt) {
this.AddOutPt(edge1, pt);
this.AddOutPt(edge2, pt);
ClipperLib.Clipper.SwapSides(edge1, edge2);
ClipperLib.Clipper.SwapPolyIndexes(edge1, edge2);
};
ClipperLib.Clipper.prototype.IntersectEdges = function (e1, e2, pt, protects) {
var e1stops = (ClipperLib.Protects.ipLeft & protects) == 0 && e1.nextInLML == null && e1.xtop == pt.X && e1.ytop == pt.Y;
var e2stops = (ClipperLib.Protects.ipRight & protects) == 0 && e2.nextInLML == null && e2.xtop == pt.X && e2.ytop == pt.Y;
var e1Contributing = (e1.outIdx >= 0);
var e2contributing = (e2.outIdx >= 0);
if (e1.polyType == e2.polyType) {
if (this.IsEvenOddFillType(e1)) {
var oldE1WindCnt = e1.windCnt;
e1.windCnt = e2.windCnt;
e2.windCnt = oldE1WindCnt;
}
else {
if (e1.windCnt + e2.windDelta == 0) e1.windCnt = -e1.windCnt;
else e1.windCnt += e2.windDelta;
if (e2.windCnt - e1.windDelta == 0) e2.windCnt = -e2.windCnt;
else e2.windCnt -= e1.windDelta;
}
}
else {
if (!this.IsEvenOddFillType(e2)) e1.windCnt2 += e2.windDelta;
else e1.windCnt2 = (e1.windCnt2 == 0) ? 1 : 0;
if (!this.IsEvenOddFillType(e1)) e2.windCnt2 -= e1.windDelta;
else e2.windCnt2 = (e2.windCnt2 == 0) ? 1 : 0;
}
var e1FillType, e2FillType, e1FillType2, e2FillType2;
if (e1.polyType == ClipperLib.PolyType.ptSubject) {
e1FillType = this.m_SubjFillType;
e1FillType2 = this.m_ClipFillType;
}
else {
e1FillType = this.m_ClipFillType;
e1FillType2 = this.m_SubjFillType;
}
if (e2.polyType == ClipperLib.PolyType.ptSubject) {
e2FillType = this.m_SubjFillType;
e2FillType2 = this.m_ClipFillType;
}
else {
e2FillType = this.m_ClipFillType;
e2FillType2 = this.m_SubjFillType;
}
var e1Wc, e2Wc;
switch (e1FillType) {
case ClipperLib.PolyFillType.pftPositive:
e1Wc = e1.windCnt;
break;
case ClipperLib.PolyFillType.pftNegative:
e1Wc = -e1.windCnt;
break;
default:
e1Wc = ClipperLib.Math_Abs_Int32(e1.windCnt);
break;
}
switch (e2FillType) {
case ClipperLib.PolyFillType.pftPositive:
e2Wc = e2.windCnt;
break;
case ClipperLib.PolyFillType.pftNegative:
e2Wc = -e2.windCnt;
break;
default:
e2Wc = ClipperLib.Math_Abs_Int32(e2.windCnt);
break;
}
if (e1Contributing && e2contributing) {
if (e1stops || e2stops || (e1Wc != 0 && e1Wc != 1) || (e2Wc != 0 && e2Wc != 1) || (e1.polyType != e2.polyType && this.m_ClipType != ClipperLib.ClipType.ctXor)) this.AddLocalMaxPoly(e1, e2, pt);
else this.DoBothEdges(e1, e2, pt);
}
else if (e1Contributing) {
if ((e2Wc == 0 || e2Wc == 1) && (this.m_ClipType != ClipperLib.ClipType.ctIntersection || e2.polyType == ClipperLib.PolyType.ptSubject || (e2.windCnt2 != 0))) this.DoEdge1(e1, e2, pt);
}
else if (e2contributing) {
if ((e1Wc == 0 || e1Wc == 1) && (this.m_ClipType != ClipperLib.ClipType.ctIntersection || e1.polyType == ClipperLib.PolyType.ptSubject || (e1.windCnt2 != 0))) this.DoEdge2(e1, e2, pt);
}
else if ((e1Wc == 0 || e1Wc == 1) && (e2Wc == 0 || e2Wc == 1) && !e1stops && !e2stops) {
var e1Wc2, e2Wc2;
switch (e1FillType2) {
case ClipperLib.PolyFillType.pftPositive:
e1Wc2 = e1.windCnt2;
break;
case ClipperLib.PolyFillType.pftNegative:
e1Wc2 = -e1.windCnt2;
break;
default:
e1Wc2 = ClipperLib.Math_Abs_Int32(e1.windCnt2);
break;
}
switch (e2FillType2) {
case ClipperLib.PolyFillType.pftPositive:
e2Wc2 = e2.windCnt2;
break;
case ClipperLib.PolyFillType.pftNegative:
e2Wc2 = -e2.windCnt2;
break;
default:
e2Wc2 = ClipperLib.Math_Abs_Int32(e2.windCnt2);
break;
}
if (e1.polyType != e2.polyType) this.AddLocalMinPoly(e1, e2, pt);
else if (e1Wc == 1 && e2Wc == 1) switch (this.m_ClipType) {
case ClipperLib.ClipType.ctIntersection:
if (e1Wc2 > 0 && e2Wc2 > 0) this.AddLocalMinPoly(e1, e2, pt);
break;
case ClipperLib.ClipType.ctUnion:
if (e1Wc2 <= 0 && e2Wc2 <= 0) this.AddLocalMinPoly(e1, e2, pt);
break;
case ClipperLib.ClipType.ctDifference:
if (((e1.polyType == ClipperLib.PolyType.ptClip) && (e1Wc2 > 0) && (e2Wc2 > 0)) || ((e1.polyType == ClipperLib.PolyType.ptSubject) && (e1Wc2 <= 0) && (e2Wc2 <= 0))) this.AddLocalMinPoly(e1, e2, pt);
break;
case ClipperLib.ClipType.ctXor:
this.AddLocalMinPoly(e1, e2, pt);
break;
}
else ClipperLib.Clipper.SwapSides(e1, e2);
}
if ((e1stops != e2stops) && ((e1stops && (e1.outIdx >= 0)) || (e2stops && (e2.outIdx >= 0)))) {
ClipperLib.Clipper.SwapSides(e1, e2);
ClipperLib.Clipper.SwapPolyIndexes(e1, e2);
}
if (e1stops) this.DeleteFromAEL(e1);
if (e2stops) this.DeleteFromAEL(e2);
};
ClipperLib.Clipper.prototype.DeleteFromAEL = function (e) {
var AelPrev = e.prevInAEL;
var AelNext = e.nextInAEL;
if (AelPrev == null && AelNext == null && (e != this.m_ActiveEdges)) return;
if (AelPrev != null) AelPrev.nextInAEL = AelNext;
else this.m_ActiveEdges = AelNext;
if (AelNext != null) AelNext.prevInAEL = AelPrev;
e.nextInAEL = null;
e.prevInAEL = null;
};
ClipperLib.Clipper.prototype.DeleteFromSEL = function (e) {
var SelPrev = e.prevInSEL;
var SelNext = e.nextInSEL;
if (SelPrev == null && SelNext == null && (e != this.m_SortedEdges)) return;
if (SelPrev != null) SelPrev.nextInSEL = SelNext;
else this.m_SortedEdges = SelNext;
if (SelNext != null) SelNext.prevInSEL = SelPrev;
e.nextInSEL = null;
e.prevInSEL = null;
};
ClipperLib.Clipper.prototype.UpdateEdgeIntoAEL = function (e) {
if (e.Value.nextInLML == null) ClipperLib.Error("UpdateEdgeIntoAEL: invalid call");
var AelPrev = e.Value.prevInAEL;
var AelNext = e.Value.nextInAEL;
e.Value.nextInLML.outIdx = e.Value.outIdx;
if (AelPrev != null) AelPrev.nextInAEL = e.Value.nextInLML;
else this.m_ActiveEdges = e.Value.nextInLML;
if (AelNext != null) AelNext.prevInAEL = e.Value.nextInLML;
e.Value.nextInLML.side = e.Value.side;
e.Value.nextInLML.windDelta = e.Value.windDelta;
e.Value.nextInLML.windCnt = e.Value.windCnt;
e.Value.nextInLML.windCnt2 = e.Value.windCnt2;
e.Value = e.Value.nextInLML;
e.Value.prevInAEL = AelPrev;
e.Value.nextInAEL = AelNext;
if (e.Value.dx != ClipperLib.ClipperBase.horizontal) this.InsertScanbeam(e.Value.ytop);
};
ClipperLib.Clipper.prototype.ProcessHorizontals = function () {
var horzEdge = this.m_SortedEdges;
while (horzEdge != null) {
this.DeleteFromSEL(horzEdge);
this.ProcessHorizontal(horzEdge);
horzEdge = this.m_SortedEdges;
}
};
ClipperLib.Clipper.prototype.ProcessHorizontal = function (horzEdge) {
var Direction;
var horzLeft, horzRight;
if (horzEdge.xcurr < horzEdge.xtop) {
horzLeft = horzEdge.xcurr;
horzRight = horzEdge.xtop;
Direction = ClipperLib.Direction.dLeftToRight;
}
else {
horzLeft = horzEdge.xtop;
horzRight = horzEdge.xcurr;
Direction = ClipperLib.Direction.dRightToLeft;
}
var eMaxPair;
if (horzEdge.nextInLML != null) eMaxPair = null;
else eMaxPair = this.GetMaximaPair(horzEdge);
var e = this.GetNextInAEL(horzEdge, Direction);
while (e != null) {
var eNext = this.GetNextInAEL(e, Direction);
if (eMaxPair != null || ((Direction == ClipperLib.Direction.dLeftToRight) && (e.xcurr <= horzRight)) || ((Direction == ClipperLib.Direction.dRightToLeft) && (e.xcurr >= horzLeft))) {
if (e.xcurr == horzEdge.xtop && eMaxPair == null) {
if (this.SlopesEqual(e, horzEdge.nextInLML, this.m_UseFullRange)) {
if (horzEdge.outIdx >= 0 && e.outIdx >= 0) this.AddJoin(horzEdge.nextInLML, e, horzEdge.outIdx, -1);
break;
}
else if (e.dx < horzEdge.nextInLML.dx) break;
}
if (e == eMaxPair) {
if (Direction == ClipperLib.Direction.dLeftToRight) this.IntersectEdges(horzEdge, e, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), 0);
else this.IntersectEdges(e, horzEdge, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), 0);
if (eMaxPair.outIdx >= 0) ClipperLib.Error("ProcessHorizontal error");
return;
}
else if (e.dx == ClipperLib.ClipperBase.horizontal && !this.IsMinima(e) && !(e.xcurr > e.xtop)) {
if (Direction == ClipperLib.Direction.dLeftToRight) this.IntersectEdges(horzEdge, e, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), (this.IsTopHorz(horzEdge, e.xcurr)) ? ClipperLib.Protects.ipLeft : ClipperLib.Protects.ipBoth);
else this.IntersectEdges(e, horzEdge, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), (this.IsTopHorz(horzEdge, e.xcurr)) ? ClipperLib.Protects.ipRight : ClipperLib.Protects.ipBoth);
}
else if (Direction == ClipperLib.Direction.dLeftToRight) {
this.IntersectEdges(horzEdge, e, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), (this.IsTopHorz(horzEdge, e.xcurr)) ? ClipperLib.Protects.ipLeft : ClipperLib.Protects.ipBoth);
}
else {
this.IntersectEdges(e, horzEdge, new ClipperLib.IntPoint(e.xcurr, horzEdge.ycurr), (this.IsTopHorz(horzEdge, e.xcurr)) ? ClipperLib.Protects.ipRight : ClipperLib.Protects.ipBoth);
}
this.SwapPositionsInAEL(horzEdge, e);
}
else if ((Direction == ClipperLib.Direction.dLeftToRight && e.xcurr > horzRight && horzEdge.nextInSEL == null) || (Direction == ClipperLib.Direction.dRightToLeft && e.xcurr < horzLeft && horzEdge.nextInSEL == null)) break;
e = eNext;
}
if (horzEdge.nextInLML != null) {
if (horzEdge.outIdx >= 0) this.AddOutPt(horzEdge, new ClipperLib.IntPoint(horzEdge.xtop, horzEdge.ytop));
(function () {
horzEdge = {
Value: horzEdge
};
var $res = this.UpdateEdgeIntoAEL(horzEdge);
horzEdge = horzEdge.Value;
return $res;
})
.call(this);
}
else {
if (horzEdge.outIdx >= 0) this.IntersectEdges(horzEdge, eMaxPair, new ClipperLib.IntPoint(horzEdge.xtop, horzEdge.ycurr), ClipperLib.Protects.ipBoth);
this.DeleteFromAEL(eMaxPair);
this.DeleteFromAEL(horzEdge);
}
};
ClipperLib.Clipper.prototype.IsTopHorz = function (horzEdge, XPos) {
var e = this.m_SortedEdges;
while (e != null) {
if ((XPos >= Math.min(e.xcurr, e.xtop)) && (XPos <= Math.max(e.xcurr, e.xtop))) return false;
e = e.nextInSEL;
}
return true;
};
ClipperLib.Clipper.prototype.GetNextInAEL = function (e, Direction) {
return Direction == ClipperLib.Direction.dLeftToRight ? e.nextInAEL : e.prevInAEL;
};
ClipperLib.Clipper.prototype.IsMinima = function (e) {
return e != null && (e.prev.nextInLML != e) && (e.next.nextInLML != e);
};
ClipperLib.Clipper.prototype.IsMaxima = function (e, Y) {
return (e != null && e.ytop == Y && e.nextInLML == null);
};
ClipperLib.Clipper.prototype.IsIntermediate = function (e, Y) {
return (e.ytop == Y && e.nextInLML != null);
};
ClipperLib.Clipper.prototype.GetMaximaPair = function (e) {
if (!this.IsMaxima(e.next, e.ytop) || (e.next.xtop != e.xtop)) return e.prev;
else return e.next;
};
ClipperLib.Clipper.prototype.ProcessIntersections = function (botY, topY) {
if (this.m_ActiveEdges == null) return true;
try {
if (this.alternateAlgo)
this.BuildIntersectList2(botY, topY);
else
this.BuildIntersectList2(botY, topY);
if (this.m_IntersectNodes == null) return true;
if (this.FixupIntersections()) this.ProcessIntersectList();
else return false;
}
catch ($$e2) {
this.m_SortedEdges = null;
this.DisposeIntersectNodes();
ClipperLib.Error("ProcessIntersections error");
}
return true;
};
ClipperLib.Clipper.prototype.BuildIntersectList2 = function (botY, topY) {
if (this.m_ActiveEdges == null) return;
var e = this.m_ActiveEdges;
e.tmpX = ClipperLib.Clipper.TopX(e, topY);
this.m_SortedEdges = e;
this.m_SortedEdges.prevInSEL = null;
e = e.nextInAEL;
while (e != null) {
e.prevInSEL = e.prevInAEL;
e.prevInSEL.nextInSEL = e;
e.nextInSEL = null;
e.tmpX = ClipperLib.Clipper.TopX(e, topY);
e = e.nextInAEL;
}
var isModified = true;
var intersectArray = [];
while (isModified && this.m_SortedEdges != null) {
isModified = false;
e = this.m_SortedEdges;
while (e.nextInSEL != null) {
var eNext = e.nextInSEL;
var pt = new ClipperLib.IntPoint();
if (e.tmpX > eNext.tmpX && (function () {
pt = {
Value: pt
};
var $res = this.IntersectPoint(e, eNext, pt);
pt = pt.Value;
return $res;
})
.call(this)) {
if (pt.Y > botY) {
pt.Y = botY;
pt.X = ClipperLib.Clipper.TopX(e, pt.Y);
}
this.SwapPositionsInSEL(e, eNext);
intersectArray.push(this.CreateIntersectNode(e, eNext, pt));
isModified = true;
}
else e = eNext;
}
if (e.prevInSEL != null) e.prevInSEL.nextInSEL = null;
else break;
}
this.AddBulkIntersectNodes(intersectArray);
this.m_SortedEdges = null;
};
ClipperLib.Clipper.prototype.AddBulkIntersectNodes = function (intersectArray) {
intersectArray.sort(function (a, b) {
return ClipperLib.Clipper.prototype.ProcessParam1BeforeParam2(a, b) ? -1 : 1;
});
var startNode = this.m_IntersectNodes;
for (var i = 0; i < intersectArray.length; i++) {
var newNode = intersectArray[i];
if (this.m_IntersectNodes == null) this.m_IntersectNodes = newNode;
else if (this.ProcessParam1BeforeParam2(newNode, this.m_IntersectNodes)) {
newNode.next = this.m_IntersectNodes;
this.m_IntersectNodes = newNode;
}
else {
var iNode = startNode;
while (iNode.next != null && this.ProcessParam1BeforeParam2(iNode.next, newNode))
iNode = iNode.next;
newNode.next = iNode.next;
iNode.next = newNode;
}
startNode = newNode;
}
}
ClipperLib.Clipper.prototype.CreateIntersectNode = function (e1, e2, pt) {
var newNode = new ClipperLib.IntersectNode();
newNode.edge1 = e1;
newNode.edge2 = e2;
newNode.pt = pt;
newNode.next = null;
return newNode;
}
ClipperLib.Clipper.prototype.BuildIntersectList = function (botY, topY) {
if (this.m_ActiveEdges == null) return;
var e = this.m_ActiveEdges;
e.tmpX = ClipperLib.Clipper.TopX(e, topY);
this.m_SortedEdges = e;
this.m_SortedEdges.prevInSEL = null;
e = e.nextInAEL;
while (e != null) {
e.prevInSEL = e.prevInAEL;
e.prevInSEL.nextInSEL = e;
e.nextInSEL = null;
e.tmpX = ClipperLib.Clipper.TopX(e, topY);
e = e.nextInAEL;
}
var isModified = true;
while (isModified && this.m_SortedEdges != null) {
isModified = false;
e = this.m_SortedEdges;
while (e.nextInSEL != null) {
var eNext = e.nextInSEL;
var pt = new ClipperLib.IntPoint();
if (e.tmpX > eNext.tmpX && (function () {
pt = {
Value: pt
};
var $res = this.IntersectPoint(e, eNext, pt);
pt = pt.Value;
return $res;
})
.call(this)) {
if (pt.Y > botY) {
pt.Y = botY;
pt.X = ClipperLib.Clipper.TopX(e, pt.Y);
}
this.AddIntersectNode(e, eNext, pt);
this.SwapPositionsInSEL(e, eNext);
isModified = true;
}
else e = eNext;
}
if (e.prevInSEL != null) e.prevInSEL.nextInSEL = null;
else break;
}
this.m_SortedEdges = null;
};
ClipperLib.Clipper.prototype.FixupIntersections = function () {
if (this.m_IntersectNodes.next == null) return true;
this.CopyAELToSEL();
var int1 = this.m_IntersectNodes;
var int2 = this.m_IntersectNodes.next;
while (int2 != null) {
var e1 = int1.edge1;
var e2;
if (e1.prevInSEL == int1.edge2) e2 = e1.prevInSEL;
else if (e1.nextInSEL == int1.edge2) e2 = e1.nextInSEL;
else {
while (int2 != null) {
if (int2.edge1.nextInSEL == int2.edge2 || int2.edge1.prevInSEL == int2.edge2) break;
else int2 = int2.next;
}
if (int2 == null) return false;
this.SwapIntersectNodes(int1, int2);
e1 = int1.edge1;
e2 = int1.edge2;
}
this.SwapPositionsInSEL(e1, e2);
int1 = int1.next;
int2 = int1.next;
}
this.m_SortedEdges = null;
return (int1.edge1.prevInSEL == int1.edge2 || int1.edge1.nextInSEL == int1.edge2);
};
ClipperLib.Clipper.prototype.ProcessIntersectList = function () {
while (this.m_IntersectNodes != null) {
var iNode = this.m_IntersectNodes.next;
this.IntersectEdges(this.m_IntersectNodes.edge1, this.m_IntersectNodes.edge2, this.m_IntersectNodes.pt, ClipperLib.Protects.ipBoth);
this.SwapPositionsInAEL(this.m_IntersectNodes.edge1, this.m_IntersectNodes.edge2);
this.m_IntersectNodes = null;
this.m_IntersectNodes = iNode;
}
};
/*
--------------------------------
Round speedtest: http://jsperf.com/fastest-round
--------------------------------
*/
var R1 = function (a) {
return a < 0 ? Math.ceil(a - 0.5) : Math.round(a)
};
var R2 = function (a) {
return a < 0 ? Math.ceil(a - 0.5) : Math.floor(a + 0.5)
};
var R3 = function (a) {
return a < 0 ? -Math.round(Math.abs(a)) : Math.round(a)
};
var R4 = function (a) {
if (a < 0) {
a -= 0.5;
return a < -2147483648 ? Math.ceil(a) : a | 0;
} else {
a += 0.5;
return a > 2147483647 ? Math.floor(a) : a | 0;
}
};
if (browser.msie) ClipperLib.Clipper.Round = R1;
else if (browser.chromium) ClipperLib.Clipper.Round = R3;
else if (browser.safari) ClipperLib.Clipper.Round = R4;
else ClipperLib.Clipper.Round = R2; // eg. browser.chrome || browser.firefox || browser.opera
ClipperLib.Clipper.TopX = function (edge, currentY) {
if (currentY == edge.ytop) return edge.xtop;
return edge.xbot + ClipperLib.Clipper.Round(edge.dx * (currentY - edge.ybot));
};
ClipperLib.Clipper.prototype.AddIntersectNode = function (e1, e2, pt) {
var newNode = new ClipperLib.IntersectNode();
newNode.edge1 = e1;
newNode.edge2 = e2;
newNode.pt = pt;
newNode.next = null;
if (this.m_IntersectNodes == null) this.m_IntersectNodes = newNode;
else if (this.ProcessParam1BeforeParam2(newNode, this.m_IntersectNodes)) {
newNode.next = this.m_IntersectNodes;
this.m_IntersectNodes = newNode;
}
else {
var iNode = this.m_IntersectNodes;
while (iNode.next != null && this.ProcessParam1BeforeParam2(iNode.next, newNode))
iNode = iNode.next;
newNode.next = iNode.next;
iNode.next = newNode;
}
};
ClipperLib.Clipper.prototype.ProcessParam1BeforeParam2 = function (node1, node2) {
var result;
if (node1.pt.Y == node2.pt.Y) {
if (node1.edge1 == node2.edge1 || node1.edge2 == node2.edge1) {
result = node2.pt.X > node1.pt.X;
return node2.edge1.dx > 0 ? !result : result;
}
else if (node1.edge1 == node2.edge2 || node1.edge2 == node2.edge2) {
result = node2.pt.X > node1.pt.X;
return node2.edge2.dx > 0 ? !result : result;
}
else return node2.pt.X > node1.pt.X;
}
else return node1.pt.Y > node2.pt.Y;
};
ClipperLib.Clipper.prototype.SwapIntersectNodes = function (int1, int2) {
var e1 = int1.edge1;
var e2 = int1.edge2;
var p = int1.pt;
int1.edge1 = int2.edge1;
int1.edge2 = int2.edge2;
int1.pt = int2.pt;
int2.edge1 = e1;
int2.edge2 = e2;
int2.pt = p;
};
ClipperLib.Clipper.prototype.IntersectPoint = function (edge1, edge2, ip) {
var b1, b2;
if (this.SlopesEqual(edge1, edge2, this.m_UseFullRange)) return false;
else if (edge1.dx == 0) {
ip.Value.X = edge1.xbot;
if (edge2.dx == ClipperLib.ClipperBase.horizontal) {
ip.Value.Y = edge2.ybot;
}
else {
b2 = edge2.ybot - (edge2.xbot / edge2.dx);
ip.Value.Y = ClipperLib.Clipper.Round(ip.Value.X / edge2.dx + b2);
}
}
else if (edge2.dx == 0) {
ip.Value.X = edge2.xbot;
if (edge1.dx == ClipperLib.ClipperBase.horizontal) {
ip.Value.Y = edge1.ybot;
}
else {
b1 = edge1.ybot - (edge1.xbot / edge1.dx);
ip.Value.Y = ClipperLib.Clipper.Round(ip.Value.X / edge1.dx + b1);
}
}
else {
b1 = edge1.xbot - edge1.ybot * edge1.dx;
b2 = edge2.xbot - edge2.ybot * edge2.dx;
var q = (b2 - b1) / (edge1.dx - edge2.dx);
ip.Value.Y = ClipperLib.Clipper.Round(q);
if (ClipperLib.Math_Abs_Double(edge1.dx) < ClipperLib.Math_Abs_Double(edge2.dx)) ip.Value.X = ClipperLib.Clipper.Round(edge1.dx * q + b1);
else ip.Value.X = ClipperLib.Clipper.Round(edge2.dx * q + b2);
}
if (ip.Value.Y < edge1.ytop || ip.Value.Y < edge2.ytop) {
if (edge1.ytop > edge2.ytop) {
ip.Value.X = edge1.xtop;
ip.Value.Y = edge1.ytop;
return ClipperLib.Clipper.TopX(edge2, edge1.ytop) < edge1.xtop;
}
else {
ip.Value.X = edge2.xtop;
ip.Value.Y = edge2.ytop;
return ClipperLib.Clipper.TopX(edge1, edge2.ytop) > edge2.xtop;
}
}
else return true;
};
ClipperLib.Clipper.prototype.DisposeIntersectNodes = function () {
while (this.m_IntersectNodes != null) {
var iNode = this.m_IntersectNodes.next;
this.m_IntersectNodes = null;
this.m_IntersectNodes = iNode;
}
};
ClipperLib.Clipper.prototype.ProcessEdgesAtTopOfScanbeam = function (topY) {
var e = this.m_ActiveEdges;
var ePrev;
while (e != null) {
if (this.IsMaxima(e, topY) && this.GetMaximaPair(e)
.dx != ClipperLib.ClipperBase.horizontal) {
ePrev = e.prevInAEL;
this.DoMaxima(e, topY);
if (ePrev == null) e = this.m_ActiveEdges;
else e = ePrev.nextInAEL;
}
else {
if (this.IsIntermediate(e, topY) && e.nextInLML.dx == ClipperLib.ClipperBase.horizontal) {
if (e.outIdx >= 0) {
this.AddOutPt(e, new ClipperLib.IntPoint(e.xtop, e.ytop));
for (var i = 0; i < this.m_HorizJoins.length; ++i) {
var pt = new ClipperLib.IntPoint(),
pt2 = new ClipperLib.IntPoint();
var hj = this.m_HorizJoins[i];
if ((function () {
pt = {
Value: pt
};
pt2 = {
Value: pt2
};
var $res = this.GetOverlapSegment(new ClipperLib.IntPoint(hj.edge.xbot, hj.edge.ybot),
new ClipperLib.IntPoint(hj.edge.xtop, hj.edge.ytop),
new ClipperLib.IntPoint(e.nextInLML.xbot, e.nextInLML.ybot),
new ClipperLib.IntPoint(e.nextInLML.xtop, e.nextInLML.ytop), pt, pt2);
pt = pt.Value;
pt2 = pt2.Value;
return $res;
})
.call(this)) this.AddJoin(hj.edge, e.nextInLML, hj.savedIdx, e.outIdx);
}
this.AddHorzJoin(e.nextInLML, e.outIdx);
}
(function () {
e = {
Value: e
};
var $res = this.UpdateEdgeIntoAEL(e);
e = e.Value;
return $res;
})
.call(this);
this.AddEdgeToSEL(e);
}
else {
e.xcurr = ClipperLib.Clipper.TopX(e, topY);
e.ycurr = topY;
}
e = e.nextInAEL;
}
}
this.ProcessHorizontals();
e = this.m_ActiveEdges;
while (e != null) {
if (this.IsIntermediate(e, topY)) {
if (e.outIdx >= 0) this.AddOutPt(e, new ClipperLib.IntPoint(e.xtop, e.ytop));
(function () {
e = {
Value: e
};
var $res = this.UpdateEdgeIntoAEL(e);
e = e.Value;
return $res;
})
.call(this);
ePrev = e.prevInAEL;
var eNext = e.nextInAEL;
if (ePrev != null && ePrev.xcurr == e.xbot && ePrev.ycurr == e.ybot && e.outIdx >= 0 && ePrev.outIdx >= 0 && ePrev.ycurr > ePrev.ytop && this.SlopesEqual(e, ePrev, this.m_UseFullRange)) {
this.AddOutPt(ePrev, new ClipperLib.IntPoint(e.xbot, e.ybot));
this.AddJoin(e, ePrev, -1, -1);
}
else if (eNext != null && eNext.xcurr == e.xbot && eNext.ycurr == e.ybot && e.outIdx >= 0 && eNext.outIdx >= 0 && eNext.ycurr > eNext.ytop && this.SlopesEqual(e, eNext, this.m_UseFullRange)) {
this.AddOutPt(eNext, new ClipperLib.IntPoint(e.xbot, e.ybot));
this.AddJoin(e, eNext, -1, -1);
}
}
e = e.nextInAEL;
}
};
ClipperLib.Clipper.prototype.DoMaxima = function (e, topY) {
var eMaxPair = this.GetMaximaPair(e);
var X = e.xtop;
var eNext = e.nextInAEL;
while (eNext != eMaxPair) {
if (eNext == null) ClipperLib.Error("DoMaxima error");
this.IntersectEdges(e, eNext, new ClipperLib.IntPoint(X, topY), ClipperLib.Protects.ipBoth);
eNext = eNext.nextInAEL;
}
if (e.outIdx < 0 && eMaxPair.outIdx < 0) {
this.DeleteFromAEL(e);
this.DeleteFromAEL(eMaxPair);
}
else if (e.outIdx >= 0 && eMaxPair.outIdx >= 0) {
this.IntersectEdges(e, eMaxPair, new ClipperLib.IntPoint(X, topY), ClipperLib.Protects.ipNone);
}
else ClipperLib.Error("DoMaxima error");
};
ClipperLib.Clipper.ReversePolygons = function (polys) {
var len = polys.length,
poly;
for (var i = 0; i < len; i++) {
if (polys[i] instanceof Array) polys[i].reverse();
}
};
ClipperLib.Clipper.Orientation = function (poly) {
return this.Area(poly) >= 0;
};
ClipperLib.Clipper.prototype.PointCount = function (pts) {
if (pts == null) return 0;
var result = 0;
var p = pts;
do {
result++;
p = p.next;
}
while (p != pts);
return result;
};
ClipperLib.Clipper.prototype.BuildResult = function (polyg) {
ClipperLib.Clear(polyg);
var outRec, len = this.m_PolyOuts.length;
for (var i = 0; i < len; i++) {
outRec = this.m_PolyOuts[i];
if (outRec.pts == null) continue;
var p = outRec.pts;
var cnt = this.PointCount(p);
if (cnt < 3) continue;
var pg = new ClipperLib.Polygon(cnt);
for (var j = 0; j < cnt; j++) {
//pg.push(p.pt);
pg.push(new ClipperLib.IntPoint(p.pt.X, p.pt.Y)); // Have to create new point, because the point can be a reference to other point
p = p.prev;
}
polyg.push(pg);
}
};
ClipperLib.Clipper.prototype.BuildResultEx = function (polyg) {
ClipperLib.Clear(polyg);
var i = 0;
while (i < this.m_PolyOuts.length) {
var outRec = this.m_PolyOuts[i++];
if (outRec.pts == null) break;
var p = outRec.pts;
var cnt = this.PointCount(p);
if (cnt < 3) continue;
var epg = new ClipperLib.ExPolygon();
epg.outer = new ClipperLib.Polygon();
epg.holes = new ClipperLib.Polygons();
for (var j = 0; j < cnt; j++) {
//epg.outer.push(p.pt);
epg.outer.push(new ClipperLib.IntPoint(p.pt.X, p.pt.Y)); // Have to create new point, because the point can be a reference to other point
p = p.prev;
}
while (i < this.m_PolyOuts.length) {
outRec = this.m_PolyOuts[i];
if (outRec.pts == null || !outRec.isHole) break;
var pg = new ClipperLib.Polygon();
p = outRec.pts;
do {
//pg.push(p.pt);
pg.push(new ClipperLib.IntPoint(p.pt.X, p.pt.Y)); // Have to create new point, because the point can be a reference to other point
p = p.prev;
}
while (p != outRec.pts);
epg.holes.push(pg);
i++;
}
polyg.push(epg);
}
};
ClipperLib.Clipper.prototype.FixupOutPolygon = function (outRec) {
var lastOK = null;
outRec.pts = outRec.bottomPt;
var pp = outRec.bottomPt;
for (; ;) {
if (pp.prev == pp || pp.prev == pp.next) {
this.DisposeOutPts(pp);
outRec.pts = null;
outRec.bottomPt = null;
return;
}
if (ClipperLib.ClipperBase.PointsEqual(pp.pt, pp.next.pt) || this.SlopesEqual(pp.prev.pt, pp.pt, pp.next.pt, this.m_UseFullRange)) {
lastOK = null;
var tmp = pp;
if (pp == outRec.bottomPt) outRec.bottomPt = null;
pp.prev.next = pp.next;
pp.next.prev = pp.prev;
pp = pp.prev;
tmp = null;
}
else if (pp == lastOK) break;
else {
if (lastOK == null) lastOK = pp;
pp = pp.next;
}
}
if (outRec.bottomPt == null) {
outRec.bottomPt = this.GetBottomPt(pp);
outRec.bottomPt.idx = outRec.idx;
outRec.pts = outRec.bottomPt;
}
};
ClipperLib.Clipper.prototype.JoinPoints = function (j, p1, p2) {
p1.Value = null;
p2.Value = null;
var outRec1 = this.m_PolyOuts[j.poly1Idx];
var outRec2 = this.m_PolyOuts[j.poly2Idx];
if (outRec1 == null || outRec2 == null) return false;
var pp1a = outRec1.pts;
var pp2a = outRec2.pts;
var pt1 = j.pt2a,
pt2 = j.pt2b;
var pt3 = j.pt1a,
pt4 = j.pt1b;
if (!(function () {
pp1a = {
Value: pp1a
};
pt1 = {
Value: pt1
};
pt2 = {
Value: pt2
};
var $res = this.FindSegment(pp1a, this.m_UseFullRange, pt1, pt2);
pp1a = pp1a.Value;
pt1 = pt1.Value;
pt2 = pt2.Value;
return $res;
})
.call(this)) return false;
if (outRec1 == outRec2) {
pp2a = pp1a.next;
if (!(function () {
pp2a = {
Value: pp2a
};
pt3 = {
Value: pt3
};
pt4 = {
Value: pt4
};
var $res = this.FindSegment(pp2a, this.m_UseFullRange, pt3, pt4);
pp2a = pp2a.Value;
pt3 = pt3.Value;
pt4 = pt4.Value;
return $res;
})
.call(this) || (pp2a == pp1a)) return false;
}
else if (!(function () {
pp2a = {
Value: pp2a
};
pt3 = {
Value: pt3
};
pt4 = {
Value: pt4
};
var $res = this.FindSegment(pp2a, this.m_UseFullRange, pt3, pt4);
pp2a = pp2a.Value;
pt3 = pt3.Value;
pt4 = pt4.Value;
return $res;
})
.call(this)) return false;
if (!(function () {
pt1 = {
Value: pt1
};
pt2 = {
Value: pt2
};
var $res = this.GetOverlapSegment(pt1.Value, pt2.Value, pt3, pt4, pt1, pt2);
pt1 = pt1.Value;
pt2 = pt2.Value;
return $res;
})
.call(this)) {
return false;
}
var p3, p4, prev = pp1a.prev;
if (ClipperLib.ClipperBase.PointsEqual(pp1a.pt, pt1)) p1.Value = pp1a;
else if (ClipperLib.ClipperBase.PointsEqual(prev.pt, pt1)) p1.Value = prev;
else p1.Value = this.InsertPolyPtBetween(pp1a, prev, pt1);
if (ClipperLib.ClipperBase.PointsEqual(pp1a.pt, pt2)) p2.Value = pp1a;
else if (ClipperLib.ClipperBase.PointsEqual(prev.pt, pt2)) p2.Value = prev;
else if ((p1.Value == pp1a) || (p1.Value == prev)) p2.Value = this.InsertPolyPtBetween(pp1a, prev, pt2);
else if (this.Pt3IsBetweenPt1AndPt2(pp1a.pt, p1.Value.pt, pt2)) p2.Value = this.InsertPolyPtBetween(pp1a, p1.Value, pt2);
else p2.Value = this.InsertPolyPtBetween(p1.Value, prev, pt2);
prev = pp2a.prev;
if (ClipperLib.ClipperBase.PointsEqual(pp2a.pt, pt1)) p3 = pp2a;
else if (ClipperLib.ClipperBase.PointsEqual(prev.pt, pt1)) p3 = prev;
else p3 = this.InsertPolyPtBetween(pp2a, prev, pt1);
if (ClipperLib.ClipperBase.PointsEqual(pp2a.pt, pt2)) p4 = pp2a;
else if (ClipperLib.ClipperBase.PointsEqual(prev.pt, pt2)) p4 = prev;
else if ((p3 == pp2a) || (p3 == prev)) p4 = this.InsertPolyPtBetween(pp2a, prev, pt2);
else if (this.Pt3IsBetweenPt1AndPt2(pp2a.pt, p3.pt, pt2)) p4 = this.InsertPolyPtBetween(pp2a, p3, pt2);
else p4 = this.InsertPolyPtBetween(p3, prev, pt2);
if (p1.Value.next == p2.Value && p3.prev == p4) {
p1.Value.next = p3;
p3.prev = p1.Value;
p2.Value.prev = p4;
p4.next = p2.Value;
return true;
}
else if (p1.Value.prev == p2.Value && p3.next == p4) {
p1.Value.prev = p3;
p3.next = p1.Value;
p2.Value.next = p4;
p4.prev = p2.Value;
return true;
}
else return false;
};
ClipperLib.Clipper.prototype.FixupJoinRecs = function (j, pt, startIdx) {
for (var k = startIdx; k < this.m_Joins.length; k++) {
var j2 = this.m_Joins[k];
if (j2.poly1Idx == j.poly1Idx && this.PointIsVertex(j2.pt1a, pt)) j2.poly1Idx = j.poly2Idx;
if (j2.poly2Idx == j.poly1Idx && this.PointIsVertex(j2.pt2a, pt)) j2.poly2Idx = j.poly2Idx;
}
};
ClipperLib.Clipper.prototype.JoinCommonEdges = function () {
var k, orec;
for (var i = 0; i < this.m_Joins.length; i++) {
var j = this.m_Joins[i];
var p1, p2;
if (!(function () {
p1 = {
Value: p1
};
p2 = {
Value: p2
};
var $res = this.JoinPoints(j, p1, p2);
p1 = p1.Value;
p2 = p2.Value;
return $res;
})
.call(this)) continue;
var outRec1 = this.m_PolyOuts[j.poly1Idx];
var outRec2 = this.m_PolyOuts[j.poly2Idx];
if (outRec1 == outRec2) {
outRec1.pts = this.GetBottomPt(p1);
outRec1.bottomPt = outRec1.pts;
outRec1.bottomPt.idx = outRec1.idx;
outRec2 = this.CreateOutRec();
this.m_PolyOuts.push(outRec2);
outRec2.idx = this.m_PolyOuts.length - 1;
j.poly2Idx = outRec2.idx;
outRec2.pts = this.GetBottomPt(p2);
outRec2.bottomPt = outRec2.pts;
outRec2.bottomPt.idx = outRec2.idx;
if (this.PointInPolygon(outRec2.pts.pt, outRec1.pts, this.m_UseFullRange)) {
outRec2.isHole = !outRec1.isHole;
outRec2.FirstLeft = outRec1;
this.FixupJoinRecs(j, p2, i + 1);
this.FixupOutPolygon(outRec1);
this.FixupOutPolygon(outRec2);
if ((outRec2.isHole ^ this.m_ReverseOutput) == (this.Area(outRec2, this.m_UseFullRange) > 0))
this.ReversePolyPtLinks(outRec2.pts);
}
else if (this.PointInPolygon(outRec1.pts.pt, outRec2.pts, this.m_UseFullRange)) {
outRec2.isHole = outRec1.isHole;
outRec1.isHole = !outRec2.isHole;
outRec2.FirstLeft = outRec1.FirstLeft;
outRec1.FirstLeft = outRec2;
this.FixupJoinRecs(j, p2, i + 1);
this.FixupOutPolygon(outRec1);
this.FixupOutPolygon(outRec2);
if ((outRec1.isHole ^ this.m_ReverseOutput) == (this.Area(outRec1, this.m_UseFullRange) > 0))
this.ReversePolyPtLinks(outRec1.pts);
if (this.m_UsingExPolygons && outRec1.isHole) for (k = 0; k < this.m_PolyOuts.length; ++k) {
orec = this.m_PolyOuts[k];
if (orec.isHole && orec.bottomPt != null && orec.FirstLeft == outRec1) orec.FirstLeft = outRec2;
}
}
else {
outRec2.isHole = outRec1.isHole;
outRec2.FirstLeft = outRec1.FirstLeft;
this.FixupJoinRecs(j, p2, i + 1);
this.FixupOutPolygon(outRec1);
this.FixupOutPolygon(outRec2);
if (this.m_UsingExPolygons && outRec2.pts != null) for (k = 0; k < this.m_PolyOuts.length; ++k) {
orec = this.m_PolyOuts[k];
if (orec.isHole && orec.bottomPt != null && orec.FirstLeft == outRec1 && this.PointInPolygon(orec.bottomPt.pt, outRec2.pts, this.m_UseFullRange)) orec.FirstLeft = outRec2;
}
}
}
else {
if (this.m_UsingExPolygons) for (k = 0; k < this.m_PolyOuts.length; ++k)
if (this.m_PolyOuts[k].isHole && this.m_PolyOuts[k].bottomPt != null && this.m_PolyOuts[k].FirstLeft == outRec2) this.m_PolyOuts[k].FirstLeft = outRec1;
this.FixupOutPolygon(outRec1);
if (outRec1.pts != null) {
outRec1.isHole = this.Area(outRec1, this.m_UseFullRange) < 0;
if (outRec1.isHole && outRec1.FirstLeft == null) outRec1.FirstLeft = outRec2.FirstLeft;
}
var OKIdx = outRec1.idx;
var ObsoleteIdx = outRec2.idx;
outRec2.pts = null;
outRec2.bottomPt = null;
outRec2.AppendLink = outRec1;
for (k = i + 1; k < this.m_Joins.length; k++) {
var j2 = this.m_Joins[k];
if (j2.poly1Idx == ObsoleteIdx) j2.poly1Idx = OKIdx;
if (j2.poly2Idx == ObsoleteIdx) j2.poly2Idx = OKIdx;
}
}
}
};
ClipperLib.Clipper.FullRangeNeeded = function (pts) {
var result = false;
for (var i = 0; i < pts.length; i++) {
if (ClipperLib.Math_Abs_Int64(pts[i].X) > ClipperLib.ClipperBase.hiRange || ClipperLib.Math_Abs_Int64(pts[i].Y) > ClipperLib.ClipperBase.hiRange)
ClipperLib.Error("Coordinate exceeds range bounds in FullRangeNeeded().");
else if (ClipperLib.Math_Abs_Int64(pts[i].X) > ClipperLib.ClipperBase.loRange || ClipperLib.Math_Abs_Int64(pts[i].Y) > ClipperLib.ClipperBase.loRange) {
result = true;
}
}
return result;
};
ClipperLib.Clipper.prototype.Area = ClipperLib.Clipper.Area = function () {
var arg = arguments;
var i, a;
if (arg.length == 1) // function ( poly )
{
var poly = arg[0];
var highI = poly.length - 1;
if (highI < 2) return 0;
if (ClipperLib.Clipper.FullRangeNeeded(poly)) {
a = new Int128(poly[highI].X + poly[0].X).multiply(new Int128(poly[0].Y - poly[highI].Y));
for (i = 1; i <= highI; ++i)
a = a.add(new Int128(poly[i - 1].X + poly[i].X).multiply(new Int128(poly[i].Y - poly[i - 1].Y)));
return parseFloat(a.toString()) / 2;
}
else {
var area = (poly[highI].X + poly[0].X) * (poly[0].Y - poly[highI].Y);
for (i = 1; i <= highI; ++i)
area += (poly[i - 1].X + poly[i].X) * (poly[i].Y - poly[i - 1].Y);
return area / 2;
}
}
else if (arg.length == 2) // function (outRec, UseFull64BitRange)
{
var outRec = arg[0];
var UseFull64BitRange = arg[1];
var op = outRec.pts;
if (op == null) return 0;
if (UseFull64BitRange) {
a = new Int128(Int128.ZERO);
do {
a = a.add(new Int128(op.pt.X + op.prev.pt.X).multiply(new Int128(op.prev.pt.Y - op.pt.Y)));
op = op.next;
} while (op != outRec.pts);
return parseFloat(a.toString()) / 2; // This could be something faster!
}
else {
a = 0;
do {
a = a + (op.pt.X + op.prev.pt.X) * (op.prev.pt.Y - op.pt.Y);
op = op.next;
}
while (op != outRec.pts);
return a / 2;
}
}
};
ClipperLib.Clipper.BuildArc = function (pt, a1, a2, r) {
var steps = Math.sqrt(ClipperLib.Math_Abs_Double(r)) * ClipperLib.Math_Abs_Double(a2 - a1);
steps = steps / 4; // to avoid overload
// If you want to make steps independent of scaling factor (scale have to be set),
// the following line does the trick:
// steps = steps / Math.sqrt(scale) * 2;
// If you want that changing scale factor has some influence to steps, uncomment also the following line:
// It may be desirable, that bigger precision ( = bigger scaling factor) needs more steps.
// steps += Math.pow(scale, 0.2);
if (steps < 6) steps = 6;
if (steps > ClipperLib.MaxSteps) steps = ClipperLib.MaxSteps; // to avoid overload
// if (steps > 1048576) steps = 1048576; // 0x100000
// if (steps > ClipperLib.MaxSteps) steps = ClipperLib.MaxSteps; // 0x100000
// Had to change 1048576 to lower value, because when coordinates are near or above lorange, program starts hanging
// Adjust this value to meet your needs, maybe 10 is enough for most purposes
var n = ClipperLib.Cast_Int32(steps);
var result = new ClipperLib.Polygon();
var da = (a2 - a1) / (n - 1);
var a = a1;
for (var i = 0; i < n; ++i) {
result.push(new ClipperLib.IntPoint(pt.X + ClipperLib.Clipper.Round(Math.cos(a) * r), pt.Y + ClipperLib.Clipper.Round(Math.sin(a) * r)));
a += da;
}
return result;
};
ClipperLib.Clipper.GetUnitNormal = function (pt1, pt2) {
var dx = (pt2.X - pt1.X);
var dy = (pt2.Y - pt1.Y);
if ((dx == 0) && (dy == 0)) return new ClipperLib.Clipper.DoublePoint(0, 0);
var f = 1 / Math.sqrt(dx * dx + dy * dy);
dx *= f;
dy *= f;
return new ClipperLib.Clipper.DoublePoint(dy, -dx);
};
ClipperLib.Clipper.prototype.OffsetPolygons = function (poly, delta, jointype, MiterLimit, AutoFix) {
var a = arguments;
if (a.length == 4) AutoFix = true;
else if (a.length == 3) {
MiterLimit = 2;
AutoFix = true;
}
else if (a.length == 2) {
jointype = ClipperLib.JoinType.jtSquare;
MiterLimit = 2;
AutoFix = true;
}
if (isNaN(delta)) ClipperLib.Error("Delta is not a number");
else if (isNaN(MiterLimit)) ClipperLib.Error("MiterLimit is not a number");
var result = {};
new ClipperLib.Clipper.PolyOffsetBuilder(poly, result, delta, jointype, MiterLimit, AutoFix);
if (result.Value) result = result.Value;
else result = [
[]
];
return result;
};
ClipperLib.Clipper.prototype.SimplifyPolygon = function (poly, fillType) {
var result = new ClipperLib.Polygons();
var c = new ClipperLib.Clipper();
if (c.AddPolygon(poly, ClipperLib.PolyType.ptSubject))
c.Execute(ClipperLib.ClipType.ctUnion, result, fillType, fillType);
return result;
};
ClipperLib.Clipper.prototype.SimplifyPolygons = function (polys, fillType) {
var result = new ClipperLib.Polygons();
var c = new ClipperLib.Clipper();
if (c.AddPolygons(polys, ClipperLib.PolyType.ptSubject))
c.Execute(ClipperLib.ClipType.ctUnion, result, fillType, fillType);
return result;
};
var ce = ClipperLib.Clipper;
var ce2 = ClipperLib.ClipperBase;
var p;
if (typeof (Object.getOwnPropertyNames) == 'undefined') {
for (p in ce2.prototype)
if (typeof (ce.prototype[p]) == 'undefined' || ce.prototype[p] == Object.prototype[p]) ce.prototype[p] = ce2.prototype[p];
for (p in ce2)
if (typeof (ce[p]) == 'undefined') ce[p] = ce2[p];
ce.$baseCtor = ce2;
}
else {
var props = Object.getOwnPropertyNames(ce2.prototype);
for (var i = 0; i < props.length; i++)
if (typeof (Object.getOwnPropertyDescriptor(ce.prototype, props[i])) == 'undefined') Object.defineProperty(ce.prototype, props[i], Object.getOwnPropertyDescriptor(ce2.prototype, props[i]));
for (p in ce2)
if (typeof (ce[p]) == 'undefined') ce[p] = ce2[p];
ce.$baseCtor = ce2;
}
ClipperLib.Clipper.DoublePoint = function (x, y) {
this.X = x;
this.Y = y;
};
ClipperLib.Clipper.PolyOffsetBuilder = function (pts, solution, delta, jointype, MiterLimit, AutoFix) {
this.pts = null; // Polygons
this.currentPoly = null; // Polygon
this.normals = null;
this.delta = 0;
this.m_R = 0;
this.m_i = 0;
this.m_j = 0;
this.m_k = 0;
this.botPt = null; // This is "this." because it is ref in original c# code
if (delta == 0) {
solution.Value = pts;
return;
}
this.pts = pts;
this.delta = delta;
var i, j;
//AutoFix - fixes polygon orientation if necessary and removes
//duplicate vertices. Can be set false when you're sure that polygon
//orientation is correct and that there are no duplicate vertices.
if (AutoFix) {
var Len = this.pts.length,
botI = 0;
while (botI < Len && this.pts[botI].length == 0) botI++;
if (botI == Len) {
//solution.Value = new ClipperLib.Polygons();
return;
}
//botPt: used to find the lowermost (in inverted Y-axis) & leftmost point
//This point (on pts[botI]) must be on an outer polygon ring and if
//its orientation is false (counterclockwise) then assume all polygons
//need reversing ...
this.botPt = this.pts[botI][0]; // This is ported with different logic than other C# refs
// adding botPt to object's property it's accessible through object's
// methods
// => All other ref's are now ported using rather complex object.Value
// technique, which seems to work.
for (i = botI; i < Len; ++i) {
if (this.UpdateBotPt(this.pts[i][0])) botI = i;
for (j = this.pts[i].length - 1; j > 0; j--) {
if (ClipperLib.ClipperBase.PointsEqual(this.pts[i][j], this.pts[i][j - 1])) {
this.pts[i].splice(j, 1);
}
else if (this.UpdateBotPt(this.pts[i][j])) botI = i;
}
}
if (!ClipperLib.Clipper.Orientation(this.pts[botI])) ClipperLib.Clipper.ReversePolygons(this.pts);
}
if (MiterLimit <= 1) MiterLimit = 1;
var RMin = 2 / (MiterLimit * MiterLimit);
this.normals = [];
var deltaSq = delta * delta;
solution.Value = new ClipperLib.Polygons();
//ClipperLib.Clear(solution.Value);
var len;
for (this.m_i = 0; this.m_i < this.pts.length; this.m_i++) {
len = this.pts[this.m_i].length;
if (len > 1 && this.pts[this.m_i][0].X == this.pts[this.m_i][len - 1].X &&
this.pts[this.m_i][0].Y == this.pts[this.m_i][len - 1].Y) {
len--;
}
if (len == 0 || (len < 3 && delta <= 0)) {
continue;
}
else if (len == 1) {
var arc;
arc = ClipperLib.Clipper.BuildArc(this.pts[this.m_i][len - 1], 0, ClipperLib.PI2, delta);
solution.Value.push(arc);
continue;
}
//build normals ...
ClipperLib.Clear(this.normals);
for (j = 0; j < len - 1; ++j)
this.normals.push(ClipperLib.Clipper.GetUnitNormal(this.pts[this.m_i][j], this.pts[this.m_i][j + 1]));
this.normals.push(ClipperLib.Clipper.GetUnitNormal(this.pts[this.m_i][len - 1], this.pts[this.m_i][0]));
this.currentPoly = new ClipperLib.Polygon();
this.m_k = len - 1;
for (this.m_j = 0; this.m_j < len; ++this.m_j) {
switch (jointype) {
case ClipperLib.JoinType.jtMiter:
this.m_R = 1 + (this.normals[this.m_j].X * this.normals[this.m_k].X + this.normals[this.m_j].Y * this.normals[this.m_k].Y);
if (this.m_R >= RMin) this.DoMiter();
else this.DoSquare(MiterLimit);
break;
case ClipperLib.JoinType.jtRound:
this.DoRound();
break;
case ClipperLib.JoinType.jtSquare:
this.DoSquare(1);
break;
}
this.m_k = this.m_j;
}
solution.Value.push(this.currentPoly);
}
//finally, clean up untidy corners ...
var clpr = new ClipperLib.Clipper();
clpr.AddPolygons(solution.Value, ClipperLib.PolyType.ptSubject);
if (delta > 0) {
clpr.Execute(ClipperLib.ClipType.ctUnion, solution.Value, ClipperLib.PolyFillType.pftPositive, ClipperLib.PolyFillType.pftPositive);
}
else {
var r = clpr.GetBounds();
var outer = new ClipperLib.Polygon();
outer.push(new ClipperLib.IntPoint(r.left - 10, r.bottom + 10));
outer.push(new ClipperLib.IntPoint(r.right + 10, r.bottom + 10));
outer.push(new ClipperLib.IntPoint(r.right + 10, r.top - 10));
outer.push(new ClipperLib.IntPoint(r.left - 10, r.top - 10));
clpr.AddPolygon(outer, ClipperLib.PolyType.ptSubject);
clpr.Execute(ClipperLib.ClipType.ctUnion, solution.Value, ClipperLib.PolyFillType.pftNegative, ClipperLib.PolyFillType.pftNegative);
if (solution.Value.length > 0) {
solution.Value.splice(0, 1);
for (i = 0; i < solution.Value.length; i++)
solution.Value[i].reverse();
}
}
};
//ClipperLib.Clipper.PolyOffsetBuilder.buffLength = 128;
ClipperLib.Clipper.PolyOffsetBuilder.prototype.UpdateBotPt = function (pt) {
if (pt.Y > this.botPt.Y || (pt.Y == this.botPt.Y && pt.X < this.botPt.X)) {
this.botPt = pt;
return true;
}
else return false;
};
ClipperLib.Clipper.PolyOffsetBuilder.prototype.AddPoint = function (pt) {
this.currentPoly.push(pt);
};
ClipperLib.Clipper.PolyOffsetBuilder.prototype.DoSquare = function (mul) {
var pt1 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_k].X * this.delta)),
ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_k].Y * this.delta)));
var pt2 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_j].X * this.delta)),
ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_j].Y * this.delta)));
if ((this.normals[this.m_k].X * this.normals[this.m_j].Y - this.normals[this.m_j].X * this.normals[this.m_k].Y) * this.delta >= 0) {
var a1 = Math.atan2(this.normals[this.m_k].Y, this.normals[this.m_k].X);
var a2 = Math.atan2(-this.normals[this.m_j].Y, -this.normals[this.m_j].X);
a1 = Math.abs(a2 - a1);
if (a1 > ClipperLib.PI) a1 = ClipperLib.PI2 - a1;
var dx = Math.tan((ClipperLib.PI - a1) / 4) * Math.abs(this.delta * mul);
pt1 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64((pt1.X - this.normals[this.m_k].Y * dx)),
ClipperLib.Cast_Int64((pt1.Y + this.normals[this.m_k].X * dx)));
this.AddPoint(pt1);
pt2 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64((pt2.X + this.normals[this.m_j].Y * dx)),
ClipperLib.Cast_Int64((pt2.Y - this.normals[this.m_j].X * dx)));
this.AddPoint(pt2);
}
else {
this.AddPoint(pt1);
this.AddPoint(this.pts[this.m_i][this.m_j]);
this.AddPoint(pt2);
}
};
ClipperLib.Clipper.PolyOffsetBuilder.prototype.DoMiter = function () {
if ((this.normals[this.m_k].X * this.normals[this.m_j].Y - this.normals[this.m_j].X * this.normals[this.m_k].Y) * this.delta >= 0) {
var q = this.delta / this.m_R;
this.AddPoint(new ClipperLib.IntPoint(
ClipperLib.Cast_Int64(
ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + (this.normals[this.m_k].X + this.normals[this.m_j].X) * q)),
ClipperLib.Cast_Int64(
ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + (this.normals[this.m_k].Y + this.normals[this.m_j].Y) * q))));
}
else {
var pt1 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_k].X * this.delta)),
ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_k].Y * this.delta)));
var pt2 = new ClipperLib.IntPoint(ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_j].X * this.delta)),
ClipperLib.Cast_Int64(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_j].Y * this.delta)));
this.AddPoint(pt1);
this.AddPoint(this.pts[this.m_i][this.m_j]);
this.AddPoint(pt2);
}
};
ClipperLib.Clipper.PolyOffsetBuilder.prototype.DoRound = function () {
var pt1 = new ClipperLib.IntPoint(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_k].X * this.delta),
ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_k].Y * this.delta));
var pt2 = new ClipperLib.IntPoint(ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].X + this.normals[this.m_j].X * this.delta),
ClipperLib.Clipper.Round(this.pts[this.m_i][this.m_j].Y + this.normals[this.m_j].Y * this.delta));
this.AddPoint(pt1);
//round off reflex angles (ie > 180 deg) unless almost flat (ie < 10deg).
//cross product normals < 0 . angle > 180 deg.
//dot product normals == 1 . no angle
if ((this.normals[this.m_k].X * this.normals[this.m_j].Y - this.normals[this.m_j].X * this.normals[this.m_k].Y) * this.delta >= 0) {
if ((this.normals[this.m_j].X * this.normals[this.m_k].X + this.normals[this.m_j].Y * this.normals[this.m_k].Y) < 0.985) {
var a1 = Math.atan2(this.normals[this.m_k].Y, this.normals[this.m_k].X);
var a2 = Math.atan2(this.normals[this.m_j].Y, this.normals[this.m_j].X);
if (this.delta > 0 && a2 < a1) a2 += ClipperLib.PI2;
else if (this.delta < 0 && a2 > a1) a2 -= ClipperLib.PI2;
var arc = ClipperLib.Clipper.BuildArc(this.pts[this.m_i][this.m_j], a1, a2, this.delta);
for (var m = 0; m < arc.length; m++)
this.AddPoint(arc[m]);
}
}
else this.AddPoint(this.pts[this.m_i][this.m_j]);
this.AddPoint(pt2);
};
ClipperLib.Error = function (message) {
try {
throw new Error(message);
}
catch (err) {
alert(err.message);
}
};
// Make deep copy of Polygons or Polygon
// so that also IntPoint objects are cloned and not only referenced
// This should be the fastest way
ClipperLib.Clone = function (polygon) {
if (!(polygon instanceof Array)) return [];
if (polygon.length == 0) return [];
else if (polygon.length == 1 && polygon[0].length == 0) return [
[]
];
var isPolygons = polygon[0] instanceof Array;
if (!isPolygons) polygon = [polygon];
var len = polygon.length, plen, i, j, result;
var results = [];
for (i = 0; i < len; i++) {
plen = polygon[i].length;
result = [];
for (j = 0; j < plen; j++) {
result.push({X: polygon[i][j].X, Y: polygon[i][j].Y});
}
results.push(result);
}
if (!isPolygons) results = results[0];
return results;
};
// Clean() joins vertices that are too near each other
// and causes distortion to offsetted polygons without cleaning
ClipperLib.Clean = function (polygon, delta) {
if (!(polygon instanceof Array)) return [];
var isPolygons = polygon[0] instanceof Array;
var polygon = ClipperLib.Clone(polygon);
if (typeof delta != "number" || delta === null) {
ClipperLib.Error("Delta is not a number in Clean().");
return polygon;
}
if (polygon.length == 0 || (polygon.length == 1 && polygon[0].length == 0) || delta < 0) return polygon;
if (!isPolygons) polygon = [polygon];
var k_length = polygon.length;
var len, poly, result, d, p, j, i;
var results = [];
for (var k = 0; k < k_length; k++) {
poly = polygon[k];
len = poly.length;
if (len == 0) continue;
else if (len < 3) {
result = poly;
results.push(result);
continue;
}
result = poly;
d = delta * delta;
//d = Math.floor(c_delta * c_delta);
p = poly[0];
j = 1;
for (i = 1; i < len; i++) {
if ((poly[i].X - p.X) * (poly[i].X - p.X) +
(poly[i].Y - p.Y) * (poly[i].Y - p.Y) <= d)
continue;
result[j] = poly[i];
p = poly[i];
j++;
}
p = poly[j - 1];
if ((poly[0].X - p.X) * (poly[0].X - p.X) +
(poly[0].Y - p.Y) * (poly[0].Y - p.Y) <= d)
j--;
if (j < len)
result.splice(j, len - j);
if (result.length) results.push(result);
}
if (!isPolygons && results.length) results = results[0];
else if (!isPolygons && results.length == 0) results = [];
else if (isPolygons && results.length == 0) results = [
[]
];
return results;
}
// Removes points that doesn't affect much to the visual appearance.
// If middle point is at or under certain distance (tolerance) of the line between
// start and end point, the middle point is removed.
ClipperLib.Lighten = function (polygon, tolerance) {
if (!(polygon instanceof Array)) return [];
if (typeof tolerance != "number" || tolerance === null) {
ClipperLib.Error("Tolerance is not a number in Lighten().")
return ClipperLib.Clone(polygon);
}
if (polygon.length === 0 || (polygon.length == 1 && polygon[0].length === 0) || tolerance < 0) {
return ClipperLib.Clone(polygon);
}
if (!(polygon[0] instanceof Array)) polygon = [polygon];
var i, j, poly, k, poly2, plen, A, B, P, d, rem, addlast;
var bxax, byay, nL;
var len = polygon.length;
var results = [];
for (i = 0; i < len; i++) {
poly = polygon[i];
for (k = 0; k < 1000000; k++) // could be forever loop, but wiser to restrict max repeat count
{
poly2 = [];
plen = poly.length;
// the first have to added to the end, if first and last are not the same
// this way we ensure that also the actual last point can be removed if needed
if (poly[plen - 1].X != poly[0].X || poly[plen - 1].Y != poly[0].Y) {
addlast = 1;
poly.push({X: poly[0].X, Y: poly[0].Y});
plen = poly.length;
}
else addlast = 0;
rem = []; // Indexes of removed points
for (j = 0; j < plen - 2; j++) {
A = poly[j]; // Start point of line segment
P = poly[j + 1]; // Middle point. This is the one to be removed.
B = poly[j + 2]; // End point of line segment
bxax = B.X - A.X;
byay = B.Y - A.Y;
d = 0;
if (bxax !== 0 || byay !== 0) // To avoid Nan, when A==P && P==B. And to avoid peaks (A==B && A!=P), which have lenght, but not area.
{
nL = Math.sqrt(bxax * bxax + byay * byay);
// d is the perpendicular distance from P to (infinite) line AB.
d = Math.abs((P.X - A.X) * byay - (P.Y - A.Y) * bxax) / nL;
}
if (d <= tolerance) {
rem[j + 1] = 1;
j++; // when removed, transfer the pointer to the next one
}
}
// add all unremoved points to poly2
poly2.push({X: poly[0].X, Y: poly[0].Y});
for (j = 1; j < plen - 1; j++)
if (!rem[j]) poly2.push({X: poly[j].X, Y: poly[j].Y});
poly2.push({X: poly[plen - 1].X, Y: poly[plen - 1].Y});
// if the first point was added to the end, remove it
if (addlast) poly.pop();
// break, if there was not anymore removed points
if (!rem.length) break;
// else continue looping using poly2, to check if there are points to remove
else poly = poly2;
}
plen = poly2.length;
// remove duplicate from end, if needed
if (poly2[plen - 1].X == poly2[0].X && poly2[plen - 1].Y == poly2[0].Y) {
poly2.pop();
}
if (poly2.length > 2) // to avoid two-point-polygons
results.push(poly2);
}
if (!polygon[0] instanceof Array) results = results[0];
if (typeof (results) == "undefined") results = [
[]
];
return results;
}
if (typeof(document) !== "undefined") window.ClipperLib = ClipperLib;
else self['ClipperLib'] = ClipperLib;
})();
<!DOCTYPE html>
<html>
<head>
<script src="clipper_unminified.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.0/raphael-min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/1.8.3/jquery.min.js"></script>
<script>
/*
* jQuery repeatedclick v1.0.5
* Dual licensed under the MIT (http://www.opensource.org/licenses/mit-license.php)
* and GPL (http://www.opensource.org/licenses/gpl-license.php) licenses.
* Written by: Alexandr Zykov <[email protected]>
*/
if (typeof(jQuery) != "undefined")
jQuery.fn.hold = function (h, j) {
var c = jQuery.extend({duration: 350, speed: 0.85, min: 50}, j);
"undefined" === typeof jQuery.repeatedEvents && (jQuery.repeatedEvents = []);
jQuery.repeatedEvents.push(h);
var k = jQuery.repeatedEvents.length - 1, d, e;
return this.each(function () {
d = function (f, b, a) {
var g = this;
jQuery.repeatedEvents[f].call(g, a);
e = setTimeout(function () {
d.call(g, f, b > c.min ? b * c.speed : b, a)
}, b)
};
jQuery(this).mousedown(function (a) {
d.call(this, k, c.duration, a)
});
var a = function () {
"undefined" !== typeof e && clearInterval(e)
};
jQuery(this).mouseout(a);
jQuery(this).mouseup(a)
})
};
</script>
<script>
/*
* TotalStorage v. 1.1.2
* Copyright (c) 2012 Jared Novack & Upstatement (upstatement.com)
* Dual licensed under the MIT and GPL licenses:
* http://www.opensource.org/licenses/mit-license.php
* http://www.gnu.org/licenses/gpl.html
* @author Jared Novack/[email protected]
* @url http://upstatement.com/blog/2012/01/jquery-local-storage-done-right-and-easy/
*/
if (typeof(jQuery) != "undefined")
(function (c) {
var e = window.localStorage, f;
f = "undefined" == typeof e || "undefined" == typeof window.JSON ? !1 : !0;
c.totalStorage = function (b, a) {
return c.totalStorage.impl.init(b, a)
};
c.totalStorage.setItem = function (b, a) {
return c.totalStorage.impl.setItem(b, a)
};
c.totalStorage.getItem = function (b) {
return c.totalStorage.impl.getItem(b)
};
c.totalStorage.getAll = function () {
return c.totalStorage.impl.getAll()
};
c.totalStorage.deleteItem = function (b) {
return c.totalStorage.impl.deleteItem(b)
};
c.totalStorage.impl = {init: function (b, a) {
return"undefined" != typeof a ? this.setItem(b, a) : this.getItem(b)
}, setItem: function (b, a) {
if (!f)try {
return c.cookie(b, a), a
} catch (d) {
console.log("Local Storage not supported by this browser. Install the cookie plugin on your site to take advantage of the same functionality. You can get it at https://github.com/carhartl/jquery-cookie")
}
var g = JSON.stringify(a);
e.setItem(b, g);
return this.parseResult(g)
}, getItem: function (b) {
if (!f)try {
return this.parseResult(c.cookie(b))
} catch (a) {
return null
}
return this.parseResult(e.getItem(b))
},
deleteItem: function (b) {
if (!f)try {
return c.cookie(b, null), !0
} catch (a) {
return!1
}
e.removeItem(b);
return!0
}, getAll: function () {
var b = [];
if (f)for (d in e)d.length && b.push({key: d, value: this.parseResult(e.getItem(d))}); else try {
for (var a = document.cookie.split(";"), d = 0; d < a.length; d++) {
var g = a[d].split("=")[0];
b.push({key: g, value: this.parseResult(c.cookie(g))})
}
} catch (h) {
return null
}
return b
}, parseResult: function (b) {
var a;
try {
a = JSON.parse(b), "true" == a && (a = !0), "false" == a && (a = !1), parseFloat(a) == a && "object" != typeof a &&
(a = parseFloat(a))
} catch (c) {
}
return a
}}
})(jQuery);
</script>
<meta charset=utf-8/>
<title>Javascript Clipper Main Demo</title>
<style>
.textarea_hide_buttons {
margin-bottom: 6px;
margin-top: 4px;
}
body {
background-color: #F8F8F8;
}
p {
margin: 0px;
padding: 0px;
}
th {
background-color: #DDDDDD;
font-weight: bold;
padding: 0px;
}
body, th, td, input, legend, fieldset, p, b, button, select, textarea {
font-size: 13px;
font-family: Arial, Helvetica, sans-serif;
}
#p, #_p {
background-color: #ffffff;
border: 2px solid #AAAAAA;
}
.svg_mypath {
stroke: black;
stroke-width: 1;
fill: black;
fill-opacity: 0.3;
}
#p1, #_p1 {
fill: #00009C;
fill-opacity: 0.06;
stroke: #D3D3DA;
stroke-opacity: 1.0;
stroke-width: 0.8;
}
#p2, #_p2 {
fill: #9C0000;
fill-opacity: 0.06;
stroke: #FFA07A;
stroke-opacity: 1.0;
stroke-width: 0.8;
}
#p3, #_p3 {
fill: #80ff9C;
fill-opacity: 0.5;
stroke: #003300;
stroke-opacity: 1.0;
stroke-width: 0.8;
}
#svg_source_textarea {
width: 1215px;
height: 300px;
}
#benchmark_exports_textarea {
width: 1215px;
height: 300px;
}
div#svgcontainer {
margin-right: 2px;
margin-top: 6px;
}
div#benchmark_div_outer {
width: 100%;
}
div#benchmark_div {
width: 100%;
}
#benchmark_fieldset {
width: 367px;
min-width: 367px;
height: 350px;
}
.buttons {
display: inline-block;
padding: 0px;
padding-left: 12px;
padding-right: 12px;
background-color: #666666;
color: white;
border-radius: 10px;
text-align: center;
line-height: 20px;
font-size: 13px;
cursor: pointer;
margin-bottom: 2px;
margin-top: 2px;
font-family: Helvetica, Arial, sans-serif;
white-space: nowrap;
}
LEGEND {
font-weight: bold;
}
FIELDSET {
background-color: #EEEEEE;
/*
border-radius: 4px 4px 7px 7px;
-moz-border-radius: 4px 4px 7px 7px;
-webkit-border-radius: 4px 4px 7px 7px;
*/
border: 2px solid #AAAAAA;
white-space: nowrap;
}
table {
border: 0px;
}
.subpolylinks {
cursor: pointer;
font-size: 13px;
padding: 2px;
}
.subpolylinks:hover {
background-color: #BBBBBB;
}
.subpolylinks_disabled {
font-size: 13px;
padding: 2px;
}
/* To prevent odd text resizing in mobile Safari */
@media only screen and (max-device-width: 480px) {
.body, p, fieldset, td, th {
-webkit-text-size-adjust: 100%
}
}
.polygon_explorer {
text-align: left;
border-collapse: collapse;
white-space: normal;
width: 461px;
}
.polygon_explorer th {
font-weight: bold;
}
.polygon_explorer > tr > th, .polygon_explorer > tr > td,
.polygon_explorer > tbody > tr > th, .polygon_explorer > tbody > tr > td {
border: 1px solid #666666;
padding: 3px 4px 3px 4px;
vertical-align: top;
}
.polygons_fieldset {
border-collapse: collapse;
width: 100%;
}
.polygons_fieldset td {
vertical-align: top;
white-space: nowrap;
padding: 0px;
padding-right: 10px;
}
.polygons_fieldset input[type=radio] {
margin-left: 0px;
}
.random {
border-collapse: collapse;
width: 100%;
}
.random td {
padding: 0px;
padding-right: 10px;
}
.random_inputs {
width: 25px
}
.random th {
padding: 0px;
padding-right:
}
.random th {
background-color: transparent;
}
.random_new_th {
background-color: transparent;
}
.random_new_th button {
margin: 0px;
}
.random button, .random input {
margin-bottom: 0px;
margin-top: 0px;
}
#subj_polygon_count {
margin-bottom: 2px;
}
.filltype_cliptype {
border-collapse: collapse;
}
.filltype_cliptype td {
vertical-align: top;
padding: 0px;
}
.offset_outer {
border-collapse: collapse;
width: 100%;
}
.offset_outer td {
vertical-align: top;
white-space: nowrap;
padding: 0px;
}
.offset_outer input[type=radio] {
margin-left: 0px;
}
.offset_inner input[type=text] {
margin-bottom: 2px;
}
.offset_inner {
border-collapse: collapse;
}
.offset_inner td {
vertical-align: top;
height: 100%;
}
#custom_polygons_fieldset {
display: none;
}
#random_polygons_fieldset {
display: none;
}
#misc_fieldset {
min-height: 81px;
}
.custom_polygon_input {
width: 285px;
height: 15px;
padding: 2px;
line-height: 15px;
}
#polygon_explorer_string_inp {
margin: 0px;
height: 15px;
width: 323px;
padding: 2px;
line-height: 15px;
}
#custom_polygon_inputs {
display: block;
}
#custom_poly_message_box_green {
display: none;
opacity: 0;
position: absolute;
top: 0px;
left: 0px;
height: 0px;
width: 200px;
vertical-align: middle;
text-align: center;
border: 2px solid #106712;
background-color: #A3F1A5;
padding: 8px;
padding-left: 11px;
padding-right: 11px;
border-radius: 6px;
font-size: 16px;
z-index: 2;
}
.custom_polygon_inputs_table {
border-collapse: collapse;
}
#help_builtin_polygon_sets, #help_output_format, #help_custom_polygon {
padding-left: 7px;
padding-right: 7px;
}
/*
#outertable > tbody > tr > td {
border:1px solid red;
}
*/
</style>
</head>
<body>
<table id="outertable" style="margin-bottom:5px">
<tr>
<td id="td0" style="vertical-align:top;width:353px;min-width:353px">
<FIELDSET id="PolygonsFieldset">
<LEGEND title="Builtin polygons">Polygons</LEGEND>
<table class="polygons_fieldset">
<tr>
<td>
<input type="radio" name="polygons" value="0">Arrows
<br>
<input type="radio" name="polygons" value="1">Texts
<br>
<input type="radio" name="polygons" value="2">Rects
<br>
</td>
<td>
<input type="radio" name="polygons" value="3">Same
<br>
<input type="radio" name="polygons" value="4" title="Random rectangles">RandRects
<br>
<input type="radio" name="polygons" value="5" title="Random polygons">Random
<br>
</td>
<td>
<input type="radio" name="polygons" value="6">Star & rect
<br>
<input type="radio" name="polygons" value="7">Spiral
<br>
<input type="radio" name="polygons" value="8">Grid & star
<br>
</td>
<td style="padding-right:0px">
<input type="radio" name="polygons" value="9">Glyph
<br>
<input type="radio" name="polygons" value="10" title="Your own custom polygon sets">Custom
<br>
</td>
</tr>
</table>
</FIELDSET>
<FIELDSET id="custom_polygons_fieldset" style="position:relative">
<LEGEND>Your own custom polygons</LEGEND>
<select name="custom_polygons_select"
id="custom_polygons_select" title="Saved custom polygon sets"></select>
<select name="sample_custom_polygon" id="sample_custom_polygon" title="Get Builtin polygon set">
<option value="">- Builtin -</option>
<option value="0">Arrows</option>
<option value="1">Texts</option>
<option value="2">Rects</option>
<option value="3">Same</option>
<option value="4">RandRects</option>
<option value="5">Random</option>
<option value="6">Star & rect</option>
<option value="7">Spiral</option>
<option value="8">Grid & star</option>
<option value="9">Glyph</option>
</select>
<button id="help_builtin_polygon_sets" title="Get help about Builtin polygon sets">?</button>
<div id="custom_polygon_inputs">
<table class="custom_polygon_inputs_table">
<tr>
<td>
<b title="Here you can type or paste custom subject polygons. Click the '?' button below to get help about string formats.">Subj</b>
</td>
<td>
<textarea class="custom_polygon_input" id="custom_polygon_subj" value=''></textarea>
</td>
</tr>
<tr>
<td>
<b title="Here you can type or paste custom clip polygons. Click the '?' button below to get help about string formats.">Clip</b>
</td>
<td>
<textarea class="custom_polygon_input" id="custom_polygon_clip" value=''></textarea>
</td>
</tr>
</table>
<button id="save_custom_polygon" title="Update polygon set">Save</button>
<button id="add_as_new_custom_polygon" title="Save as a new custom polygon set">Save as new</button>
<button id="remove_custom_polygon" title="Remove selected custom polygon set">Del</button>
<button id="removeall_custom_polygon" title="Remove all custom polygon sets">Del all</button>
<button id="help_custom_polygon" title="Get help about custom polygons">?</button>
</div>
<div id="custom_poly_message_box_green"></div>
</FIELDSET>
<FIELDSET id="random_polygons_fieldset">
<LEGEND>Random polygon settings</LEGEND>
<table class="random">
<thead>
<tr>
<th class="random_new_th">
<button id="generate_random_polygons" title="Generate random polygons">New</button>
</th>
<th>Polygon count</th>
<th style="padding-right:0px">Point count</th>
</tr>
</thead>
<tbody>
<tr>
<td><b>Subject</b>
</td>
<td>
<button id="subj_polygon_count_minus">&nbsp;-&nbsp;</button>
<input class="random_inputs" type="text" id="subj_polygon_count"
value="1">
<button id="subj_polygon_count_plus">&nbsp;+&nbsp;</button>
</td>
<td style="padding-right:0px">
<button id="subj_point_count_minus">&nbsp;-&nbsp;</button>
<input class="random_inputs" type="text" id="subj_point_count"
value="1">
<button id="subj_point_count_plus">&nbsp;+&nbsp;</button>
</td>
</tr>
<tr>
<td><b>Clip</b>
</td>
<td>
<button id="clip_polygon_count_minus">&nbsp;-&nbsp;</button>
<input class="random_inputs" type="text" id="clip_polygon_count"
value="1">
<button id="clip_polygon_count_plus">&nbsp;+&nbsp;</button>
</td>
<td style="padding-right:0px">
<button id="clip_point_count_minus">&nbsp;-&nbsp;</button>
<input class="random_inputs" type="text" id="clip_point_count"
value="1">
<button id="clip_point_count_plus">&nbsp;+&nbsp;</button>
</td>
</tr>
</table>
</FIELDSET>
<table id="filltype_cliptype_table" class="filltype_cliptype" style="width:100%">
<tr>
<td>
<FIELDSET>
<LEGEND>Subject FillType</LEGEND>
<input type="radio" name="subject_fillType" value="0">EvenOdd&nbsp;
<input type="radio" name="subject_fillType" value="1">NonZero
</FIELDSET>
</td>
<td>
<FIELDSET>
<LEGEND>Clip FillType</LEGEND>
<input type="radio" name="clip_fillType" value="0">EvenOdd&nbsp;
<input type="radio" name="clip_fillType" value="1">NonZero
</FIELDSET>
</td>
</tr>
</table>
<FIELDSET id="ClipTypeFieldset">
<LEGEND>Clip type (operation)</LEGEND>
<p>
<input type="radio" name="clipType" value="">No&nbsp;&nbsp;
<input type="radio" name="clipType" value="0">Intersect&nbsp;&nbsp;
<input type="radio" name="clipType" value="1">Union&nbsp;&nbsp;
<input type="radio" name="clipType" value="2">Difference&nbsp;&nbsp;
<input type="radio" name="clipType" value="3" checked>Xor</p>
</FIELDSET>
<FIELDSET id="CleaningFieldset">
<LEGEND>Cleaning and simplifying</LEGEND>
<p>
<b>Clean</b><input type="checkbox" id="clean"
title="Clean. Joins sequential vertices if they are almost coincident. This is needed to prevent polygon breakage when offsetting. Not included in original Clipper, yet. This operation is done before offsetting."
value="1">
<input style="width:28px" type="text" value="" id="cleandelta"
title="Delta. The distance used for Clean. Sequential vertices are joined if they are at or below this distance of each other.">&nbsp;&nbsp;
<b>Simplify</b><input type="checkbox" id="Simplify"
title="Simplify. Convert complex (self-intersecting) polygon to simple ones. Produces multiple polygons, if there are self-intersections. Note! Not work in self-intersections that are too small, you have to remove too-near-vertices using Clean feature. This operation is done before offsetting."
value="1">&nbsp;&nbsp;
<b>Lighten</b><input type="checkbox" id="lighten"
title="Lighten. Removes unnecessary vertices. Not included in original Clipper. This operation is done after offsetting."
value="1">
<input style="width:28px;" type="text" value="" id="lighten_distance"
title="Tolerance. The distance that is used when lightening polygon. Removes middle vertex of three sequential vertices, if the middle vertex is under (or at) this distance of the line between start and end vertex.">
</p>
</FIELDSET>
<FIELDSET id="OffsettingFieldset">
<LEGEND>Offsetting</LEGEND>
<table class="offset_outer">
<tr>
<td style="padding-right:6px;">
<b><span style="display:block;padding-bottom:3px">Polygon</span></b>
<input type="radio" name="offsettable_poly" value="1">Subject
<br>
<input type="radio" name="offsettable_poly" value="2">Clip
<br>
<input type="radio" id="offsettable_poly3" name="offsettable_poly" value="3" checked>Solution
</td>
<td style="padding-right:6px;">
<b><span style="display:block;padding-bottom:3px">JoinType</span></b>
<input type="radio" name="joinType" value="0" checked>Square
<br>
<input type="radio" name="joinType" value="1">Round
<br>
<input type="radio" name="joinType" value="2">Miter
<br>
</td>
<td style="padding-right:0px">
<table class="offset_inner">
<tr>
<td>
<b>Delta</b>
<button id="delta_minus">&nbsp;-&nbsp;</button>
<input type="text" id="delta" value="" style="width:50px">
<button id="delta_plus">&nbsp;+&nbsp;</button>
</td>
</tr>
<tr>
<td><b>MiterLimit</b>
<button id="miterLimit_minus">&nbsp;-&nbsp;</button>
<input type="text" id="miterLimit" value="" style="width:20px">
<button id="miterLimit_plus">&nbsp;+&nbsp;</button>
</td>
</tr>
<tr>
<td style="padding-bottom: 0px; padding-top: 3px;">
<b title="AutoFix">AutoFix</b>
<input type="checkbox" id="AutoFix"
title="AutoFix. Fixes polygon orientation if necessary and removes duplicate vertices. Can be set false when you are sure that polygon orientation is correct and that there are no duplicate vertices."
value="1">
</td>
</tr>
</table>
</td>
</tr>
</table>
</FIELDSET>
<FIELDSET id="ScaleFieldset">
<LEGEND>Scale</LEGEND>
<p style="width:200px;">
<button id="scale_minus">&nbsp;-&nbsp;</button>
<input type="text" id="scale" value="" style="width:80px">
<button id="scale_plus">&nbsp;+&nbsp;</button>
<span id="scaletest"></span>&nbsp; <b>Bigints used:</b>
<span
id="biginteger_used"></span>
</p>
</FIELDSET>
<FIELDSET id="misc_fieldset">
<LEGEND>Misc</LEGEND>
<button id="show_svg_source" title="Show current SVG source in textarea at the bottom of the page">Show SVG source
B
</button>
&nbsp;<b>Bevel</b>
<input type="checkbox" id="bevel"
value="1" title="Uncheck this to speedup drawing">&nbsp;&nbsp;
<b>Explorer</b>
<input type="checkbox" id="sub_poly_links_update"
value="1" title="Enable/disable Polygon Explorer">
<br>
<button id="benchmark1" title="Execute Normal benchmark">Run NB</button>
<button id="benchmark1b" title="Execute Normal benchmark 5x">Run NB 5x</button>
<button id="benchmark2" title="Execute Big Integer benchmark">Run BIB</button>
<button id="benchmark2b" title="Execute Big Integer benchmark 5x">Run BIB 5x</button>
</FIELDSET>
</td>
<td id="td1" style="vertical-align:top;width:506px;">
<div id="svgcontainer"></div>
<FIELDSET id="polygon_explorer_fieldset">
<LEGEND>Polygon explorer</LEGEND>
<div id="polygon_explorer_div" style="padding-left:1px;/*overflow-y:auto;overflow-x:visible;max-height:300px*/">
<table class="polygon_explorer">
<tr>
<td colspan="4" style="padding:0px;white-space:nowrap; border:0px">
<table style="padding:0px;border-collapse:collapse">
<tr>
<td style="padding:0px;vertical-align:middle">
<textarea id="polygon_explorer_string_inp"
title="When you click the numbers below, here comes coordinates of the clicked polygon (or polygons) as string."
value="">Click below to see coordinates here!</textarea>
</td>
<td style="padding:0px;vertical-align:middle">
<select id="output_format" title="Output format" style="width:73px;margin-left:5px">
<option value="0">Clipper</option>
<option value="1">Plain</option>
<option value="2">SVG</option>
</select>&nbsp;&nbsp;<b>Ex</b> <input type="checkbox" id="ExPolygons"
value="1" title="Show solution as ExPolygons">
</td>
</tr>
</table>
</td>
</tr>
<tr>
<th style="white-space:nowrap; width:31px;" title="Polygon type">Type</th>
<th style="white-space:nowrap; width:35px;" title="Count of subpolygons">Polys</th>
<th style="white-space:nowrap; width:40px;"
title="Count of points in all sub polygons. Hover numbers below to see the whole multi-polygon highlighted. Click them to see coordinates as strings and the area of multi-polygon (sum of areas of sub polygons).">
Points
</th>
<th style="white-space:nowrap;"
title="Count of points in sub polygons. Hover numbers below to see sub polygons highlighted. Click to see coordinates as strings and the area of sub polygon."
>Points in subpolygons
</th>
</tr>
<tr>
<td id="subject_box" title="Subject">
<b>Subj</b>
</td>
<td>
<span id="subj_subpolygons"></span>
</td>
<td>
<span id="subj_points_total" class="subpolylinks"
onclick="popup_path(undefined,'1')"
onmouseover="show_path(undefined,'1')" onmouseout="hide_path()"
></span>
</td>
<td>
<span id="subj_points_in_subpolygons"></span>
</td>
</tr>
<tr>
<td id="clip_box" title="Clip">
<b>Clip</b>
</td>
<td>
<span id="clip_subpolygons">
</span>
</td>
<td><span id="clip_points_total" class="subpolylinks"
onclick="popup_path(undefined,'2')"
onmouseover="show_path(undefined,'2')" onmouseout="hide_path()"
></span>
</td>
<td>
<span id="clip_points_in_subpolygons"></span>
</td>
</tr>
<tr>
<td id="solution_box" title="Solution">
<b>Solu</b>
</td>
<td>
<span id="solution_subpolygons"></span>
</td>
<td>
<span id="solution_points_total" class="subpolylinks"
onclick="popup_path(undefined,'3')"
onmouseover="show_path(undefined,'3')" onmouseout="hide_path()"
></span>
</td>
<td>
<span id="solution_points_in_subpolygons">
</span>
</td>
</tr>
<tr>
<td>
<b>Total</b>
</td>
<td>
<b><span id="all_subpolygons"></span></b>
</td>
<td>
<b><span id="points_total"></span></b>
</td>
<td style="padding-bottom:0px;padding-top:0px;text-align:right;white-space:nowrap">
<div id="area" style="padding:0px;display:inline-block;width:90%;text-align:left"
title="Click above numbers to see the area of sub polygons."
>Area: Click above to see polygon area!
</div>
<button id="help_output_format" title="Get help about output format">?</button>
</td>
</tr>
</table>
</div>
</FIELDSET>
</td>
<td id="td2" style="width:353px;vertical-align:top;">
<FIELDSET id="benchmark_fieldset_f" style="min-height:517px">
<LEGEND id="#benchmark_fieldset">Benchmark</LEGEND>
<div id="benchmark_div_outer">
<div id="benchmark_div"></div>
<div id="benchmark_multiple_table_cont"></div>
</div>
</FIELDSET>
</td>
<td id="td3" style="vertical-align:top;">
<div id="svg_source_container2"></div>
</td>
</tr>
</table>
<div id="svg_source_container"></div>
<div id="benchmark_exports_container"></div>
<script>
//var scale = 10000000000000;
var scale = 100;
var global_do_not_round_and_scale = false;
var scale_addition = 100;
var polygons_default = 1;
var joinType = 0;
var miterLimit = 2.0;
var delta = -1;
var AutoFix = true;
var ExPolygons = false;
var SolutionPolygonClicked = false;
var SolutionPolygonString = "";
var Simplify = true;
var clean = true;
var cleandelta_default = 0.1;
var cleandelta = cleandelta_default;
var lighten = true;
var lighten_distance_default = 0.1;
var lighten_distance = lighten_distance_default;
var offsettable_poly = 3;
var clipType = 3;
var subject_fillType = 1;
var clip_fillType = 1;
var ss, cc, sss, cpr, sss2, cpr2, p, ok = 1;
var off_result;
var SVG = {}, p, p1, p2, p3;
var random_subj;
var random_clip;
var rnd_sett_defaults = {};
rnd_sett_defaults.rect = {};
rnd_sett_defaults.norm = {};
rnd_sett_defaults.rect.default = {
clip_polygon_count: 1,
clip_point_count: 4,
subj_polygon_count: 2,
subj_point_count: 8
};
rnd_sett_defaults.rect.min = {
clip_polygon_count: 1,
clip_point_count: 4,
subj_polygon_count: 1,
subj_point_count: 4
};
rnd_sett_defaults.rect.max = {
clip_polygon_count: 100,
clip_point_count: 100,
subj_polygon_count: 100,
subj_point_count: 100
};
rnd_sett_defaults.norm.default = {
clip_polygon_count: 1,
clip_point_count: 3,
subj_polygon_count: 2,
subj_point_count: 8
};
rnd_sett_defaults.norm.min = {
clip_polygon_count: 1,
clip_point_count: 3,
subj_polygon_count: 1,
subj_point_count: 3
};
rnd_sett_defaults.norm.max = {
clip_polygon_count: 100,
clip_point_count: 100,
subj_polygon_count: 100,
subj_point_count: 100
};
rnd_sett_defaults.current = "norm";
var bevel = 0;
var mypath = null;
var scaled_paths = [];
var subj_points_total = 0;
var clip_points_total = 0;
var solution_points_total = 0;
var subj_subpolygons = 0;
var clip_subpolygons = 0;
var solution_subpolygons = 0;
var bench;
var sub_poly_links_update = 1;
var benchmark1_globals = {};
var browserg;
var repeat_times;
var repeat = 0;
var clicked_benchmark_button_id;
var benchmark_exports = "";
var benchmark_running = 0;
var benchmark_automatic_click = 0;
// default_custom_subject_polygon = '[[{"X":90.58,"Y":144.44},{"X":84.83,"Y":141.55},{"X":84.84,"Y":141.56},{"X":79.29,"Y":137.67},{"X":73.79,"Y":132.61},{"X":73.8,"Y":132.62},{"X":67.31,"Y":125.14},{"X":63.22,"Y":119.64},{"X":98.93,"Y":130.63},{"X":98.91,"Y":130.61},{"X":99.02,"Y":130.66},{"X":99.02,"Y":130.65},{"X":97.46,"Y":134.4},{"X":93.93,"Y":139.6},{"X":90.86,"Y":144.22},{"X":90.87,"Y":144.21}]]';
// The Phantom image
var default_custom_subject_polygon = '[[{"X":-29.42,"Y":106.23},{"X":-26.7,"Y":105.45},{"X":-11.55,"Y":101.73},{"X":0.12,"Y":99.72},{"X":9.69,"Y":98.94},{"X":17.63,"Y":99.17},{"X":24.29,"Y":100.23},{"X":23.93,"Y":92.91},{"X":24.08,"Y":81.36},{"X":25.12,"Y":70.62},{"X":26.99,"Y":60.59},{"X":29.66,"Y":51.22},{"X":33.11,"Y":42.43},{"X":37.31,"Y":34.29},{"X":42.16,"Y":26.9},{"X":47.77,"Y":20.05},{"X":53.98,"Y":13.98},{"X":60.82,"Y":8.61},{"X":68.2,"Y":4.01},{"X":76.04,"Y":0.24},{"X":84.37,"Y":-2.73},{"X":92.97,"Y":-4.8},{"X":101.89,"Y":-5.98},{"X":110.88,"Y":-6.23},{"X":120,"Y":-5.53},{"X":128.99,"Y":-3.88},{"X":137.63,"Y":-1.36},{"X":145.94,"Y":2.06},{"X":153.87,"Y":6.32},{"X":161.29,"Y":11.4},{"X":168.03,"Y":17.14},{"X":174.22,"Y":23.66},{"X":179.7,"Y":30.8},{"X":184.49,"Y":38.59},{"X":188.54,"Y":46.95},{"X":191.9,"Y":56.1},{"X":194.44,"Y":65.81},{"X":196.16,"Y":76.18},{"X":197,"Y":87.09},{"X":196.94,"Y":98.79},{"X":195.88,"Y":111.18},{"X":207.51,"Y":112.86},{"X":220.6,"Y":115.46},{"X":233.18,"Y":118.91},{"X":245.2,"Y":123.18},{"X":256.72,"Y":128.26},{"X":267.97,"Y":134.26},{"X":278.87,"Y":141.16},{"X":289.73,"Y":149.16},{"X":300.72,"Y":158.47},{"X":312.07,"Y":169.38},{"X":312.06,"Y":169.37},{"X":336.14,"Y":177.32},{"X":355.16,"Y":183.81},{"X":367.78,"Y":189},{"X":378.1,"Y":194.19},{"X":387.04,"Y":199.69},{"X":395.07,"Y":205.73},{"X":400.6,"Y":210.78},{"X":405.18,"Y":216.34},{"X":409.93,"Y":223.8},{"X":420.35,"Y":243.91},{"X":425.91,"Y":253.51},{"X":431.07,"Y":260.69},{"X":436.35,"Y":266.43},{"X":436.34,"Y":266.42},{"X":443.69,"Y":272.81},{"X":469.07,"Y":291.49},{"X":477.42,"Y":298.56},{"X":483.68,"Y":305.01},{"X":487.79,"Y":310.56},{"X":497.28,"Y":326.73},{"X":502.53,"Y":334.76},{"X":511.53,"Y":345.81},{"X":337.97,"Y":345.86},{"X":315.38,"Y":321.1},{"X":303.25,"Y":308.84},{"X":303.26,"Y":308.85},{"X":292.72,"Y":299.47},{"X":292.69,"Y":299.5},{"X":290.88,"Y":300.26},{"X":290.89,"Y":300.25},{"X":283.26,"Y":304.12},{"X":283.27,"Y":304.11},{"X":275.98,"Y":308.87},{"X":268.76,"Y":314.74},{"X":261.42,"Y":322.02},{"X":261.43,"Y":322.01},{"X":253.55,"Y":331.26},{"X":242.73,"Y":345.84},{"X":-29.42,"Y":345.86}],[{"X":127.69,"Y":269.76},{"X":127.68,"Y":269.77},{"X":133.88,"Y":268.63},{"X":133.87,"Y":268.64},{"X":139.54,"Y":266.65},{"X":139.53,"Y":266.66},{"X":144.66,"Y":263.86},{"X":144.65,"Y":263.87},{"X":149.4,"Y":260.18},{"X":149.39,"Y":260.19},{"X":153.81,"Y":255.5},{"X":158.06,"Y":249.51},{"X":162.29,"Y":241.74},{"X":166.81,"Y":231.1},{"X":166.8,"Y":231.11},{"X":176.87,"Y":202.06},{"X":182.04,"Y":188.77},{"X":182.04,"Y":188.78},{"X":182.02,"Y":188.77},{"X":181.74,"Y":188.64},{"X":181.75,"Y":188.65},{"X":179.85,"Y":188.27},{"X":178.29,"Y":188.8},{"X":178.3,"Y":188.79},{"X":176.74,"Y":190.4},{"X":176.75,"Y":190.39},{"X":173.98,"Y":196.05},{"X":173.14,"Y":196.87},{"X":172.34,"Y":196.57},{"X":171.01,"Y":194.4},{"X":174.86,"Y":188.74},{"X":180.8,"Y":179.01},{"X":185.49,"Y":169.61},{"X":185.48,"Y":169.62},{"X":188.88,"Y":160.79},{"X":188.87,"Y":160.8},{"X":191.15,"Y":152.4},{"X":191.14,"Y":152.41},{"X":192.39,"Y":144.4},{"X":192.38,"Y":144.41},{"X":192.65,"Y":136.76},{"X":192.65,"Y":136.77},{"X":191.95,"Y":129.43},{"X":191.96,"Y":129.44},{"X":190.36,"Y":122.45},{"X":189.2,"Y":126.33},{"X":186.64,"Y":132.72},{"X":183.5,"Y":138.19},{"X":179.85,"Y":142.75},{"X":175.82,"Y":146.34},{"X":171.49,"Y":149.01},{"X":166.88,"Y":150.79},{"X":162.01,"Y":151.66},{"X":157.07,"Y":151.61},{"X":152.08,"Y":150.61},{"X":147.07,"Y":148.62},{"X":142.21,"Y":145.63},{"X":137.56,"Y":141.62},{"X":133.27,"Y":136.62},{"X":129.46,"Y":130.68},{"X":129.03,"Y":130.72},{"X":129.04,"Y":130.71},{"X":124.06,"Y":131.87},{"X":124.07,"Y":131.86},{"X":119.96,"Y":133.69},{"X":116.46,"Y":136.26},{"X":116.47,"Y":136.25},{"X":113.27,"Y":139.82},{"X":107.23,"Y":149.32},{"X":104.14,"Y":153.18},{"X":101,"Y":155.72},{"X":97.49,"Y":157.4},{"X":93.11,"Y":158.38},{"X":88.02,"Y":158.61},{"X":83.43,"Y":157.91},{"X":79.22,"Y":156.35},{"X":75,"Y":153.77},{"X":70.26,"Y":149.63},{"X":60.57,"Y":138.77},{"X":54.38,"Y":132.43},{"X":54.39,"Y":132.44},{"X":49.11,"Y":128.26},{"X":43.79,"Y":125.19},{"X":43.54,"Y":126.2},{"X":43.55,"Y":126.19},{"X":43.13,"Y":133.69},{"X":43.13,"Y":133.68},{"X":43.66,"Y":140.71},{"X":43.65,"Y":140.7},{"X":45.09,"Y":147.46},{"X":45.08,"Y":147.45},{"X":47.49,"Y":154.24},{"X":47.48,"Y":154.23},{"X":51.23,"Y":161.76},{"X":63.98,"Y":182.09},{"X":67.55,"Y":189.18},{"X":69.73,"Y":195.4},{"X":70.92,"Y":201.58},{"X":71.16,"Y":207.87},{"X":70.43,"Y":214.58},{"X":68.63,"Y":221.85},{"X":68.56,"Y":221.68},{"X":68.68,"Y":203.83},{"X":68.68,"Y":203.84},{"X":68.16,"Y":197.1},{"X":68.17,"Y":197.11},{"X":67.05,"Y":192.75},{"X":67.06,"Y":192.76},{"X":65.5,"Y":189.85},{"X":65.51,"Y":189.86},{"X":63.49,"Y":187.86},{"X":60.94,"Y":186.59},{"X":60.95,"Y":186.6},{"X":57.55,"Y":186.01},{"X":57.78,"Y":189.41},{"X":57.77,"Y":189.4},{"X":59.32,"Y":201.46},{"X":59.31,"Y":201.45},{"X":61.68,"Y":212.76},{"X":61.67,"Y":212.75},{"X":64.76,"Y":223.11},{"X":64.75,"Y":223.1},{"X":68.47,"Y":232.45},{"X":72.73,"Y":240.71},{"X":77.44,"Y":247.84},{"X":82.45,"Y":253.81},{"X":82.44,"Y":253.8},{"X":87.82,"Y":258.8},{"X":93.45,"Y":262.8},{"X":93.44,"Y":262.79},{"X":99.44,"Y":265.89},{"X":99.43,"Y":265.88},{"X":105.67,"Y":268.07},{"X":105.66,"Y":268.06},{"X":112.58,"Y":269.42},{"X":112.57,"Y":269.41},{"X":120.65,"Y":270.08}],[{"X":121.49,"Y":240.71},{"X":118.05,"Y":239.68},{"X":115.54,"Y":238.06},{"X":113.86,"Y":235.97},{"X":112.96,"Y":233.43},{"X":112.86,"Y":230.29},{"X":112.9,"Y":230.33},{"X":114.24,"Y":231.32},{"X":114.23,"Y":231.31},{"X":117.3,"Y":232.52},{"X":117.29,"Y":232.51},{"X":122.05,"Y":233.17},{"X":132.1,"Y":233.03},{"X":140.49,"Y":233.06},{"X":139.8,"Y":234.49},{"X":137.79,"Y":236.7},{"X":134.63,"Y":238.72},{"X":130.41,"Y":240.25},{"X":125.83,"Y":240.94}],[{"X":90.76,"Y":219.11},{"X":91.11,"Y":218.28},{"X":92.59,"Y":216.41},{"X":95.22,"Y":214.69},{"X":99.6,"Y":213.14},{"X":106.53,"Y":211.9},{"X":117,"Y":211.2},{"X":143.09,"Y":211.3},{"X":151.49,"Y":210.93},{"X":151.31,"Y":211.49},{"X":149.72,"Y":213.46},{"X":147.43,"Y":214.88},{"X":143.97,"Y":215.82},{"X":138.58,"Y":216.09},{"X":117.5,"Y":214.37},{"X":117.51,"Y":214.38},{"X":108.81,"Y":214.35},{"X":101.81,"Y":215.22},{"X":101.82,"Y":215.21},{"X":95.89,"Y":216.85},{"X":95.9,"Y":216.84},{"X":90.72,"Y":219.22}],[{"X":131.9,"Y":198.38},{"X":131.72,"Y":195.25},{"X":132.38,"Y":192.96},{"X":133.78,"Y":191.21},{"X":136.84,"Y":189.34},{"X":140.91,"Y":187.09},{"X":142.52,"Y":185.35},{"X":143.3,"Y":183.24},{"X":143.3,"Y":183.25},{"X":143.22,"Y":183.13},{"X":141.54,"Y":180.12},{"X":139.65,"Y":174.9},{"X":138.32,"Y":168.14},{"X":137.62,"Y":158.87},{"X":137.76,"Y":144.77},{"X":142.77,"Y":159.79},{"X":145.49,"Y":170.4},{"X":146.69,"Y":178.4},{"X":146.78,"Y":184.55},{"X":146.02,"Y":189.28},{"X":144.58,"Y":192.89},{"X":142.54,"Y":195.66},{"X":139.94,"Y":197.69},{"X":136.5,"Y":199.14},{"X":132.22,"Y":199.86}],[{"X":108.17,"Y":188.43},{"X":106.35,"Y":187.41},{"X":105.13,"Y":185.72},{"X":104.56,"Y":183.25},{"X":104.61,"Y":183.31},{"X":104.81,"Y":183.53},{"X":104.8,"Y":183.52},{"X":106.39,"Y":184.64},{"X":106.38,"Y":184.63},{"X":108.44,"Y":184.92},{"X":114,"Y":183.64},{"X":116.83,"Y":183.49},{"X":119.09,"Y":184.26},{"X":121.14,"Y":185.99},{"X":120.5,"Y":186.31},{"X":114.98,"Y":188.22},{"X":110.98,"Y":188.8}],[{"X":163.89,"Y":141.31},{"X":166.4,"Y":137.22},{"X":174.87,"Y":127.72},{"X":177.63,"Y":123.62},{"X":179.28,"Y":119.59},{"X":179.28,"Y":119.6},{"X":179.24,"Y":119.6},{"X":179.25,"Y":119.59},{"X":176.7,"Y":119.79},{"X":176.71,"Y":119.78},{"X":170.65,"Y":120.87},{"X":170.66,"Y":120.86},{"X":164.55,"Y":122.95},{"X":164.56,"Y":122.94},{"X":148.93,"Y":130.62},{"X":148.93,"Y":130.61},{"X":149.21,"Y":132.61},{"X":149.2,"Y":132.6},{"X":150.57,"Y":136.41},{"X":152.55,"Y":139.41},{"X":152.54,"Y":139.4},{"X":155.11,"Y":141.71},{"X":155.1,"Y":141.7},{"X":158.43,"Y":143.41},{"X":158.42,"Y":143.4},{"X":162.62,"Y":144.5}],[{"X":90.87,"Y":144.21},{"X":90.86,"Y":144.22},{"X":93.93,"Y":139.6},{"X":97.47,"Y":134.4},{"X":99.02,"Y":130.65},{"X":99.02,"Y":130.66},{"X":98.92,"Y":130.62},{"X":98.93,"Y":130.63},{"X":63.23,"Y":119.65},{"X":67.31,"Y":125.14},{"X":73.8,"Y":132.62},{"X":73.79,"Y":132.61},{"X":79.3,"Y":137.68},{"X":84.84,"Y":141.56},{"X":84.83,"Y":141.55},{"X":90.59,"Y":144.44}],[{"X":128.47,"Y":127.41},{"X":128.46,"Y":127.42},{"X":134.55,"Y":125.73},{"X":134.54,"Y":125.74},{"X":141.57,"Y":122.7},{"X":141.56,"Y":122.71},{"X":154.9,"Y":115.21},{"X":164.19,"Y":110.32},{"X":171.16,"Y":107.57},{"X":177.57,"Y":106},{"X":183.83,"Y":105.4},{"X":190.12,"Y":105.74},{"X":182.19,"Y":101.42},{"X":182.2,"Y":101.43},{"X":175.96,"Y":98.85},{"X":175.97,"Y":98.86},{"X":170.13,"Y":97.45},{"X":170.14,"Y":97.46},{"X":164,"Y":96.98},{"X":157.13,"Y":97.46},{"X":157.08,"Y":97.55},{"X":156.47,"Y":98.59},{"X":153.09,"Y":103.35},{"X":149.58,"Y":106.83},{"X":145.83,"Y":109.3},{"X":141.79,"Y":110.89},{"X":137.21,"Y":111.67},{"X":131.97,"Y":111.55},{"X":125.78,"Y":110.38},{"X":117.8,"Y":107.78},{"X":92.32,"Y":96.78},{"X":92.33,"Y":96.79},{"X":83.94,"Y":93.95},{"X":83.95,"Y":93.96},{"X":77.47,"Y":92.65},{"X":77.48,"Y":92.66},{"X":72.09,"Y":92.45},{"X":67.49,"Y":93.16},{"X":67.5,"Y":93.15},{"X":63.28,"Y":94.72},{"X":63.29,"Y":94.71},{"X":59.49,"Y":97.13},{"X":55.9,"Y":100.53},{"X":55.91,"Y":100.52},{"X":52.52,"Y":105.09},{"X":49.25,"Y":111.18},{"X":52.77,"Y":110.06},{"X":58.91,"Y":108.7},{"X":64.9,"Y":108.31},{"X":70.86,"Y":108.86},{"X":77.27,"Y":110.42},{"X":84.53,"Y":113.27},{"X":105.04,"Y":123.87},{"X":105.03,"Y":123.86},{"X":111.74,"Y":126.51},{"X":111.73,"Y":126.5},{"X":117.29,"Y":127.77},{"X":117.28,"Y":127.76},{"X":122.82,"Y":128.06}]]';
// The Phantom Scull Ring
var default_custom_clip_polygon = '[[{"X":385.77,"Y":166.95},{"X":386.68,"Y":182.21},{"X":386.15,"Y":187.13},{"X":384.96,"Y":190.99},{"X":382.95,"Y":194.32},{"X":379.91,"Y":197.42},{"X":375.8,"Y":200.17},{"X":370.59,"Y":202.47},{"X":364.37,"Y":204.13},{"X":357.67,"Y":204.95},{"X":350.93,"Y":204.85},{"X":344.71,"Y":203.87},{"X":339.59,"Y":202.19},{"X":335.52,"Y":199.95},{"X":332.61,"Y":197.37},{"X":330.67,"Y":194.53},{"X":329.62,"Y":191.41},{"X":329.4,"Y":187.83},{"X":330.02,"Y":183.99},{"X":331.34,"Y":181.14},{"X":333.27,"Y":179.05},{"X":336.11,"Y":177.4},{"X":340.71,"Y":176.05},{"X":352.22,"Y":173.98},{"X":356.3,"Y":172.51},{"X":361.25,"Y":169.4},{"X":370.03,"Y":163.12},{"X":373.94,"Y":161.22},{"X":377.01,"Y":160.6},{"X":379.53,"Y":160.94},{"X":381.9,"Y":162.19},{"X":384.3,"Y":164.67},{"X":385.77,"Y":166.93},{"X":385.77,"Y":166.95}],[{"X":270.77,"Y":163.84},{"X":273.88,"Y":163.4},{"X":276.36,"Y":163.91},{"X":280.12,"Y":166.08},{"X":285.42,"Y":169.2},{"X":291.02,"Y":171.33},{"X":299.79,"Y":173.33},{"X":307.68,"Y":175.13},{"X":311.52,"Y":176.78},{"X":314.01,"Y":178.71},{"X":315.73,"Y":181.15},{"X":316.83,"Y":184.41},{"X":317.2,"Y":188.89},{"X":316.7,"Y":192.44},{"X":315.33,"Y":195.6},{"X":313.05,"Y":198.56},{"X":309.77,"Y":201.19},{"X":305.39,"Y":203.45},{"X":299.95,"Y":205.12},{"X":293.7,"Y":205.99},{"X":287.12,"Y":205.96},{"X":280.75,"Y":205},{"X":275.02,"Y":203.24},{"X":270.05,"Y":200.77},{"X":266.08,"Y":197.81},{"X":263.1,"Y":194.47},{"X":260.95,"Y":190.7},{"X":259.68,"Y":186.54},{"X":259.29,"Y":181.92},{"X":259.86,"Y":176.86},{"X":261.36,"Y":171.68},{"X":263.13,"Y":168.56},{"X":265.48,"Y":166.25},{"X":268.64,"Y":164.55},{"X":270.77,"Y":163.84},{"X":270.77,"Y":163.84}],[{"X":303.94,"Y":227.57},{"X":303.74,"Y":222.63},{"X":304.44,"Y":218.35},{"X":305.94,"Y":214.59},{"X":308.24,"Y":211.27},{"X":311.35,"Y":208.46},{"X":315.13,"Y":206.32},{"X":319.55,"Y":204.91},{"X":324.31,"Y":204.37},{"X":329.09,"Y":204.77},{"X":333.4,"Y":206.03},{"X":337.08,"Y":208.03},{"X":339.99,"Y":210.61},{"X":342.11,"Y":213.69},{"X":343.43,"Y":217.19},{"X":343.94,"Y":221.2},{"X":343.59,"Y":225.38},{"X":342.53,"Y":228.01},{"X":340.69,"Y":230.17},{"X":337.79,"Y":232.02},{"X":333.81,"Y":233.34},{"X":328.46,"Y":234.03},{"X":322.02,"Y":233.83},{"X":315.23,"Y":232.68},{"X":309.37,"Y":230.81},{"X":305.39,"Y":228.73},{"X":303.99,"Y":227.61},{"X":303.94,"Y":227.57}],[{"X":421.16,"Y":64.34},{"X":421.16,"Y":64.45},{"X":422.7,"Y":96.81},{"X":424.28,"Y":149.29},{"X":425.26,"Y":216.18},{"X":425.57,"Y":327.02},{"X":220.65,"Y":327.02},{"X":220.41,"Y":312.05},{"X":219.69,"Y":230.46},{"X":219.9,"Y":144.66},{"X":221.01,"Y":65.92},{"X":233.74,"Y":65.34},{"X":264.63,"Y":64.78},{"X":325.38,"Y":65.31},{"X":363.93,"Y":65.31},{"X":393.46,"Y":64.48},{"X":420.05,"Y":62.8},{"X":420.8,"Y":62.92},{"X":421.16,"Y":64.34}],[{"X":404.57,"Y":215.13},{"X":404.58,"Y":215.08},{"X":405.83,"Y":205.87},{"X":406.55,"Y":194.49},{"X":406.3,"Y":183.7},{"X":405.16,"Y":173.57},{"X":403.17,"Y":164.03},{"X":400.34,"Y":155.01},{"X":396.73,"Y":146.61},{"X":392.34,"Y":138.76},{"X":387.23,"Y":131.54},{"X":381.47,"Y":125.02},{"X":375.05,"Y":119.16},{"X":368.04,"Y":114.01},{"X":360.39,"Y":109.58},{"X":352.3,"Y":105.99},{"X":343.88,"Y":103.28},{"X":335.22,"Y":101.47},{"X":326.42,"Y":100.59},{"X":317.57,"Y":100.64},{"X":308.77,"Y":101.63},{"X":300.3,"Y":103.53},{"X":292.06,"Y":106.33},{"X":284.32,"Y":109.94},{"X":277.01,"Y":114.36},{"X":270.11,"Y":119.65},{"X":263.83,"Y":125.67},{"X":258.83,"Y":131.66},{"X":253.44,"Y":139.77},{"X":248.35,"Y":149.13},{"X":244.11,"Y":158.88},{"X":241,"Y":168.28},{"X":239,"Y":177.21},{"X":238.09,"Y":185.4},{"X":238.17,"Y":192.9},{"X":239.13,"Y":199.57},{"X":240.91,"Y":205.61},{"X":243.45,"Y":211.04},{"X":246.09,"Y":214.67},{"X":249.63,"Y":217.9},{"X":254.56,"Y":221.04},{"X":270.07,"Y":228.23},{"X":275.76,"Y":231.57},{"X":279.77,"Y":234.93},{"X":282.82,"Y":238.64},{"X":285.08,"Y":242.87},{"X":286.61,"Y":247.72},{"X":287.31,"Y":253.67},{"X":287.19,"Y":274.82},{"X":287.8,"Y":281.39},{"X":288.94,"Y":285.57},{"X":290.57,"Y":288.43},{"X":292.56,"Y":290.24},{"X":295.1,"Y":291.3},{"X":295.83,"Y":283.75},{"X":296,"Y":265.6},{"X":296.31,"Y":250.76},{"X":297.29,"Y":241.67},{"X":299.29,"Y":240.75},{"X":303.08,"Y":239.94},{"X":307.26,"Y":240},{"X":307.66,"Y":242.7},{"X":309.02,"Y":256.43},{"X":310.57,"Y":291.19},{"X":311.49,"Y":291.54},{"X":314.95,"Y":293.31},{"X":316.6,"Y":294.82},{"X":316.6,"Y":295.52},{"X":315.35,"Y":295.94},{"X":315.99,"Y":296.02},{"X":317.2,"Y":295.47},{"X":318.2,"Y":293.99},{"X":319.04,"Y":290.81},{"X":319.39,"Y":284.37},{"X":318.41,"Y":262.35},{"X":318.37,"Y":251.85},{"X":319.08,"Y":245.75},{"X":320.27,"Y":242.05},{"X":321.29,"Y":240.77},{"X":323.08,"Y":240.08},{"X":328.27,"Y":240},{"X":329.31,"Y":244.13},{"X":330.19,"Y":252.03},{"X":330.63,"Y":273.92},{"X":331.25,"Y":284.25},{"X":332.33,"Y":289.99},{"X":333.7,"Y":293.31},{"X":335.26,"Y":295.12},{"X":337.01,"Y":295.94},{"X":337.81,"Y":294.94},{"X":339.33,"Y":291.52},{"X":340.32,"Y":286.68},{"X":340.61,"Y":279.64},{"X":339.38,"Y":260.18},{"X":339.31,"Y":251.49},{"X":340.05,"Y":245.84},{"X":341.39,"Y":241.87},{"X":342.36,"Y":240.75},{"X":344.31,"Y":240.04},{"X":351.49,"Y":240},{"X":351.67,"Y":240.9},{"X":352.45,"Y":248.35},{"X":352.58,"Y":274.14},{"X":353.27,"Y":282.2},{"X":354.5,"Y":287.31},{"X":356.13,"Y":290.68},{"X":358.13,"Y":292.86},{"X":359.27,"Y":289.89},{"X":360.62,"Y":284.2},{"X":361.32,"Y":276.58},{"X":361.16,"Y":258.89},{"X":361.45,"Y":247.8},{"X":365.47,"Y":242.21},{"X":370.89,"Y":236.4},{"X":377.72,"Y":230.54},{"X":385.94,"Y":224.78},{"X":395.31,"Y":219.41},{"X":404.57,"Y":215.13}]]';
var output_format = 1;
var ClipperLib_MaxSteps_original; // this hold the original ClipperLib.MaxSteps value
// during benchmarks, ClipperLib.MaxSteps is set to 10
// to make benchmarks comparable
var ismousedown = false; // Not yet in use
var bench_glob = [];
var bench_elapsed_time = 0;
window.onerror = function (message, url, linenumber) {
console.log("JavaScript error: " + message + " on line " + linenumber + " for " + url);
}
window.onload = function () {
"use strict";
if (typeof (jQuery) == "undefined") alert("Failed to load jQuery. Please reload the page. If this fails repeatedly, please check the path of the jQuery library or use an own copy of jQuery.");
else if (typeof (Raphael) == "undefined") alert("Failed to load Raphael. Please reload the page. If this fails repeatedly, please check the path of the Raphael library or use an own copy of Raphael.");
else if (typeof (ClipperLib) == "undefined") alert("Failed to load ClipperLib. Please check that ClipperLib is loaded correctly.");
else {
ClipperLib_MaxSteps_original = ClipperLib.MaxSteps;
browserg = get_browser();
bench = new benchmark("bench");
p = SVG.create();
main();
}
}
function get_browser() {
var nav = navigator.userAgent.toString().toLowerCase();
var browser = {};
if (nav.indexOf("chrome") != -1 && nav.indexOf("chromium") == -1) browser.chrome = 1;
else browser.chrome = 0;
if (nav.indexOf("chromium") != -1) browser.chromium = 1;
else browser.chromium = 0;
if (nav.indexOf("safari") != -1 && nav.indexOf("chrome") == -1 && nav.indexOf("chromium") == -1) browser.safari = 1;
else browser.safari = 0;
if (nav.indexOf("firefox") != -1) browser.firefox = 1;
else browser.firefox = 0;
if (nav.indexOf("firefox/17") != -1) browser.firefox17 = 1;
else browser.firefox17 = 0;
if (nav.indexOf("firefox/15") != -1) browser.firefox15 = 1;
else browser.firefox15 = 0;
if (nav.indexOf("firefox/3") != -1) browser.firefox3 = 1;
else browser.firefox3 = 0;
if (nav.indexOf("opera") != -1) browser.opera = 1;
else browser.opera = 0;
if (nav.indexOf("msie 10") != -1) browser.msie10 = 1;
else browser.msie10 = 0;
if (nav.indexOf("msie 9") != -1) browser.msie9 = 1;
else browser.msie9 = 0;
if (nav.indexOf("msie 8") != -1) browser.msie8 = 1;
else browser.msie8 = 0;
if (nav.indexOf("msie 7") != -1) browser.msie7 = 1;
else browser.msie7 = 0;
if (nav.indexOf("msie ") != -1) browser.msie = 1;
else browser.msie = 0;
for (var i in browser) {
if (browser[i] === 1 && !i.match(/[0-9]/)) browser["browser"] = i;
}
browser.version = $.browser.version;
return browser;
}
// Helper tool to piece together Raphael's paths into strings again
Array.prototype.flatten || (Array.prototype.flatten = function () {
return this.reduce(function (a, b) {
return a.concat('function' === typeof b.flatten ? b.flatten() : b);
}, []);
});
/* Normalizes inputted polygon string
Input can be JSON-stringified version of the following:
- Array of arrays of points [[{"X":10,"Y":10},{"X":10,"Y":10},{"X":10,"Y":10}][{"X":10,"Y":10},{"X":10,"Y":10},{"X":10,"Y":10}]]
- Array of points [{"X":10,"Y":10},{"X":10,"Y":10},{"X":10,"Y":10}]
- The aboves in lowercase
- Array of x,y-coordinates eg. [0, 10, 20, 30, 40, 50] or [0 10 20 30 40 50] or [0 10, 20 30, 40 50];
- The above without []
- SVG path string with commands MLVHZ and mlvhz
Returns normalized Clipper Polygons object stringified or false in failure
*/
function normalize_clipper_poly(polystr, quiet) {
if (typeof (polystr) != "string") return false;
polystr = polystr.trim();
var np, txt;
if (polystr.substr(0, 1).toUpperCase() === "M") {
np = svgpath_to_clipper_polygons(polystr);
if (np === false) {
txt = "Unable to parse SVG path string.\n";
txt += "Click OK to continue.\n";
if (!quiet) alert(txt);
return false;
}
else return JSON.stringify(np);
}
polystr = polystr.replace(/[\s,]+/g, ",");
if (polystr.substr(0, 1) !== "[") polystr = "[" + polystr;
if (polystr.substr(-1, 1) !== "]") polystr = polystr + "]";
try {
var poly = JSON.parse(polystr);
}
catch (err) {
txt = "Unable to parse polygon string.\n";
txt += "Error: " + err.message + "\n";
txt += "Click OK to continue.\n";
if (!quiet) alert(txt);
return false;
}
// if only points without "X" and "Y"
var temp_n = [], i;
if (isArray(poly) && poly.length && typeof (poly[0]) == "number") {
var len = poly.length;
for (i = 0; i < len; i = i + 2) {
temp_n.push({
X: poly[i],
Y: poly[i + 1]
});
}
poly = temp_n;
}
// if an array of array of points without "X" and "Y"
var temp_n2 = [], i, j, len, len2;
if (isArray(poly) && poly.length && isArray(poly[0]) && typeof(poly[0][0]) != "undefined" &&
typeof(poly[0][0].X) == "undefined" &&
typeof(poly[0][0].x) == "undefined") {
len2 = poly.length;
for (j = 0; j < len2; j++) {
temp_n = [];
len = poly[j].length;
for (i = 0; i < len; i = i + 2) {
temp_n.push({
X: poly[j][i],
Y: poly[j][i + 1]
});
}
temp_n2.push(temp_n);
}
poly = temp_n2;
}
// if not array of arrays, convert to array of arrays
if (isArray(poly) && poly.length > 0 && !isArray(poly[0])) poly = [poly];
var pp, n = [
[]
],
m, pm, x, y;
np = [
[]
];
for (i = 0, m = poly.length; i < m; i++) {
np[i] = [];
for (j = 0, pm = poly[i].length; j < pm; j++) {
pp = {};
y = null;
x = null;
if (typeof (poly[i][j].X) != "undefined" && !isNaN(Number(poly[i][j].X))) x = Number(poly[i][j].X);
else if (typeof (poly[i][j].x) != "undefined" && !isNaN(Number(poly[i][j].x))) x = Number(poly[i][j].x);
if (typeof (poly[i][j].Y) != "undefined" && !isNaN(Number(poly[i][j].Y))) y = Number(poly[i][j].Y);
else if (typeof (poly[i][j].y) != "undefined" && !isNaN(Number(poly[i][j].y))) y = Number(poly[i][j].y);
if (y !== null && x !== null) {
pp.X = x;
pp.Y = y;
np[i].push(pp);
}
else {
txt = "Unable to parse polygon string.\n";
txt += "Error: Coordinates are not in a right form.\n";
txt += "Click OK to continue.\n";
if (!quiet) alert(txt);
return false;
}
;
}
}
return JSON.stringify(np);
}
// helper function for normalize_clipper_poly()
function svgpath_to_clipper_polygons(d) {
var arr;
d = d.trim();
arr = Raphael.parsePathString(d); // str to array
arr = Raphael._pathToAbsolute(arr); // mahvstcsqz -> uppercase
var str = arr.flatten().join(" ");
var paths = str.replace(/M/g, '|M').split("|");
var k, polygons_arr = [],
polygon_arr = [];
for (k = 0; k < paths.length; k++) {
if (paths[k].trim() === "") continue;
arr = Raphael.parsePathString(paths[k].trim());
polygon_arr = [];
var i = 0,
j, m = arr.length,
letter = "",
x = 0,
y = 0,
pt = {},
subpath_start = {};
subpath_start.x = "";
subpath_start.y = "";
for (; i < m; i++) {
letter = arr[i][0].toUpperCase();
if (letter != "M" && letter != "L" && letter != "Z") continue;
if (letter != "Z") {
for (j = 1; j < arr[i].length; j = j + 2) {
if (letter == "V") y = arr[i][j];
else if (letter == "H") x = arr[i][j];
else {
x = arr[i][j];
y = arr[i][j + 1];
}
pt = {};
pt.X = null;
pt.Y = null;
if (typeof (x) != "undefined" && !isNaN(Number(x))) pt.X = Number(x);
if (typeof (y) != "undefined" && !isNaN(Number(y))) pt.Y = Number(y);
if (pt.X !== null && pt.Y !== null) {
polygon_arr.push(pt);
}
else {
return false;
}
}
}
if ((letter != "Z" && subpath_start.x === "") || letter == "M") {
subpath_start.x = x;
subpath_start.y = y;
}
if (letter == "Z") {
x = subpath_start.x;
y = subpath_start.y;
}
}
polygons_arr.push(polygon_arr);
}
return polygons_arr;
}
function format_output(polystr, ExPolygonsOrNot) {
var txt, returnAsExPolygons = false;
if (typeof(polystr) != "string" || polystr === "") return "";
if (ExPolygonsOrNot == "ExPolygons" && ExPolygons) returnAsExPolygons = true;
try {
var poly = JSON.parse(polystr);
}
catch (err) {
txt = "Unable to parse polygon for output.\n";
txt += "Error: " + err.message + "\n";
txt += "Click OK to continue.\n";
alert(txt);
return "";
}
var i, j, m, n, newpolystr = "";
if (output_format === 0) // Clipper
{
//return polystr;
}
else if (output_format == 1) // Plain
{
m = poly.length;
for (i = 0; i < m; i++) {
newpolystr += "[";
n = poly[i].length;
for (j = 0; j < n; j++) {
newpolystr += poly[i][j].X + "," + poly[i][j].Y;
if (j !== n - 1) newpolystr += ", ";
}
newpolystr += "]";
if (i !== m - 1) newpolystr += ",";
}
//return "[" + newpolystr + "]";
polystr = "[" + newpolystr + "]";
}
else if (output_format == 2) // SVG
{
m = poly.length;
for (i = 0; i < m; i++) {
n = poly[i].length;
for (j = 0; j < n; j++) {
if (j === 0) newpolystr += "M";
else newpolystr += "L";
newpolystr += poly[i][j].X + "," + poly[i][j].Y;
if (j !== n - 1) newpolystr += " ";
}
newpolystr += "Z";
if (i !== m - 1) newpolystr += " ";
}
if (newpolystr.trim() == "Z") newpolystr = "";
//return newpolystr;
polystr = newpolystr;
}
if (!returnAsExPolygons) return polystr;
else {
if (!cpr) cpr = new ClipperLib.Clipper();
else cpr.Clear();
cpr.AddPolygons(poly, ClipperLib.PolyType.ptSubject);
sss = new ClipperLib.ExPolygons();
cpr.Execute(clipType, sss, subject_fillType, clip_fillType);
polystr = JSON.stringify(sss);
return polystr;
}
/*
else if (output_format == 3) // ExPolygons
{
m = poly.length;
for(i = 0; i < m; i++)
{
newpolystr += "[";
n = poly[i].length;
for(j = 0; j < n; j++)
{
newpolystr += poly[i][j].X + "," + poly[i][j].Y;
if (j !== n - 1) newpolystr += ", ";
}
newpolystr += "]";
if (i !== m - 1) newpolystr += ",";
}
return "[" + newpolystr + "]";
}
*/
}
;
function get_gb_and_arrow() {
var gb_poly = '[[{"X":189,"Y":154},{"X":192,"Y":155},{"X":193,"Y":154},{"X":194,"Y":151},{"X":194,"Y":150},{"X":196,"Y":151},{"X":199,"Y":151},{"X":200,"Y":150},{"X":203,"Y":149},{"X":204,"Y":149},{"X":205,"Y":149},{"X":206,"Y":149},{"X":209,"Y":150},{"X":211,"Y":149},{"X":212,"Y":149},{"X":212,"Y":152},{"X":212,"Y":155},{"X":213,"Y":156},{"X":213,"Y":156},{"X":213,"Y":159},{"X":214,"Y":159},{"X":216,"Y":161},{"X":216,"Y":161},{"X":216,"Y":163},{"X":217,"Y":163},{"X":218,"Y":166},{"X":218,"Y":166},{"X":217,"Y":163},{"X":217,"Y":162},{"X":218,"Y":162},{"X":218,"Y":163},{"X":220,"Y":165},{"X":218,"Y":167},{"X":218,"Y":167},{"X":215,"Y":168},{"X":214,"Y":170},{"X":214,"Y":171},{"X":215,"Y":170},{"X":217,"Y":168},{"X":220,"Y":168},{"X":221,"Y":168},{"X":222,"Y":170},{"X":223,"Y":172},{"X":223,"Y":174},{"X":224,"Y":176},{"X":223,"Y":178},{"X":223,"Y":179},{"X":223,"Y":179},{"X":223,"Y":180},{"X":222,"Y":180},{"X":222,"Y":180},{"X":221,"Y":178},{"X":222,"Y":177},{"X":222,"Y":174},{"X":221,"Y":174},{"X":220,"Y":172},{"X":218,"Y":172},{"X":218,"Y":174},{"X":220,"Y":174},{"X":218,"Y":174},{"X":220,"Y":176},{"X":220,"Y":178},{"X":217,"Y":180},{"X":221,"Y":179},{"X":221,"Y":180},{"X":222,"Y":181},{"X":221,"Y":182},{"X":218,"Y":183},{"X":218,"Y":183},{"X":217,"Y":182},{"X":214,"Y":184},{"X":214,"Y":187},{"X":214,"Y":189},{"X":211,"Y":191},{"X":210,"Y":191},{"X":210,"Y":190},{"X":209,"Y":189},{"X":207,"Y":189},{"X":206,"Y":189},{"X":207,"Y":189},{"X":209,"Y":191},{"X":209,"Y":191},{"X":210,"Y":191},{"X":207,"Y":192},{"X":205,"Y":191},{"X":204,"Y":191},{"X":204,"Y":191},{"X":204,"Y":192},{"X":204,"Y":194},{"X":204,"Y":195},{"X":206,"Y":195},{"X":206,"Y":198},{"X":206,"Y":199},{"X":206,"Y":200},{"X":206,"Y":202},{"X":206,"Y":203},{"X":210,"Y":206},{"X":209,"Y":207},{"X":207,"Y":211},{"X":209,"Y":212},{"X":210,"Y":212},{"X":210,"Y":213},{"X":207,"Y":212},{"X":207,"Y":213},{"X":206,"Y":213},{"X":207,"Y":215},{"X":209,"Y":216},{"X":209,"Y":218},{"X":210,"Y":222},{"X":209,"Y":226},{"X":211,"Y":226},{"X":209,"Y":231},{"X":207,"Y":232},{"X":206,"Y":234},{"X":205,"Y":237},{"X":205,"Y":240},{"X":202,"Y":246},{"X":201,"Y":247},{"X":200,"Y":247},{"X":200,"Y":247},{"X":200,"Y":248},{"X":201,"Y":249},{"X":201,"Y":248},{"X":201,"Y":250},{"X":202,"Y":251},{"X":201,"Y":254},{"X":200,"Y":254},{"X":201,"Y":253},{"X":199,"Y":254},{"X":199,"Y":253},{"X":199,"Y":253},{"X":196,"Y":254},{"X":194,"Y":251},{"X":192,"Y":253},{"X":193,"Y":251},{"X":192,"Y":251},{"X":192,"Y":251},{"X":191,"Y":251},{"X":192,"Y":253},{"X":191,"Y":253},{"X":190,"Y":254},{"X":189,"Y":255},{"X":189,"Y":255},{"X":190,"Y":254},{"X":190,"Y":253},{"X":190,"Y":251},{"X":189,"Y":251},{"X":189,"Y":249},{"X":189,"Y":251},{"X":189,"Y":253},{"X":189,"Y":253},{"X":188,"Y":255},{"X":185,"Y":254},{"X":187,"Y":253},{"X":185,"Y":253},{"X":185,"Y":253},{"X":185,"Y":254},{"X":184,"Y":254},{"X":183,"Y":254},{"X":179,"Y":254},{"X":177,"Y":255},{"X":177,"Y":255},{"X":176,"Y":255},{"X":174,"Y":255},{"X":176,"Y":256},{"X":177,"Y":257},{"X":176,"Y":257},{"X":176,"Y":258},{"X":172,"Y":258},{"X":172,"Y":260},{"X":171,"Y":260},{"X":170,"Y":260},{"X":169,"Y":259},{"X":169,"Y":260},{"X":168,"Y":260},{"X":169,"Y":261},{"X":167,"Y":262},{"X":166,"Y":264},{"X":166,"Y":264},{"X":162,"Y":265},{"X":161,"Y":264},{"X":161,"Y":264},{"X":162,"Y":262},{"X":163,"Y":261},{"X":160,"Y":262},{"X":158,"Y":261},{"X":158,"Y":262},{"X":159,"Y":262},{"X":160,"Y":264},{"X":160,"Y":265},{"X":160,"Y":265},{"X":158,"Y":267},{"X":157,"Y":267},{"X":156,"Y":267},{"X":155,"Y":267},{"X":155,"Y":267},{"X":154,"Y":268},{"X":154,"Y":270},{"X":154,"Y":269},{"X":151,"Y":269},{"X":150,"Y":272},{"X":149,"Y":272},{"X":149,"Y":270},{"X":147,"Y":270},{"X":147,"Y":272},{"X":146,"Y":272},{"X":146,"Y":272},{"X":144,"Y":270},{"X":143,"Y":272},{"X":141,"Y":270},{"X":141,"Y":272},{"X":140,"Y":272},{"X":140,"Y":271},{"X":140,"Y":273},{"X":139,"Y":273},{"X":139,"Y":273},{"X":139,"Y":273},{"X":138,"Y":272},{"X":138,"Y":273},{"X":137,"Y":273},{"X":137,"Y":273},{"X":136,"Y":272},{"X":137,"Y":270},{"X":136,"Y":269},{"X":136,"Y":270},{"X":135,"Y":270},{"X":134,"Y":270},{"X":134,"Y":272},{"X":133,"Y":272},{"X":133,"Y":272},{"X":132,"Y":272},{"X":132,"Y":270},{"X":129,"Y":272},{"X":130,"Y":272},{"X":128,"Y":272},{"X":128,"Y":271},{"X":127,"Y":272},{"X":127,"Y":271},{"X":127,"Y":271},{"X":129,"Y":269},{"X":132,"Y":269},{"X":134,"Y":267},{"X":134,"Y":267},{"X":128,"Y":269},{"X":127,"Y":269},{"X":128,"Y":268},{"X":129,"Y":267},{"X":136,"Y":266},{"X":136,"Y":266},{"X":136,"Y":265},{"X":136,"Y":265},{"X":135,"Y":264},{"X":135,"Y":262},{"X":135,"Y":264},{"X":134,"Y":266},{"X":126,"Y":266},{"X":126,"Y":267},{"X":124,"Y":267},{"X":123,"Y":267},{"X":122,"Y":267},{"X":122,"Y":267},{"X":122,"Y":267},{"X":122,"Y":266},{"X":123,"Y":267},{"X":123,"Y":266},{"X":123,"Y":265},{"X":123,"Y":265},{"X":125,"Y":265},{"X":126,"Y":264},{"X":125,"Y":264},{"X":125,"Y":262},{"X":127,"Y":261},{"X":127,"Y":262},{"X":127,"Y":261},{"X":129,"Y":261},{"X":128,"Y":261},{"X":130,"Y":259},{"X":133,"Y":259},{"X":134,"Y":259},{"X":132,"Y":259},{"X":129,"Y":260},{"X":128,"Y":260},{"X":127,"Y":260},{"X":127,"Y":260},{"X":122,"Y":262},{"X":122,"Y":261},{"X":121,"Y":261},{"X":121,"Y":260},{"X":122,"Y":259},{"X":121,"Y":258},{"X":119,"Y":259},{"X":118,"Y":260},{"X":118,"Y":258},{"X":117,"Y":258},{"X":117,"Y":256},{"X":119,"Y":256},{"X":121,"Y":256},{"X":119,"Y":255},{"X":119,"Y":255},{"X":121,"Y":254},{"X":121,"Y":254},{"X":126,"Y":251},{"X":127,"Y":250},{"X":127,"Y":251},{"X":127,"Y":251},{"X":128,"Y":249},{"X":129,"Y":249},{"X":132,"Y":249},{"X":128,"Y":248},{"X":127,"Y":248},{"X":127,"Y":249},{"X":126,"Y":248},{"X":123,"Y":249},{"X":122,"Y":248},{"X":121,"Y":248},{"X":119,"Y":248},{"X":121,"Y":249},{"X":121,"Y":249},{"X":118,"Y":248},{"X":119,"Y":249},{"X":117,"Y":249},{"X":116,"Y":248},{"X":116,"Y":248},{"X":116,"Y":247},{"X":117,"Y":247},{"X":116,"Y":246},{"X":117,"Y":246},{"X":117,"Y":247},{"X":118,"Y":247},{"X":118,"Y":245},{"X":119,"Y":245},{"X":121,"Y":244},{"X":123,"Y":244},{"X":123,"Y":245},{"X":125,"Y":246},{"X":126,"Y":244},{"X":125,"Y":243},{"X":126,"Y":242},{"X":126,"Y":244},{"X":127,"Y":246},{"X":129,"Y":246},{"X":129,"Y":246},{"X":129,"Y":246},{"X":132,"Y":246},{"X":132,"Y":245},{"X":128,"Y":245},{"X":128,"Y":243},{"X":129,"Y":244},{"X":129,"Y":243},{"X":130,"Y":240},{"X":128,"Y":239},{"X":128,"Y":239},{"X":133,"Y":238},{"X":134,"Y":237},{"X":135,"Y":238},{"X":135,"Y":237},{"X":134,"Y":237},{"X":134,"Y":237},{"X":135,"Y":235},{"X":136,"Y":235},{"X":138,"Y":236},{"X":138,"Y":235},{"X":141,"Y":236},{"X":147,"Y":234},{"X":147,"Y":235},{"X":148,"Y":235},{"X":149,"Y":234},{"X":151,"Y":234},{"X":151,"Y":234},{"X":154,"Y":234},{"X":150,"Y":233},{"X":149,"Y":233},{"X":149,"Y":232},{"X":148,"Y":231},{"X":147,"Y":231},{"X":147,"Y":232},{"X":144,"Y":235},{"X":143,"Y":234},{"X":141,"Y":236},{"X":140,"Y":235},{"X":141,"Y":234},{"X":139,"Y":234},{"X":137,"Y":234},{"X":137,"Y":233},{"X":136,"Y":232},{"X":137,"Y":234},{"X":134,"Y":234},{"X":134,"Y":235},{"X":129,"Y":234},{"X":128,"Y":234},{"X":132,"Y":234},{"X":135,"Y":231},{"X":136,"Y":229},{"X":138,"Y":229},{"X":138,"Y":227},{"X":139,"Y":226},{"X":139,"Y":224},{"X":141,"Y":223},{"X":140,"Y":223},{"X":138,"Y":223},{"X":138,"Y":223},{"X":139,"Y":222},{"X":143,"Y":217},{"X":146,"Y":217},{"X":147,"Y":217},{"X":147,"Y":217},{"X":149,"Y":217},{"X":147,"Y":217},{"X":149,"Y":217},{"X":149,"Y":217},{"X":150,"Y":217},{"X":150,"Y":217},{"X":150,"Y":216},{"X":151,"Y":215},{"X":150,"Y":215},{"X":149,"Y":215},{"X":149,"Y":214},{"X":150,"Y":213},{"X":150,"Y":213},{"X":150,"Y":212},{"X":148,"Y":212},{"X":146,"Y":213},{"X":139,"Y":213},{"X":138,"Y":213},{"X":138,"Y":211},{"X":138,"Y":211},{"X":137,"Y":213},{"X":137,"Y":213},{"X":137,"Y":212},{"X":137,"Y":211},{"X":137,"Y":210},{"X":137,"Y":210},{"X":138,"Y":210},{"X":137,"Y":209},{"X":137,"Y":210},{"X":137,"Y":209},{"X":137,"Y":209},{"X":137,"Y":209},{"X":137,"Y":209},{"X":136,"Y":210},{"X":135,"Y":210},{"X":134,"Y":209},{"X":133,"Y":209},{"X":133,"Y":207},{"X":133,"Y":207},{"X":135,"Y":206},{"X":135,"Y":206},{"X":135,"Y":206},{"X":135,"Y":206},{"X":133,"Y":206},{"X":133,"Y":206},{"X":133,"Y":206},{"X":133,"Y":206},{"X":132,"Y":206},{"X":132,"Y":207},{"X":129,"Y":206},{"X":128,"Y":205},{"X":127,"Y":206},{"X":127,"Y":205},{"X":126,"Y":205},{"X":127,"Y":203},{"X":129,"Y":204},{"X":128,"Y":203},{"X":128,"Y":202},{"X":127,"Y":201},{"X":126,"Y":201},{"X":128,"Y":201},{"X":128,"Y":200},{"X":132,"Y":201},{"X":129,"Y":198},{"X":134,"Y":198},{"X":134,"Y":196},{"X":134,"Y":194},{"X":136,"Y":193},{"X":138,"Y":194},{"X":139,"Y":193},{"X":138,"Y":193},{"X":139,"Y":192},{"X":138,"Y":192},{"X":139,"Y":191},{"X":139,"Y":191},{"X":139,"Y":190},{"X":136,"Y":190},{"X":135,"Y":190},{"X":134,"Y":190},{"X":134,"Y":190},{"X":134,"Y":188},{"X":136,"Y":189},{"X":136,"Y":189},{"X":136,"Y":189},{"X":136,"Y":187},{"X":136,"Y":187},{"X":136,"Y":188},{"X":136,"Y":187},{"X":136,"Y":187},{"X":135,"Y":184},{"X":135,"Y":183},{"X":136,"Y":183},{"X":135,"Y":183},{"X":136,"Y":182},{"X":135,"Y":182},{"X":135,"Y":183},{"X":134,"Y":182},{"X":134,"Y":182},{"X":134,"Y":182},{"X":135,"Y":181},{"X":134,"Y":181},{"X":135,"Y":180},{"X":134,"Y":180},{"X":135,"Y":179},{"X":135,"Y":179},{"X":133,"Y":178},{"X":133,"Y":179},{"X":133,"Y":179},{"X":133,"Y":180},{"X":132,"Y":180},{"X":132,"Y":180},{"X":132,"Y":181},{"X":132,"Y":182},{"X":130,"Y":182},{"X":132,"Y":178},{"X":132,"Y":178},{"X":130,"Y":177},{"X":132,"Y":177},{"X":132,"Y":177},{"X":134,"Y":174},{"X":134,"Y":177},{"X":135,"Y":178},{"X":135,"Y":179},{"X":134,"Y":178},{"X":134,"Y":178},{"X":136,"Y":179},{"X":136,"Y":177},{"X":136,"Y":178},{"X":136,"Y":177},{"X":136,"Y":177},{"X":136,"Y":176},{"X":136,"Y":174},{"X":138,"Y":176},{"X":143,"Y":176},{"X":144,"Y":178},{"X":146,"Y":177},{"X":147,"Y":177},{"X":147,"Y":178},{"X":148,"Y":179},{"X":148,"Y":179},{"X":148,"Y":180},{"X":148,"Y":180},{"X":149,"Y":181},{"X":149,"Y":181},{"X":151,"Y":178},{"X":154,"Y":178},{"X":156,"Y":180},{"X":157,"Y":179},{"X":157,"Y":179},{"X":158,"Y":180},{"X":160,"Y":181},{"X":161,"Y":181},{"X":160,"Y":180},{"X":160,"Y":180},{"X":161,"Y":179},{"X":162,"Y":179},{"X":161,"Y":178},{"X":161,"Y":178},{"X":162,"Y":178},{"X":162,"Y":178},{"X":161,"Y":177},{"X":160,"Y":178},{"X":159,"Y":177},{"X":161,"Y":176},{"X":161,"Y":174},{"X":161,"Y":174},{"X":163,"Y":174},{"X":162,"Y":173},{"X":163,"Y":172},{"X":163,"Y":173},{"X":167,"Y":173},{"X":168,"Y":171},{"X":168,"Y":170},{"X":170,"Y":168},{"X":170,"Y":168},{"X":171,"Y":168},{"X":170,"Y":168},{"X":168,"Y":168},{"X":167,"Y":168},{"X":167,"Y":168},{"X":167,"Y":168},{"X":166,"Y":168},{"X":163,"Y":170},{"X":163,"Y":168},{"X":166,"Y":168},{"X":166,"Y":168},{"X":163,"Y":168},{"X":163,"Y":168},{"X":161,"Y":168},{"X":160,"Y":168},{"X":157,"Y":166},{"X":158,"Y":165},{"X":157,"Y":165},{"X":158,"Y":165},{"X":158,"Y":163},{"X":160,"Y":162},{"X":165,"Y":163},{"X":163,"Y":162},{"X":162,"Y":162},{"X":163,"Y":162},{"X":166,"Y":163},{"X":162,"Y":160},{"X":163,"Y":160},{"X":167,"Y":160},{"X":166,"Y":159},{"X":166,"Y":159},{"X":167,"Y":159},{"X":168,"Y":159},{"X":167,"Y":159},{"X":166,"Y":158},{"X":166,"Y":159},{"X":165,"Y":158},{"X":165,"Y":156},{"X":167,"Y":156},{"X":166,"Y":156},{"X":165,"Y":154},{"X":166,"Y":154},{"X":166,"Y":152},{"X":167,"Y":154},{"X":167,"Y":152},{"X":168,"Y":152},{"X":168,"Y":152},{"X":168,"Y":152},{"X":168,"Y":151},{"X":169,"Y":150},{"X":171,"Y":150},{"X":171,"Y":150},{"X":171,"Y":150},{"X":171,"Y":151},{"X":172,"Y":150},{"X":174,"Y":150},{"X":174,"Y":148},{"X":174,"Y":147},{"X":176,"Y":148},{"X":176,"Y":149},{"X":177,"Y":149},{"X":177,"Y":150},{"X":177,"Y":150},{"X":178,"Y":150},{"X":177,"Y":150},{"X":178,"Y":150},{"X":178,"Y":150},{"X":178,"Y":149},{"X":177,"Y":149},{"X":177,"Y":147},{"X":178,"Y":147},{"X":178,"Y":147},{"X":178,"Y":150},{"X":179,"Y":150},{"X":180,"Y":151},{"X":180,"Y":152},{"X":181,"Y":151},{"X":180,"Y":148},{"X":180,"Y":150},{"X":180,"Y":149},{"X":180,"Y":148},{"X":179,"Y":148},{"X":179,"Y":147},{"X":180,"Y":147},{"X":181,"Y":146},{"X":182,"Y":147},{"X":182,"Y":149},{"X":183,"Y":151},{"X":183,"Y":151},{"X":183,"Y":152},{"X":181,"Y":155},{"X":182,"Y":155},{"X":182,"Y":155},{"X":181,"Y":155},{"X":180,"Y":157},{"X":182,"Y":155},{"X":184,"Y":154},{"X":184,"Y":154},{"X":184,"Y":151},{"X":183,"Y":148},{"X":184,"Y":148},{"X":183,"Y":147},{"X":184,"Y":147},{"X":184,"Y":147},{"X":184,"Y":147},{"X":187,"Y":146},{"X":187,"Y":146},{"X":187,"Y":147},{"X":189,"Y":147},{"X":187,"Y":146},{"X":187,"Y":145},{"X":187,"Y":145},{"X":187,"Y":144},{"X":185,"Y":144},{"X":185,"Y":144},{"X":189,"Y":144},{"X":190,"Y":145},{"X":191,"Y":146},{"X":195,"Y":148},{"X":195,"Y":149},{"X":191,"Y":151}],[{"X":236,"Y":34},{"X":237,"Y":35},{"X":238,"Y":35},{"X":239,"Y":35},{"X":239,"Y":36},{"X":240,"Y":35},{"X":239,"Y":35},{"X":239,"Y":35},{"X":240,"Y":36},{"X":242,"Y":37},{"X":242,"Y":37},{"X":240,"Y":40},{"X":242,"Y":38},{"X":242,"Y":37},{"X":243,"Y":37},{"X":243,"Y":36},{"X":245,"Y":36},{"X":246,"Y":37},{"X":247,"Y":38},{"X":246,"Y":40},{"X":247,"Y":37},{"X":249,"Y":37},{"X":250,"Y":38},{"X":250,"Y":38},{"X":250,"Y":37},{"X":251,"Y":37},{"X":251,"Y":36},{"X":253,"Y":37},{"X":254,"Y":36},{"X":254,"Y":36},{"X":256,"Y":36},{"X":257,"Y":36},{"X":260,"Y":36},{"X":262,"Y":36},{"X":262,"Y":36},{"X":262,"Y":36},{"X":264,"Y":35},{"X":266,"Y":36},{"X":266,"Y":35},{"X":265,"Y":34},{"X":266,"Y":33},{"X":266,"Y":34},{"X":269,"Y":34},{"X":269,"Y":35},{"X":271,"Y":34},{"X":271,"Y":35},{"X":270,"Y":38},{"X":270,"Y":39},{"X":271,"Y":39},{"X":270,"Y":44},{"X":269,"Y":44},{"X":268,"Y":46},{"X":266,"Y":47},{"X":264,"Y":51},{"X":258,"Y":53},{"X":257,"Y":55},{"X":256,"Y":56},{"X":254,"Y":58},{"X":254,"Y":60},{"X":253,"Y":60},{"X":251,"Y":61},{"X":250,"Y":61},{"X":250,"Y":62},{"X":253,"Y":62},{"X":254,"Y":61},{"X":255,"Y":61},{"X":255,"Y":62},{"X":256,"Y":62},{"X":257,"Y":60},{"X":258,"Y":61},{"X":256,"Y":63},{"X":254,"Y":66},{"X":254,"Y":66},{"X":254,"Y":66},{"X":253,"Y":66},{"X":250,"Y":67},{"X":248,"Y":66},{"X":248,"Y":68},{"X":246,"Y":69},{"X":246,"Y":70},{"X":249,"Y":67},{"X":253,"Y":67},{"X":253,"Y":66},{"X":254,"Y":67},{"X":251,"Y":69},{"X":251,"Y":69},{"X":251,"Y":70},{"X":250,"Y":71},{"X":250,"Y":72},{"X":249,"Y":71},{"X":249,"Y":72},{"X":249,"Y":73},{"X":250,"Y":73},{"X":250,"Y":73},{"X":250,"Y":73},{"X":253,"Y":71},{"X":253,"Y":70},{"X":253,"Y":69},{"X":256,"Y":69},{"X":260,"Y":68},{"X":262,"Y":68},{"X":264,"Y":66},{"X":266,"Y":64},{"X":271,"Y":68},{"X":275,"Y":66},{"X":276,"Y":67},{"X":277,"Y":67},{"X":283,"Y":68},{"X":284,"Y":68},{"X":286,"Y":67},{"X":287,"Y":68},{"X":289,"Y":67},{"X":291,"Y":67},{"X":291,"Y":68},{"X":292,"Y":67},{"X":293,"Y":68},{"X":293,"Y":69},{"X":293,"Y":70},{"X":294,"Y":73},{"X":294,"Y":73},{"X":294,"Y":74},{"X":295,"Y":74},{"X":290,"Y":81},{"X":289,"Y":84},{"X":290,"Y":86},{"X":287,"Y":91},{"X":287,"Y":93},{"X":286,"Y":96},{"X":283,"Y":97},{"X":283,"Y":97},{"X":283,"Y":101},{"X":283,"Y":101},{"X":281,"Y":102},{"X":282,"Y":103},{"X":281,"Y":105},{"X":278,"Y":107},{"X":277,"Y":108},{"X":276,"Y":108},{"X":275,"Y":108},{"X":273,"Y":108},{"X":272,"Y":108},{"X":271,"Y":108},{"X":267,"Y":111},{"X":266,"Y":112},{"X":267,"Y":112},{"X":268,"Y":111},{"X":275,"Y":108},{"X":276,"Y":110},{"X":276,"Y":111},{"X":276,"Y":112},{"X":276,"Y":112},{"X":277,"Y":113},{"X":279,"Y":113},{"X":280,"Y":116},{"X":278,"Y":117},{"X":276,"Y":117},{"X":273,"Y":117},{"X":272,"Y":117},{"X":269,"Y":121},{"X":269,"Y":122},{"X":267,"Y":123},{"X":265,"Y":124},{"X":261,"Y":123},{"X":258,"Y":123},{"X":258,"Y":124},{"X":259,"Y":124},{"X":262,"Y":124},{"X":265,"Y":125},{"X":266,"Y":124},{"X":267,"Y":125},{"X":269,"Y":125},{"X":270,"Y":126},{"X":271,"Y":126},{"X":272,"Y":125},{"X":273,"Y":125},{"X":275,"Y":124},{"X":275,"Y":123},{"X":276,"Y":122},{"X":279,"Y":122},{"X":280,"Y":124},{"X":280,"Y":124},{"X":282,"Y":124},{"X":286,"Y":127},{"X":289,"Y":127},{"X":289,"Y":127},{"X":290,"Y":129},{"X":292,"Y":133},{"X":294,"Y":136},{"X":295,"Y":136},{"X":295,"Y":137},{"X":295,"Y":137},{"X":295,"Y":138},{"X":298,"Y":138},{"X":298,"Y":138},{"X":299,"Y":140},{"X":300,"Y":141},{"X":300,"Y":144},{"X":299,"Y":146},{"X":300,"Y":147},{"X":300,"Y":149},{"X":301,"Y":152},{"X":301,"Y":154},{"X":303,"Y":158},{"X":304,"Y":160},{"X":304,"Y":161},{"X":306,"Y":166},{"X":306,"Y":168},{"X":309,"Y":169},{"X":308,"Y":169},{"X":309,"Y":170},{"X":309,"Y":172},{"X":309,"Y":172},{"X":309,"Y":172},{"X":309,"Y":171},{"X":312,"Y":172},{"X":315,"Y":173},{"X":319,"Y":176},{"X":321,"Y":176},{"X":322,"Y":177},{"X":322,"Y":178},{"X":323,"Y":180},{"X":325,"Y":182},{"X":326,"Y":183},{"X":327,"Y":187},{"X":331,"Y":188},{"X":328,"Y":189},{"X":327,"Y":191},{"X":330,"Y":195},{"X":333,"Y":200},{"X":335,"Y":204},{"X":335,"Y":206},{"X":335,"Y":206},{"X":335,"Y":205},{"X":335,"Y":204},{"X":333,"Y":204},{"X":333,"Y":204},{"X":332,"Y":204},{"X":330,"Y":204},{"X":327,"Y":201},{"X":326,"Y":201},{"X":324,"Y":202},{"X":322,"Y":203},{"X":320,"Y":202},{"X":319,"Y":203},{"X":319,"Y":203},{"X":321,"Y":203},{"X":322,"Y":203},{"X":323,"Y":203},{"X":326,"Y":202},{"X":327,"Y":203},{"X":328,"Y":204},{"X":332,"Y":206},{"X":333,"Y":206},{"X":333,"Y":207},{"X":334,"Y":210},{"X":336,"Y":210},{"X":336,"Y":211},{"X":337,"Y":212},{"X":339,"Y":217},{"X":341,"Y":220},{"X":339,"Y":222},{"X":336,"Y":225},{"X":335,"Y":229},{"X":334,"Y":229},{"X":334,"Y":229},{"X":335,"Y":229},{"X":336,"Y":229},{"X":336,"Y":231},{"X":338,"Y":233},{"X":341,"Y":232},{"X":342,"Y":234},{"X":342,"Y":233},{"X":342,"Y":232},{"X":343,"Y":227},{"X":344,"Y":226},{"X":346,"Y":226},{"X":347,"Y":226},{"X":350,"Y":226},{"X":352,"Y":226},{"X":352,"Y":226},{"X":353,"Y":226},{"X":354,"Y":226},{"X":353,"Y":225},{"X":354,"Y":225},{"X":355,"Y":225},{"X":356,"Y":226},{"X":360,"Y":226},{"X":363,"Y":229},{"X":368,"Y":233},{"X":368,"Y":233},{"X":370,"Y":236},{"X":370,"Y":238},{"X":370,"Y":240},{"X":370,"Y":243},{"X":370,"Y":245},{"X":368,"Y":249},{"X":368,"Y":251},{"X":368,"Y":255},{"X":365,"Y":257},{"X":364,"Y":261},{"X":361,"Y":259},{"X":361,"Y":260},{"X":359,"Y":260},{"X":358,"Y":260},{"X":363,"Y":260},{"X":360,"Y":264},{"X":361,"Y":264},{"X":361,"Y":262},{"X":363,"Y":264},{"X":363,"Y":264},{"X":360,"Y":266},{"X":358,"Y":267},{"X":357,"Y":266},{"X":356,"Y":266},{"X":355,"Y":266},{"X":354,"Y":267},{"X":353,"Y":268},{"X":350,"Y":268},{"X":352,"Y":269},{"X":352,"Y":269},{"X":354,"Y":267},{"X":355,"Y":268},{"X":355,"Y":269},{"X":355,"Y":272},{"X":354,"Y":272},{"X":353,"Y":272},{"X":354,"Y":272},{"X":354,"Y":272},{"X":354,"Y":272},{"X":355,"Y":272},{"X":356,"Y":272},{"X":353,"Y":276},{"X":349,"Y":276},{"X":348,"Y":277},{"X":346,"Y":277},{"X":345,"Y":277},{"X":344,"Y":278},{"X":345,"Y":278},{"X":346,"Y":277},{"X":347,"Y":277},{"X":350,"Y":277},{"X":352,"Y":278},{"X":350,"Y":279},{"X":349,"Y":279},{"X":348,"Y":279},{"X":347,"Y":279},{"X":348,"Y":280},{"X":349,"Y":280},{"X":350,"Y":280},{"X":350,"Y":280},{"X":350,"Y":279},{"X":352,"Y":280},{"X":352,"Y":279},{"X":352,"Y":278},{"X":355,"Y":279},{"X":356,"Y":281},{"X":355,"Y":281},{"X":353,"Y":281},{"X":355,"Y":281},{"X":361,"Y":280},{"X":367,"Y":279},{"X":366,"Y":281},{"X":365,"Y":282},{"X":366,"Y":284},{"X":366,"Y":288},{"X":365,"Y":289},{"X":363,"Y":290},{"X":361,"Y":290},{"X":360,"Y":291},{"X":358,"Y":292},{"X":357,"Y":293},{"X":357,"Y":295},{"X":354,"Y":295},{"X":352,"Y":297},{"X":350,"Y":298},{"X":344,"Y":300},{"X":343,"Y":302},{"X":342,"Y":303},{"X":337,"Y":302},{"X":333,"Y":300},{"X":322,"Y":302},{"X":322,"Y":303},{"X":321,"Y":302},{"X":321,"Y":304},{"X":320,"Y":304},{"X":317,"Y":302},{"X":319,"Y":302},{"X":317,"Y":301},{"X":317,"Y":301},{"X":316,"Y":301},{"X":315,"Y":301},{"X":316,"Y":301},{"X":317,"Y":302},{"X":314,"Y":302},{"X":315,"Y":302},{"X":315,"Y":301},{"X":314,"Y":301},{"X":314,"Y":303},{"X":313,"Y":303},{"X":313,"Y":303},{"X":313,"Y":301},{"X":313,"Y":300},{"X":311,"Y":301},{"X":312,"Y":302},{"X":312,"Y":303},{"X":309,"Y":301},{"X":308,"Y":299},{"X":305,"Y":298},{"X":305,"Y":299},{"X":306,"Y":299},{"X":309,"Y":302},{"X":308,"Y":303},{"X":305,"Y":303},{"X":304,"Y":304},{"X":301,"Y":304},{"X":299,"Y":304},{"X":300,"Y":304},{"X":300,"Y":304},{"X":298,"Y":304},{"X":297,"Y":304},{"X":295,"Y":306},{"X":295,"Y":304},{"X":293,"Y":304},{"X":292,"Y":306},{"X":293,"Y":305},{"X":293,"Y":306},{"X":295,"Y":306},{"X":295,"Y":308},{"X":295,"Y":308},{"X":295,"Y":309},{"X":294,"Y":310},{"X":293,"Y":310},{"X":292,"Y":310},{"X":290,"Y":309},{"X":284,"Y":309},{"X":284,"Y":309},{"X":283,"Y":310},{"X":284,"Y":311},{"X":283,"Y":312},{"X":283,"Y":311},{"X":280,"Y":308},{"X":277,"Y":305},{"X":275,"Y":305},{"X":272,"Y":306},{"X":270,"Y":306},{"X":268,"Y":306},{"X":265,"Y":310},{"X":264,"Y":309},{"X":262,"Y":306},{"X":262,"Y":310},{"X":261,"Y":312},{"X":261,"Y":314},{"X":260,"Y":315},{"X":260,"Y":316},{"X":261,"Y":317},{"X":261,"Y":317},{"X":260,"Y":317},{"X":259,"Y":319},{"X":259,"Y":319},{"X":258,"Y":320},{"X":258,"Y":321},{"X":258,"Y":322},{"X":256,"Y":323},{"X":255,"Y":322},{"X":255,"Y":322},{"X":254,"Y":322},{"X":253,"Y":320},{"X":251,"Y":319},{"X":249,"Y":319},{"X":247,"Y":319},{"X":247,"Y":317},{"X":246,"Y":317},{"X":246,"Y":317},{"X":247,"Y":317},{"X":246,"Y":319},{"X":245,"Y":319},{"X":244,"Y":317},{"X":243,"Y":317},{"X":240,"Y":317},{"X":239,"Y":319},{"X":236,"Y":319},{"X":235,"Y":319},{"X":235,"Y":317},{"X":234,"Y":319},{"X":233,"Y":322},{"X":231,"Y":321},{"X":229,"Y":322},{"X":228,"Y":322},{"X":228,"Y":324},{"X":227,"Y":324},{"X":228,"Y":324},{"X":227,"Y":324},{"X":227,"Y":321},{"X":227,"Y":321},{"X":226,"Y":322},{"X":227,"Y":323},{"X":226,"Y":324},{"X":227,"Y":324},{"X":226,"Y":324},{"X":226,"Y":325},{"X":225,"Y":325},{"X":226,"Y":326},{"X":226,"Y":327},{"X":226,"Y":328},{"X":224,"Y":328},{"X":224,"Y":331},{"X":223,"Y":328},{"X":222,"Y":326},{"X":221,"Y":326},{"X":217,"Y":324},{"X":216,"Y":324},{"X":216,"Y":325},{"X":215,"Y":326},{"X":213,"Y":327},{"X":212,"Y":326},{"X":213,"Y":325},{"X":212,"Y":323},{"X":213,"Y":324},{"X":216,"Y":321},{"X":218,"Y":322},{"X":220,"Y":321},{"X":221,"Y":321},{"X":222,"Y":320},{"X":223,"Y":317},{"X":225,"Y":316},{"X":225,"Y":315},{"X":227,"Y":315},{"X":228,"Y":313},{"X":227,"Y":312},{"X":228,"Y":311},{"X":228,"Y":311},{"X":229,"Y":310},{"X":229,"Y":310},{"X":231,"Y":309},{"X":234,"Y":309},{"X":234,"Y":306},{"X":235,"Y":305},{"X":236,"Y":304},{"X":238,"Y":303},{"X":238,"Y":299},{"X":239,"Y":295},{"X":240,"Y":295},{"X":243,"Y":295},{"X":244,"Y":295},{"X":246,"Y":294},{"X":247,"Y":293},{"X":245,"Y":291},{"X":246,"Y":291},{"X":246,"Y":290},{"X":246,"Y":290},{"X":255,"Y":288},{"X":259,"Y":288},{"X":260,"Y":288},{"X":262,"Y":289},{"X":264,"Y":290},{"X":266,"Y":290},{"X":268,"Y":289},{"X":270,"Y":289},{"X":271,"Y":288},{"X":271,"Y":288},{"X":271,"Y":286},{"X":272,"Y":286},{"X":272,"Y":282},{"X":275,"Y":281},{"X":277,"Y":279},{"X":278,"Y":279},{"X":283,"Y":271},{"X":284,"Y":270},{"X":284,"Y":269},{"X":284,"Y":270},{"X":283,"Y":270},{"X":278,"Y":276},{"X":275,"Y":277},{"X":272,"Y":278},{"X":272,"Y":278},{"X":272,"Y":278},{"X":271,"Y":278},{"X":269,"Y":279},{"X":269,"Y":280},{"X":268,"Y":280},{"X":268,"Y":281},{"X":267,"Y":282},{"X":264,"Y":282},{"X":260,"Y":281},{"X":258,"Y":279},{"X":257,"Y":279},{"X":255,"Y":278},{"X":254,"Y":275},{"X":250,"Y":275},{"X":250,"Y":275},{"X":250,"Y":277},{"X":247,"Y":277},{"X":247,"Y":277},{"X":246,"Y":278},{"X":244,"Y":277},{"X":244,"Y":276},{"X":245,"Y":275},{"X":246,"Y":275},{"X":246,"Y":275},{"X":248,"Y":275},{"X":249,"Y":273},{"X":249,"Y":272},{"X":248,"Y":273},{"X":247,"Y":272},{"X":244,"Y":272},{"X":244,"Y":272},{"X":243,"Y":271},{"X":243,"Y":270},{"X":244,"Y":271},{"X":243,"Y":269},{"X":243,"Y":268},{"X":243,"Y":269},{"X":242,"Y":269},{"X":242,"Y":270},{"X":237,"Y":270},{"X":236,"Y":272},{"X":235,"Y":272},{"X":234,"Y":273},{"X":233,"Y":275},{"X":231,"Y":275},{"X":228,"Y":273},{"X":228,"Y":272},{"X":227,"Y":272},{"X":227,"Y":272},{"X":228,"Y":272},{"X":228,"Y":272},{"X":228,"Y":272},{"X":229,"Y":272},{"X":233,"Y":271},{"X":229,"Y":270},{"X":226,"Y":270},{"X":226,"Y":272},{"X":225,"Y":270},{"X":224,"Y":269},{"X":225,"Y":269},{"X":226,"Y":268},{"X":227,"Y":268},{"X":227,"Y":266},{"X":225,"Y":266},{"X":223,"Y":265},{"X":223,"Y":262},{"X":225,"Y":264},{"X":225,"Y":262},{"X":227,"Y":261},{"X":227,"Y":260},{"X":229,"Y":261},{"X":231,"Y":261},{"X":232,"Y":261},{"X":232,"Y":260},{"X":233,"Y":261},{"X":234,"Y":260},{"X":234,"Y":259},{"X":235,"Y":259},{"X":236,"Y":257},{"X":236,"Y":257},{"X":236,"Y":256},{"X":240,"Y":256},{"X":243,"Y":254},{"X":244,"Y":254},{"X":246,"Y":253},{"X":248,"Y":250},{"X":249,"Y":248},{"X":250,"Y":244},{"X":251,"Y":243},{"X":253,"Y":242},{"X":250,"Y":242},{"X":248,"Y":240},{"X":250,"Y":237},{"X":251,"Y":235},{"X":250,"Y":236},{"X":249,"Y":235},{"X":248,"Y":235},{"X":248,"Y":234},{"X":249,"Y":232},{"X":248,"Y":231},{"X":243,"Y":231},{"X":242,"Y":232},{"X":242,"Y":233},{"X":242,"Y":234},{"X":240,"Y":235},{"X":238,"Y":233},{"X":238,"Y":234},{"X":236,"Y":233},{"X":236,"Y":234},{"X":236,"Y":233},{"X":236,"Y":232},{"X":238,"Y":231},{"X":240,"Y":229},{"X":244,"Y":226},{"X":244,"Y":223},{"X":245,"Y":223},{"X":245,"Y":223},{"X":248,"Y":220},{"X":249,"Y":218},{"X":250,"Y":218},{"X":254,"Y":217},{"X":255,"Y":217},{"X":254,"Y":216},{"X":256,"Y":216},{"X":257,"Y":217},{"X":259,"Y":217},{"X":265,"Y":215},{"X":266,"Y":215},{"X":266,"Y":216},{"X":270,"Y":220},{"X":270,"Y":217},{"X":268,"Y":214},{"X":268,"Y":214},{"X":271,"Y":212},{"X":271,"Y":213},{"X":273,"Y":216},{"X":276,"Y":217},{"X":277,"Y":217},{"X":278,"Y":216},{"X":277,"Y":216},{"X":275,"Y":216},{"X":272,"Y":213},{"X":270,"Y":210},{"X":272,"Y":204},{"X":276,"Y":202},{"X":271,"Y":202},{"X":270,"Y":199},{"X":271,"Y":195},{"X":275,"Y":194},{"X":275,"Y":195},{"X":276,"Y":193},{"X":275,"Y":193},{"X":275,"Y":193},{"X":275,"Y":193},{"X":273,"Y":192},{"X":275,"Y":190},{"X":276,"Y":190},{"X":276,"Y":189},{"X":276,"Y":187},{"X":276,"Y":185},{"X":275,"Y":185},{"X":273,"Y":188},{"X":272,"Y":188},{"X":271,"Y":185},{"X":270,"Y":189},{"X":269,"Y":191},{"X":269,"Y":191},{"X":267,"Y":189},{"X":267,"Y":188},{"X":267,"Y":188},{"X":268,"Y":188},{"X":268,"Y":183},{"X":267,"Y":187},{"X":266,"Y":187},{"X":264,"Y":183},{"X":264,"Y":181},{"X":259,"Y":176},{"X":260,"Y":172},{"X":261,"Y":170},{"X":262,"Y":168},{"X":264,"Y":167},{"X":264,"Y":166},{"X":265,"Y":163},{"X":266,"Y":162},{"X":266,"Y":163},{"X":267,"Y":163},{"X":267,"Y":162},{"X":267,"Y":162},{"X":267,"Y":161},{"X":268,"Y":160},{"X":270,"Y":161},{"X":271,"Y":161},{"X":271,"Y":160},{"X":270,"Y":160},{"X":270,"Y":160},{"X":271,"Y":160},{"X":271,"Y":159},{"X":269,"Y":159},{"X":267,"Y":160},{"X":264,"Y":160},{"X":264,"Y":160},{"X":262,"Y":160},{"X":261,"Y":160},{"X":261,"Y":160},{"X":261,"Y":160},{"X":261,"Y":163},{"X":260,"Y":163},{"X":258,"Y":162},{"X":257,"Y":163},{"X":256,"Y":162},{"X":256,"Y":163},{"X":256,"Y":162},{"X":256,"Y":162},{"X":255,"Y":163},{"X":256,"Y":165},{"X":253,"Y":167},{"X":251,"Y":167},{"X":251,"Y":165},{"X":250,"Y":166},{"X":249,"Y":166},{"X":248,"Y":165},{"X":248,"Y":162},{"X":247,"Y":165},{"X":246,"Y":162},{"X":245,"Y":162},{"X":244,"Y":161},{"X":244,"Y":162},{"X":244,"Y":162},{"X":244,"Y":165},{"X":246,"Y":165},{"X":245,"Y":166},{"X":246,"Y":169},{"X":245,"Y":169},{"X":242,"Y":169},{"X":240,"Y":166},{"X":237,"Y":165},{"X":236,"Y":162},{"X":236,"Y":162},{"X":234,"Y":163},{"X":233,"Y":165},{"X":234,"Y":169},{"X":235,"Y":169},{"X":235,"Y":169},{"X":236,"Y":170},{"X":235,"Y":170},{"X":234,"Y":169},{"X":234,"Y":168},{"X":229,"Y":162},{"X":228,"Y":160},{"X":229,"Y":158},{"X":231,"Y":158},{"X":232,"Y":158},{"X":232,"Y":160},{"X":233,"Y":162},{"X":233,"Y":161},{"X":232,"Y":158},{"X":232,"Y":157},{"X":233,"Y":154},{"X":236,"Y":150},{"X":236,"Y":147},{"X":237,"Y":146},{"X":238,"Y":144},{"X":240,"Y":144},{"X":240,"Y":141},{"X":239,"Y":140},{"X":239,"Y":140},{"X":239,"Y":139},{"X":239,"Y":138},{"X":237,"Y":137},{"X":237,"Y":136},{"X":235,"Y":134},{"X":236,"Y":133},{"X":236,"Y":130},{"X":235,"Y":127},{"X":236,"Y":126},{"X":237,"Y":125},{"X":238,"Y":127},{"X":243,"Y":127},{"X":240,"Y":125},{"X":238,"Y":124},{"X":237,"Y":122},{"X":237,"Y":124},{"X":238,"Y":124},{"X":237,"Y":124},{"X":236,"Y":124},{"X":236,"Y":122},{"X":237,"Y":121},{"X":236,"Y":121},{"X":236,"Y":119},{"X":236,"Y":119},{"X":235,"Y":121},{"X":236,"Y":122},{"X":236,"Y":124},{"X":235,"Y":124},{"X":235,"Y":124},{"X":234,"Y":128},{"X":233,"Y":128},{"X":233,"Y":126},{"X":232,"Y":124},{"X":232,"Y":127},{"X":229,"Y":126},{"X":229,"Y":124},{"X":229,"Y":124},{"X":229,"Y":127},{"X":228,"Y":127},{"X":229,"Y":129},{"X":227,"Y":128},{"X":226,"Y":127},{"X":227,"Y":124},{"X":228,"Y":122},{"X":229,"Y":121},{"X":229,"Y":119},{"X":233,"Y":118},{"X":233,"Y":117},{"X":235,"Y":115},{"X":234,"Y":116},{"X":233,"Y":116},{"X":232,"Y":118},{"X":228,"Y":119},{"X":226,"Y":124},{"X":225,"Y":124},{"X":225,"Y":124},{"X":225,"Y":125},{"X":225,"Y":126},{"X":225,"Y":128},{"X":227,"Y":130},{"X":227,"Y":132},{"X":226,"Y":133},{"X":225,"Y":134},{"X":224,"Y":136},{"X":224,"Y":137},{"X":224,"Y":138},{"X":224,"Y":138},{"X":224,"Y":139},{"X":222,"Y":144},{"X":222,"Y":144},{"X":222,"Y":144},{"X":223,"Y":145},{"X":222,"Y":147},{"X":220,"Y":148},{"X":217,"Y":147},{"X":217,"Y":144},{"X":220,"Y":144},{"X":220,"Y":139},{"X":220,"Y":138},{"X":221,"Y":136},{"X":221,"Y":135},{"X":224,"Y":129},{"X":222,"Y":132},{"X":221,"Y":130},{"X":221,"Y":129},{"X":223,"Y":126},{"X":221,"Y":127},{"X":221,"Y":126},{"X":223,"Y":122},{"X":222,"Y":123},{"X":221,"Y":126},{"X":220,"Y":126},{"X":221,"Y":124},{"X":222,"Y":123},{"X":222,"Y":122},{"X":223,"Y":121},{"X":224,"Y":121},{"X":223,"Y":119},{"X":224,"Y":117},{"X":223,"Y":119},{"X":222,"Y":119},{"X":223,"Y":117},{"X":223,"Y":115},{"X":223,"Y":112},{"X":224,"Y":112},{"X":225,"Y":108},{"X":229,"Y":108},{"X":227,"Y":107},{"X":226,"Y":107},{"X":226,"Y":107},{"X":226,"Y":107},{"X":225,"Y":107},{"X":226,"Y":106},{"X":227,"Y":106},{"X":228,"Y":105},{"X":229,"Y":105},{"X":227,"Y":105},{"X":226,"Y":105},{"X":227,"Y":102},{"X":228,"Y":102},{"X":228,"Y":102},{"X":229,"Y":101},{"X":232,"Y":101},{"X":233,"Y":101},{"X":232,"Y":101},{"X":229,"Y":101},{"X":233,"Y":95},{"X":231,"Y":97},{"X":229,"Y":97},{"X":228,"Y":100},{"X":227,"Y":101},{"X":227,"Y":101},{"X":225,"Y":103},{"X":224,"Y":105},{"X":221,"Y":106},{"X":221,"Y":106},{"X":220,"Y":105},{"X":220,"Y":105},{"X":217,"Y":104},{"X":215,"Y":103},{"X":215,"Y":102},{"X":216,"Y":101},{"X":217,"Y":101},{"X":221,"Y":100},{"X":222,"Y":101},{"X":224,"Y":101},{"X":222,"Y":101},{"X":220,"Y":100},{"X":218,"Y":101},{"X":217,"Y":100},{"X":212,"Y":100},{"X":211,"Y":99},{"X":213,"Y":97},{"X":215,"Y":97},{"X":216,"Y":96},{"X":217,"Y":97},{"X":218,"Y":97},{"X":218,"Y":96},{"X":218,"Y":97},{"X":220,"Y":96},{"X":220,"Y":96},{"X":218,"Y":95},{"X":218,"Y":94},{"X":221,"Y":94},{"X":221,"Y":94},{"X":222,"Y":94},{"X":222,"Y":94},{"X":221,"Y":94},{"X":221,"Y":94},{"X":221,"Y":93},{"X":217,"Y":92},{"X":218,"Y":93},{"X":220,"Y":93},{"X":218,"Y":92},{"X":220,"Y":90},{"X":221,"Y":89},{"X":221,"Y":90},{"X":222,"Y":91},{"X":225,"Y":90},{"X":223,"Y":90},{"X":222,"Y":90},{"X":222,"Y":89},{"X":222,"Y":88},{"X":221,"Y":88},{"X":221,"Y":86},{"X":222,"Y":85},{"X":223,"Y":85},{"X":225,"Y":88},{"X":226,"Y":86},{"X":224,"Y":86},{"X":224,"Y":85},{"X":223,"Y":83},{"X":222,"Y":83},{"X":223,"Y":83},{"X":223,"Y":82},{"X":223,"Y":81},{"X":223,"Y":80},{"X":225,"Y":80},{"X":225,"Y":80},{"X":227,"Y":82},{"X":227,"Y":81},{"X":226,"Y":81},{"X":225,"Y":80},{"X":222,"Y":80},{"X":222,"Y":80},{"X":223,"Y":78},{"X":225,"Y":77},{"X":226,"Y":74},{"X":225,"Y":77},{"X":224,"Y":78},{"X":223,"Y":77},{"X":224,"Y":77},{"X":224,"Y":75},{"X":221,"Y":78},{"X":220,"Y":78},{"X":220,"Y":75},{"X":221,"Y":75},{"X":220,"Y":75},{"X":221,"Y":74},{"X":220,"Y":74},{"X":220,"Y":72},{"X":221,"Y":70},{"X":221,"Y":70},{"X":222,"Y":71},{"X":222,"Y":70},{"X":223,"Y":72},{"X":223,"Y":72},{"X":224,"Y":72},{"X":225,"Y":71},{"X":224,"Y":71},{"X":223,"Y":70},{"X":223,"Y":70},{"X":223,"Y":70},{"X":222,"Y":69},{"X":222,"Y":68},{"X":221,"Y":68},{"X":221,"Y":64},{"X":223,"Y":64},{"X":223,"Y":64},{"X":222,"Y":64},{"X":221,"Y":63},{"X":221,"Y":62},{"X":221,"Y":60},{"X":223,"Y":60},{"X":223,"Y":60},{"X":223,"Y":62},{"X":224,"Y":63},{"X":224,"Y":62},{"X":225,"Y":62},{"X":225,"Y":60},{"X":224,"Y":60},{"X":224,"Y":58},{"X":225,"Y":58},{"X":225,"Y":58},{"X":225,"Y":60},{"X":227,"Y":60},{"X":227,"Y":58},{"X":232,"Y":60},{"X":232,"Y":60},{"X":228,"Y":58},{"X":228,"Y":57},{"X":228,"Y":57},{"X":229,"Y":59},{"X":232,"Y":58},{"X":233,"Y":60},{"X":234,"Y":61},{"X":234,"Y":60},{"X":232,"Y":58},{"X":233,"Y":58},{"X":233,"Y":58},{"X":231,"Y":56},{"X":229,"Y":55},{"X":228,"Y":53},{"X":228,"Y":53},{"X":227,"Y":52},{"X":227,"Y":51},{"X":229,"Y":52},{"X":229,"Y":52},{"X":232,"Y":52},{"X":232,"Y":52},{"X":231,"Y":51},{"X":232,"Y":51},{"X":231,"Y":50},{"X":232,"Y":50},{"X":231,"Y":50},{"X":231,"Y":50},{"X":229,"Y":49},{"X":229,"Y":49},{"X":228,"Y":47},{"X":228,"Y":46},{"X":231,"Y":48},{"X":233,"Y":47},{"X":233,"Y":48},{"X":234,"Y":47},{"X":237,"Y":48},{"X":236,"Y":47},{"X":237,"Y":47},{"X":235,"Y":47},{"X":234,"Y":46},{"X":234,"Y":45},{"X":234,"Y":45},{"X":233,"Y":45},{"X":234,"Y":41},{"X":236,"Y":42},{"X":235,"Y":41},{"X":235,"Y":41},{"X":235,"Y":41},{"X":235,"Y":40},{"X":236,"Y":40},{"X":237,"Y":40},{"X":236,"Y":39},{"X":234,"Y":38},{"X":234,"Y":38},{"X":236,"Y":37}]]';
var arrow_poly = '[[{"X":246,"Y":146},{"X":313,"Y":146},{"X":290,"Y":168},{"X":335,"Y":191},{"X":290,"Y":235},{"X":268,"Y":191},{"X":246,"Y":213}],[{"X":246,"Y":57},{"X":268,"Y":79},{"X":290,"Y":35},{"X":335,"Y":79},{"X":290,"Y":102},{"X":313,"Y":124},{"X":246,"Y":124}],[{"X":224,"Y":146},{"X":224,"Y":213},{"X":201,"Y":191},{"X":179,"Y":235},{"X":134,"Y":191},{"X":179,"Y":168},{"X":157,"Y":146}],[{"X":179,"Y":102},{"X":134,"Y":79},{"X":179,"Y":35},{"X":201,"Y":79},{"X":224,"Y":57},{"X":224,"Y":124},{"X":157,"Y":124}]]';
return {
"ss": deserialize_clipper_poly(gb_poly),
"cc": deserialize_clipper_poly(arrow_poly)
}
}
function get_text_polys() {
var text_polygon = '[[{"X":"28.18","Y":"205.95"},{"X":"28.18","Y":"105.07"},{"X":"48.27","Y":"105.07"},{"X":"48.28","Y":"110.23"},{"X":"54.15","Y":"107.22"},{"X":"59.69","Y":"105.39"},{"X":"65.04","Y":"104.66"},{"X":"72.07","Y":"105.08"},{"X":"78.09","Y":"106.31"},{"X":"81.57","Y":"107.68"},{"X":"84.84","Y":"110.05"},{"X":"88.03","Y":"113.63"},{"X":"90.87","Y":"118.22"},{"X":"92.71","Y":"123.12"},{"X":"94.06","Y":"129.73"},{"X":"94.70","Y":"138.25"},{"X":"94.41","Y":"147.24"},{"X":"93.31","Y":"155.42"},{"X":"91.66","Y":"161.49"},{"X":"89.46","Y":"166.25"},{"X":"86.67","Y":"170.32"},{"X":"83.75","Y":"173.13"},{"X":"80.65","Y":"174.92"},{"X":"75.51","Y":"176.56"},{"X":"69.28","Y":"177.50"},{"X":"63.74","Y":"177.55"},{"X":"57.05","Y":"176.72"},{"X":"51.42","Y":"175.15"},{"X":"48.27","Y":"173.79"},{"X":"48.27","Y":"205.95"},{"X":"48.27","Y":"205.95"},{"X":"28.18","Y":"205.95"}],[{"X":"48.28","Y":"120.42"},{"X":"48.38","Y":"164.32"},{"X":"52.31","Y":"166.53"},{"X":"56.14","Y":"167.74"},{"X":"61.35","Y":"168.35"},{"X":"65.96","Y":"167.83"},{"X":"69.49","Y":"166.59"},{"X":"72.20","Y":"164.73"},{"X":"74.29","Y":"162.19"},{"X":"76.19","Y":"157.95"},{"X":"77.71","Y":"152.00"},{"X":"78.63","Y":"144.11"},{"X":"78.66","Y":"136.61"},{"X":"77.89","Y":"129.16"},{"X":"76.59","Y":"124.08"},{"X":"74.84","Y":"120.60"},{"X":"72.55","Y":"118.04"},{"X":"69.77","Y":"116.28"},{"X":"65.91","Y":"115.07"},{"X":"60.69","Y":"114.57"},{"X":"56.85","Y":"115.18"},{"X":"53.10","Y":"116.68"},{"X":"48.91","Y":"119.57"},{"X":"48.28","Y":"120.39"},{"X":"48.28","Y":"120.42"}],[{"X":"132.15","Y":"104.65"},{"X":"139.94","Y":"105.10"},{"X":"146.11","Y":"106.30"},{"X":"151.03","Y":"108.11"},{"X":"155.15","Y":"110.57"},{"X":"158.60","Y":"113.67"},{"X":"161.35","Y":"117.41"},{"X":"163.45","Y":"121.89"},{"X":"165.10","Y":"127.80"},{"X":"166.07","Y":"135.09"},{"X":"166.22","Y":"143.93"},{"X":"165.51","Y":"151.69"},{"X":"164.06","Y":"158.12"},{"X":"162.01","Y":"163.28"},{"X":"159.40","Y":"167.47"},{"X":"156.32","Y":"170.74"},{"X":"152.52","Y":"173.40"},{"X":"147.87","Y":"175.49"},{"X":"142.07","Y":"176.99"},{"X":"134.92","Y":"177.77"},{"X":"127.42","Y":"177.67"},{"X":"120.74","Y":"176.71"},{"X":"115.29","Y":"175.07"},{"X":"110.74","Y":"172.78"},{"X":"106.95","Y":"169.86"},{"X":"103.92","Y":"166.34"},{"X":"101.48","Y":"162.04"},{"X":"99.57","Y":"156.68"},{"X":"98.29","Y":"150.05"},{"X":"97.78","Y":"141.98"},{"X":"98.18","Y":"134.19"},{"X":"99.36","Y":"127.55"},{"X":"101.22","Y":"122.01"},{"X":"103.68","Y":"117.38"},{"X":"106.77","Y":"113.52"},{"X":"110.41","Y":"110.41"},{"X":"114.84","Y":"107.89"},{"X":"120.07","Y":"106.06"},{"X":"126.27","Y":"104.95"},{"X":"132.11","Y":"104.65"},{"X":"132.15","Y":"104.65"}],[{"X":"117.86","Y":"140.93"},{"X":"118.75","Y":"153.65"},{"X":"119.63","Y":"158.05"},{"X":"121.31","Y":"161.73"},{"X":"123.57","Y":"164.71"},{"X":"126.01","Y":"166.43"},{"X":"129.14","Y":"167.44"},{"X":"133.32","Y":"167.70"},{"X":"137.17","Y":"167.11"},{"X":"140.03","Y":"165.83"},{"X":"142.14","Y":"163.92"},{"X":"143.62","Y":"161.33"},{"X":"145.05","Y":"155.97"},{"X":"145.97","Y":"148.23"},{"X":"146.12","Y":"137.89"},{"X":"145.44","Y":"129.22"},{"X":"144.13","Y":"122.91"},{"X":"142.80","Y":"119.58"},{"X":"140.91","Y":"117.26"},{"X":"138.39","Y":"115.66"},{"X":"135.07","Y":"114.75"},{"X":"131.03","Y":"114.64"},{"X":"127.21","Y":"115.38"},{"X":"124.25","Y":"116.82"},{"X":"122.07","Y":"118.86"},{"X":"120.53","Y":"121.59"},{"X":"119.03","Y":"126.93"},{"X":"118.09","Y":"134.21"},{"X":"117.86","Y":"140.81"},{"X":"117.86","Y":"140.93"}],[{"X":"172.03","Y":"177.00"},{"X":"172.03","Y":"76.33"},{"X":"191.91","Y":"76.33"},{"X":"191.91","Y":"177.00"},{"X":"191.91","Y":"177.00"},{"X":"172.03","Y":"177.00"}],[{"X":"224.60","Y":"196.88"},{"X":"221.39","Y":"198.11"},{"X":"215.61","Y":"199.00"},{"X":"208.60","Y":"199.21"},{"X":"204.09","Y":"190.40"},{"X":"211.28","Y":"190.55"},{"X":"214.82","Y":"189.98"},{"X":"217.84","Y":"188.66"},{"X":"220.35","Y":"186.54"},{"X":"222.45","Y":"183.44"},{"X":"224.33","Y":"178.55"},{"X":"225.80","Y":"173.86"},{"X":"195.43","Y":"105.07"},{"X":"215.95","Y":"105.07"},{"X":"235.40","Y":"149.04"},{"X":"252.44","Y":"105.07"},{"X":"265.17","Y":"105.14"},{"X":"233.73","Y":"186.25"},{"X":"231.07","Y":"190.77"},{"X":"228.03","Y":"194.27"},{"X":"224.70","Y":"196.82"},{"X":"224.60","Y":"196.88"}],[{"X":"292.60","Y":"192.98"},{"X":"298.73","Y":"192.56"},{"X":"303.08","Y":"191.46"},{"X":"305.98","Y":"189.91"},{"X":"307.86","Y":"187.98"},{"X":"308.96","Y":"185.54"},{"X":"309.26","Y":"183.06"},{"X":"309.20","Y":"173.81"},{"X":"304.04","Y":"175.83"},{"X":"298.03","Y":"177.13"},{"X":"291.00","Y":"177.63"},{"X":"284.29","Y":"177.05"},{"X":"278.60","Y":"175.70"},{"X":"274.88","Y":"174.11"},{"X":"271.82","Y":"171.71"},{"X":"268.90","Y":"168.11"},{"X":"266.25","Y":"163.24"},{"X":"264.52","Y":"157.97"},{"X":"263.31","Y":"150.88"},{"X":"262.81","Y":"141.60"},{"X":"263.34","Y":"132.62"},{"X":"264.61","Y":"125.23"},{"X":"266.40","Y":"119.67"},{"X":"268.74","Y":"115.31"},{"X":"271.95","Y":"111.17"},{"X":"275.11","Y":"108.42"},{"X":"278.30","Y":"106.76"},{"X":"283.81","Y":"105.35"},{"X":"290.52","Y":"104.67"},{"X":"295.16","Y":"104.90"},{"X":"299.98","Y":"105.99"},{"X":"306.08","Y":"108.50"},{"X":"309.26","Y":"110.23"},{"X":"309.26","Y":"105.06"},{"X":"329.35","Y":"105.06"},{"X":"329.23","Y":"182.37"},{"X":"328.37","Y":"186.78"},{"X":"326.76","Y":"190.49"},{"X":"324.35","Y":"193.73"},{"X":"321.15","Y":"196.44"},{"X":"316.95","Y":"198.74"},{"X":"311.55","Y":"200.55"},{"X":"304.61","Y":"201.78"},{"X":"296.23","Y":"202.25"},{"X":"286.50","Y":"201.75"},{"X":"279.10","Y":"200.55"},{"X":"271.33","Y":"198.26"},{"X":"269.50","Y":"197.28"},{"X":"268.04","Y":"186.11"},{"X":"273.14","Y":"188.62"},{"X":"280.19","Y":"191.16"},{"X":"286.81","Y":"192.48"},{"X":"292.48","Y":"192.97"},{"X":"292.60","Y":"192.98"}],[{"X":"309.26","Y":"120.42"},{"X":"308.13","Y":"119.26"},{"X":"303.52","Y":"116.46"},{"X":"299.14","Y":"114.93"},{"X":"296.63","Y":"114.57"},{"X":"291.87","Y":"115.12"},{"X":"288.19","Y":"116.39"},{"X":"285.33","Y":"118.31"},{"X":"282.99","Y":"121.03"},{"X":"281.19","Y":"124.65"},{"X":"279.81","Y":"130.01"},{"X":"278.92","Y":"137.91"},{"X":"278.88","Y":"145.11"},{"X":"279.69","Y":"152.40"},{"X":"281.18","Y":"158.05"},{"X":"283.08","Y":"162.12"},{"X":"285.18","Y":"164.65"},{"X":"287.89","Y":"166.53"},{"X":"291.41","Y":"167.79"},{"X":"296.02","Y":"168.34"},{"X":"301.23","Y":"167.79"},{"X":"305.10","Y":"166.63"},{"X":"309.03","Y":"164.44"},{"X":"309.26","Y":"164.24"},{"X":"309.26","Y":"164.24"},{"X":"309.26","Y":"120.42"}],[{"X":"369.84","Y":"104.65"},{"X":"377.62","Y":"105.10"},{"X":"383.80","Y":"106.30"},{"X":"388.72","Y":"108.11"},{"X":"392.84","Y":"110.56"},{"X":"396.29","Y":"113.67"},{"X":"399.04","Y":"117.41"},{"X":"401.13","Y":"121.89"},{"X":"402.78","Y":"127.80"},{"X":"403.75","Y":"135.08"},{"X":"403.90","Y":"143.93"},{"X":"403.19","Y":"151.69"},{"X":"401.74","Y":"158.12"},{"X":"399.70","Y":"163.28"},{"X":"397.08","Y":"167.47"},{"X":"394.00","Y":"170.74"},{"X":"390.20","Y":"173.40"},{"X":"385.55","Y":"175.49"},{"X":"379.75","Y":"176.99"},{"X":"372.60","Y":"177.77"},{"X":"365.11","Y":"177.67"},{"X":"358.43","Y":"176.71"},{"X":"352.98","Y":"175.07"},{"X":"348.43","Y":"172.78"},{"X":"344.63","Y":"169.86"},{"X":"341.60","Y":"166.34"},{"X":"339.16","Y":"162.04"},{"X":"337.26","Y":"156.68"},{"X":"335.98","Y":"150.05"},{"X":"335.46","Y":"141.97"},{"X":"335.86","Y":"134.19"},{"X":"337.04","Y":"127.55"},{"X":"338.90","Y":"122.01"},{"X":"341.37","Y":"117.38"},{"X":"344.45","Y":"113.52"},{"X":"348.10","Y":"110.41"},{"X":"352.52","Y":"107.89"},{"X":"357.76","Y":"106.06"},{"X":"363.95","Y":"104.95"},{"X":"369.79","Y":"104.65"},{"X":"369.84","Y":"104.65"}],[{"X":"355.55","Y":"140.93"},{"X":"356.44","Y":"153.65"},{"X":"357.32","Y":"158.05"},{"X":"358.99","Y":"161.73"},{"X":"361.25","Y":"164.71"},{"X":"363.70","Y":"166.43"},{"X":"366.83","Y":"167.44"},{"X":"371.01","Y":"167.70"},{"X":"374.86","Y":"167.11"},{"X":"377.72","Y":"165.83"},{"X":"379.82","Y":"163.92"},{"X":"381.31","Y":"161.33"},{"X":"382.74","Y":"155.97"},{"X":"383.65","Y":"148.23"},{"X":"383.81","Y":"137.89"},{"X":"383.12","Y":"129.22"},{"X":"381.82","Y":"122.91"},{"X":"380.48","Y":"119.58"},{"X":"378.60","Y":"117.26"},{"X":"376.07","Y":"115.66"},{"X":"372.76","Y":"114.75"},{"X":"368.72","Y":"114.64"},{"X":"364.90","Y":"115.38"},{"X":"361.94","Y":"116.82"},{"X":"359.76","Y":"118.86"},{"X":"358.22","Y":"121.60"},{"X":"356.71","Y":"126.93"},{"X":"355.77","Y":"134.21"},{"X":"355.55","Y":"140.81"},{"X":"355.55","Y":"140.93"}],[{"X":"471.82","Y":"121.68"},{"X":"471.82","Y":"177.00"},{"X":"456.69","Y":"177.00"},{"X":"456.58","Y":"122.81"},{"X":"455.77","Y":"120.25"},{"X":"454.20","Y":"118.25"},{"X":"451.80","Y":"116.74"},{"X":"448.06","Y":"115.65"},{"X":"442.84","Y":"115.25"},{"X":"438.30","Y":"116.22"},{"X":"435.19","Y":"117.70"},{"X":"432.97","Y":"119.70"},{"X":"430.36","Y":"123.35"},{"X":"429.69","Y":"124.95"},{"X":"429.69","Y":"177.00"},{"X":"409.38","Y":"177.00"},{"X":"409.38","Y":"105.07"},{"X":"429.69","Y":"105.07"},{"X":"429.70","Y":"110.91"},{"X":"434.67","Y":"107.84"},{"X":"440.16","Y":"105.83"},{"X":"445.62","Y":"104.85"},{"X":"452.21","Y":"104.70"},{"X":"457.71","Y":"105.40"},{"X":"461.98","Y":"106.78"},{"X":"465.33","Y":"108.76"},{"X":"468.28","Y":"111.73"},{"X":"470.35","Y":"115.03"},{"X":"471.52","Y":"118.58"},{"X":"471.82","Y":"121.56"},{"X":"471.82","Y":"121.68"}]]';
var text_clipping = '[[{"X":"115.54","Y":"216.41"},{"X":"113.01","Y":"217.26"},{"X":"105.79","Y":"218.58"},{"X":"96.36","Y":"219.18"},{"X":"88.73","Y":"218.69"},{"X":"82.86","Y":"217.48"},{"X":"78.40","Y":"215.73"},{"X":"74.91","Y":"213.42"},{"X":"72.12","Y":"210.49"},{"X":"69.88","Y":"206.77"},{"X":"68.13","Y":"201.83"},{"X":"66.93","Y":"195.19"},{"X":"66.45","Y":"186.51"},{"X":"67.03","Y":"179.19"},{"X":"68.36","Y":"173.03"},{"X":"70.25","Y":"168.30"},{"X":"72.58","Y":"164.82"},{"X":"75.62","Y":"161.94"},{"X":"79.35","Y":"159.71"},{"X":"84.02","Y":"158.09"},{"X":"89.93","Y":"157.11"},{"X":"96.97","Y":"156.93"},{"X":"105.18","Y":"157.69"},{"X":"114.58","Y":"159.65"},{"X":"116.44","Y":"168.59"},{"X":"110.93","Y":"166.65"},{"X":"105.50","Y":"165.51"},{"X":"98.32","Y":"165.06"},{"X":"94.16","Y":"165.58"},{"X":"90.96","Y":"166.82"},{"X":"88.62","Y":"168.68"},{"X":"86.96","Y":"171.17"},{"X":"85.31","Y":"175.67"},{"X":"84.32","Y":"181.28"},{"X":"84.15","Y":"188.62"},{"X":"84.83","Y":"196.53"},{"X":"86.18","Y":"202.53"},{"X":"87.64","Y":"205.97"},{"X":"89.52","Y":"208.29"},{"X":"92.02","Y":"209.92"},{"X":"95.18","Y":"210.85"},{"X":"99.07","Y":"210.96"},{"X":"105.48","Y":"210.28"},{"X":"111.84","Y":"208.54"},{"X":"116.32","Y":"206.45"},{"X":"116.62","Y":"206.17"},{"X":"115.55","Y":"216.31"},{"X":"115.54","Y":"216.41"}],[{"X":"120.11","Y":"219.00"},{"X":"120.11","Y":"132.31"},{"X":"137.23","Y":"132.31"},{"X":"137.23","Y":"219.00"},{"X":"137.23","Y":"219.00"},{"X":"120.11","Y":"219.00"}],[{"X":"153.34","Y":"132.13"},{"X":"156.60","Y":"132.56"},{"X":"159.17","Y":"133.77"},{"X":"161.19","Y":"135.77"},{"X":"162.61","Y":"138.40"},{"X":"163.19","Y":"141.48"},{"X":"162.87","Y":"144.75"},{"X":"161.71","Y":"147.51"},{"X":"159.79","Y":"149.80"},{"X":"157.51","Y":"151.22"},{"X":"154.46","Y":"151.96"},{"X":"151.02","Y":"151.84"},{"X":"148.34","Y":"150.92"},{"X":"146.08","Y":"149.20"},{"X":"144.28","Y":"146.63"},{"X":"143.38","Y":"143.78"},{"X":"143.37","Y":"140.64"},{"X":"144.20","Y":"137.77"},{"X":"145.94","Y":"135.15"},{"X":"148.14","Y":"133.37"},{"X":"150.80","Y":"132.38"},{"X":"153.33","Y":"132.13"},{"X":"153.34","Y":"132.13"}],[{"X":"144.58","Y":"219.00"},{"X":"144.58","Y":"157.06"},{"X":"161.88","Y":"157.06"},{"X":"161.88","Y":"219.00"},{"X":"161.88","Y":"219.00"},{"X":"144.58","Y":"219.00"}],[{"X":"168.78","Y":"243.93"},{"X":"168.78","Y":"157.06"},{"X":"186.09","Y":"157.06"},{"X":"186.19","Y":"161.44"},{"X":"191.55","Y":"158.75"},{"X":"196.41","Y":"157.22"},{"X":"201.17","Y":"156.70"},{"X":"207.60","Y":"157.21"},{"X":"212.99","Y":"158.51"},{"X":"216.03","Y":"160.11"},{"X":"219.02","Y":"162.82"},{"X":"222.00","Y":"166.95"},{"X":"223.86","Y":"171.04"},{"X":"225.25","Y":"176.56"},{"X":"226.01","Y":"184.02"},{"X":"225.94","Y":"191.67"},{"X":"225.03","Y":"199.56"},{"X":"223.55","Y":"205.37"},{"X":"221.48","Y":"209.86"},{"X":"218.90","Y":"213.53"},{"X":"216.12","Y":"216.03"},{"X":"213.05","Y":"217.58"},{"X":"207.99","Y":"218.94"},{"X":"201.87","Y":"219.54"},{"X":"195.30","Y":"219.05"},{"X":"189.90","Y":"217.79"},{"X":"186.09","Y":"216.23"},{"X":"186.09","Y":"243.93"},{"X":"186.09","Y":"243.93"},{"X":"168.78","Y":"243.93"}],[{"X":"186.09","Y":"170.27"},{"X":"186.13","Y":"208.05"},{"X":"189.79","Y":"210.09"},{"X":"193.53","Y":"211.16"},{"X":"197.86","Y":"211.53"},{"X":"201.86","Y":"210.97"},{"X":"204.91","Y":"209.73"},{"X":"207.25","Y":"207.87"},{"X":"209.01","Y":"205.27"},{"X":"210.72","Y":"200.64"},{"X":"211.90","Y":"194.30"},{"X":"212.32","Y":"186.82"},{"X":"211.80","Y":"179.19"},{"X":"210.69","Y":"174.07"},{"X":"209.11","Y":"170.68"},{"X":"207.06","Y":"168.30"},{"X":"204.40","Y":"166.63"},{"X":"200.81","Y":"165.59"},{"X":"196.18","Y":"165.25"},{"X":"192.24","Y":"166.16"},{"X":"188.91","Y":"167.87"},{"X":"186.21","Y":"169.99"},{"X":"186.09","Y":"170.26"},{"X":"186.09","Y":"170.27"}],[{"X":"231.09","Y":"243.93"},{"X":"231.09","Y":"157.06"},{"X":"248.39","Y":"157.06"},{"X":"248.50","Y":"161.44"},{"X":"253.86","Y":"158.75"},{"X":"258.72","Y":"157.22"},{"X":"263.48","Y":"156.70"},{"X":"269.91","Y":"157.21"},{"X":"275.30","Y":"158.51"},{"X":"278.34","Y":"160.11"},{"X":"281.33","Y":"162.82"},{"X":"284.31","Y":"166.96"},{"X":"286.17","Y":"171.04"},{"X":"287.56","Y":"176.57"},{"X":"288.32","Y":"184.02"},{"X":"288.25","Y":"191.67"},{"X":"287.34","Y":"199.56"},{"X":"285.86","Y":"205.37"},{"X":"283.79","Y":"209.86"},{"X":"281.21","Y":"213.53"},{"X":"278.43","Y":"216.04"},{"X":"275.35","Y":"217.58"},{"X":"270.29","Y":"218.94"},{"X":"264.18","Y":"219.54"},{"X":"257.60","Y":"219.05"},{"X":"252.21","Y":"217.78"},{"X":"248.40","Y":"216.23"},{"X":"248.40","Y":"243.93"},{"X":"248.40","Y":"243.93"},{"X":"231.09","Y":"243.93"}],[{"X":"248.39","Y":"170.27"},{"X":"248.44","Y":"208.05"},{"X":"252.10","Y":"210.09"},{"X":"255.84","Y":"211.16"},{"X":"260.17","Y":"211.53"},{"X":"264.17","Y":"210.97"},{"X":"267.22","Y":"209.73"},{"X":"269.56","Y":"207.87"},{"X":"271.32","Y":"205.27"},{"X":"273.03","Y":"200.64"},{"X":"274.21","Y":"194.30"},{"X":"274.63","Y":"186.82"},{"X":"274.12","Y":"179.19"},{"X":"273.00","Y":"174.07"},{"X":"271.42","Y":"170.68"},{"X":"269.38","Y":"168.30"},{"X":"266.72","Y":"166.64"},{"X":"263.12","Y":"165.59"},{"X":"258.49","Y":"165.25"},{"X":"254.56","Y":"166.16"},{"X":"251.23","Y":"167.86"},{"X":"248.52","Y":"169.99"},{"X":"248.40","Y":"170.26"},{"X":"248.39","Y":"170.27"}],[{"X":"302.11","Y":"132.13"},{"X":"305.37","Y":"132.56"},{"X":"307.93","Y":"133.77"},{"X":"309.95","Y":"135.77"},{"X":"311.37","Y":"138.40"},{"X":"311.95","Y":"141.48"},{"X":"311.63","Y":"144.75"},{"X":"310.47","Y":"147.51"},{"X":"308.56","Y":"149.80"},{"X":"306.27","Y":"151.22"},{"X":"303.22","Y":"151.96"},{"X":"299.79","Y":"151.84"},{"X":"297.10","Y":"150.92"},{"X":"294.85","Y":"149.20"},{"X":"293.04","Y":"146.63"},{"X":"292.14","Y":"143.78"},{"X":"292.13","Y":"140.64"},{"X":"292.97","Y":"137.77"},{"X":"294.71","Y":"135.15"},{"X":"296.90","Y":"133.37"},{"X":"299.56","Y":"132.38"},{"X":"302.10","Y":"132.13"},{"X":"302.11","Y":"132.13"}],[{"X":"293.34","Y":"219.00"},{"X":"293.34","Y":"157.06"},{"X":"310.65","Y":"157.06"},{"X":"310.65","Y":"219.00"},{"X":"310.65","Y":"219.00"},{"X":"293.34","Y":"219.00"}],[{"X":"371.24","Y":"171.37"},{"X":"371.24","Y":"219.00"},{"X":"358.21","Y":"219.00"},{"X":"358.11","Y":"172.27"},{"X":"357.35","Y":"170.01"},{"X":"355.81","Y":"168.19"},{"X":"353.32","Y":"166.83"},{"X":"349.52","Y":"165.99"},{"X":"345.63","Y":"165.91"},{"X":"341.71","Y":"166.90"},{"X":"339.11","Y":"168.36"},{"X":"336.70","Y":"171.02"},{"X":"335.06","Y":"173.87"},{"X":"334.96","Y":"219.00"},{"X":"317.48","Y":"219.00"},{"X":"317.48","Y":"157.06"},{"X":"334.96","Y":"157.06"},{"X":"335.01","Y":"162.07"},{"X":"339.49","Y":"159.34"},{"X":"344.58","Y":"157.56"},{"X":"349.77","Y":"156.78"},{"X":"355.76","Y":"156.84"},{"X":"360.47","Y":"157.70"},{"X":"364.07","Y":"159.18"},{"X":"366.81","Y":"161.26"},{"X":"369.25","Y":"164.29"},{"X":"370.67","Y":"167.42"},{"X":"371.23","Y":"170.82"},{"X":"371.24","Y":"171.27"},{"X":"371.24","Y":"171.37"}],[{"X":"401.90","Y":"232.76"},{"X":"407.58","Y":"232.33"},{"X":"411.33","Y":"231.29"},{"X":"413.88","Y":"229.75"},{"X":"415.39","Y":"227.89"},{"X":"416.16","Y":"225.48"},{"X":"416.21","Y":"216.25"},{"X":"411.46","Y":"218.08"},{"X":"405.72","Y":"219.22"},{"X":"399.44","Y":"219.51"},{"X":"393.19","Y":"218.76"},{"X":"388.30","Y":"217.32"},{"X":"385.44","Y":"215.70"},{"X":"382.67","Y":"212.97"},{"X":"380.07","Y":"208.95"},{"X":"378.24","Y":"204.52"},{"X":"376.93","Y":"198.51"},{"X":"376.28","Y":"190.44"},{"X":"376.52","Y":"182.65"},{"X":"377.58","Y":"175.38"},{"X":"379.16","Y":"170.07"},{"X":"381.22","Y":"166.08"},{"X":"384.14","Y":"162.28"},{"X":"387.11","Y":"159.77"},{"X":"390.08","Y":"158.35"},{"X":"395.48","Y":"157.12"},{"X":"401.76","Y":"156.70"},{"X":"406.66","Y":"157.40"},{"X":"412.02","Y":"159.31"},{"X":"416.24","Y":"161.51"},{"X":"416.24","Y":"157.06"},{"X":"433.55","Y":"157.06"},{"X":"433.43","Y":"223.84"},{"X":"432.54","Y":"227.93"},{"X":"430.90","Y":"231.30"},{"X":"428.49","Y":"234.16"},{"X":"425.23","Y":"236.55"},{"X":"420.86","Y":"238.51"},{"X":"415.19","Y":"239.93"},{"X":"407.89","Y":"240.69"},{"X":"399.04","Y":"240.53"},{"X":"391.61","Y":"239.58"},{"X":"384.09","Y":"237.51"},{"X":"382.01","Y":"236.48"},{"X":"380.74","Y":"226.84"},{"X":"385.59","Y":"229.19"},{"X":"391.82","Y":"231.36"},{"X":"398.16","Y":"232.49"},{"X":"401.75","Y":"232.75"},{"X":"401.90","Y":"232.76"}],[{"X":"416.25","Y":"170.27"},{"X":"414.97","Y":"169.05"},{"X":"410.96","Y":"166.71"},{"X":"406.83","Y":"165.39"},{"X":"404.44","Y":"165.28"},{"X":"400.31","Y":"165.96"},{"X":"397.15","Y":"167.34"},{"X":"394.75","Y":"169.35"},{"X":"392.82","Y":"172.20"},{"X":"391.40","Y":"176.14"},{"X":"390.35","Y":"182.50"},{"X":"390.04","Y":"190.14"},{"X":"390.67","Y":"197.15"},{"X":"391.97","Y":"202.39"},{"X":"393.71","Y":"206.20"},{"X":"395.76","Y":"208.58"},{"X":"398.42","Y":"210.25"},{"X":"401.86","Y":"211.27"},{"X":"406.04","Y":"211.51"},{"X":"410.92","Y":"210.69"},{"X":"414.49","Y":"209.16"},{"X":"416.25","Y":"208.01"},{"X":"416.25","Y":"170.27"},{"X":"416.25","Y":"170.27"}]]';
return {
"ss": deserialize_clipper_poly(text_polygon),
"cc": deserialize_clipper_poly(text_clipping)
};
}
function get_rectangle_polys() {
var rectangle1 = '[[{"X":155,"Y":61.67},{"X":155,"Y":212.7},{"X":288.98,"Y":212.7},{"X":288.98,"Y":61.67},{"X":173.01999999999998,"Y":61.67},{"X":155,"Y":61.67}],[{"X":274.37,"Y":190.77},{"X":171.24,"Y":190.77},{"X":171.24,"Y":81.16},{"X":274.37,"Y":81.16},{"X":274.37,"Y":81.16},{"X":274.37,"Y":190.77}]]';
var rectangle2 = '[[{"X":217.69,"Y":126.14},{"X":217.69,"Y":277.17},{"X":351.66999999999996,"Y":277.17},{"X":351.66999999999996,"Y":126.14},{"X":253.71,"Y":126.14},{"X":217.69,"Y":126.14}],[{"X":337.05,"Y":255.25},{"X":233.93,"Y":255.25},{"X":233.93,"Y":145.63},{"X":337.05,"Y":145.63},{"X":337.05,"Y":145.63},{"X":337.05,"Y":255.25}]]';
return {
"ss": deserialize_clipper_poly(rectangle1),
"cc": deserialize_clipper_poly(rectangle2)
};
}
function get_same_self_intersecting_polys() {
var rectangle1 = '[[{"X":172,"Y":90},{"X":222,"Y":90},{"X":222,"Y":240},{"X":322,"Y":240},{"X":322,"Y":140},{"X":172,"Y":140}]]';
var rectangle2 = '[[{"X":172,"Y":90},{"X":222,"Y":90},{"X":222,"Y":240},{"X":322,"Y":240},{"X":322,"Y":140},{"X":172,"Y":140}]]';
return {
"ss": deserialize_clipper_poly(rectangle1),
"cc": deserialize_clipper_poly(rectangle2)
};
}
function get_star_and_rect() {
var star = '[[{"X":147.47,"Y":312.74},{"X":247.07,"Y":33.510000000000005},{"X":337.66,"Y":312.04},{"X":86.36,"Y":122.7},{"X":404.07,"Y":123.57},{"X":148.11,"Y":312.27},{"X":147.47,"Y":312.74}]]';
var rectangle1 = '[[{"X":336.36,"Y":261.38},{"X":155.25,"Y":260.49},{"X":155.25,"Y":84.06},{"X":336.36,"Y":84.06},{"X":336.36,"Y":261.38},{"X":336.36,"Y":261.38}]]';
return {
"ss": deserialize_clipper_poly(star),
"cc": deserialize_clipper_poly(rectangle1)
};
}
function get_spiral_and_rects() {
var spiral = '[[{"X":222.02,"Y":294.08},{"X":202.56,"Y":292.96},{"X":183.69,"Y":289.52},{"X":165.670002,"Y":283.86},{"X":148.5,"Y":275.99},{"X":132.45999999999998,"Y":266.02},{"X":117.8,"Y":254.11},{"X":104.75,"Y":240.45},{"X":93.72,"Y":225.49},{"X":84.85,"Y":209.5},{"X":78.12,"Y":192.5},{"X":73.73,"Y":175.06},{"X":71.6,"Y":156.89},{"X":71.83,"Y":139.51},{"X":74.33,"Y":123.52},{"X":79.1,"Y":108.05},{"X":86.16,"Y":93.15},{"X":95.270001,"Y":79.41},{"X":106.41,"Y":66.85},{"X":119.19,"Y":55.97},{"X":133.6,"Y":46.79},{"X":149.15,"Y":39.7},{"X":165.51999999999998,"Y":34.82},{"X":182.41,"Y":32.2300004},{"X":199.5,"Y":31.97},{"X":214.94,"Y":34.03},{"X":229.61,"Y":38.33},{"X":243.45,"Y":44.81},{"X":256.13,"Y":53.36},{"X":267.330004,"Y":63.75},{"X":276.63,"Y":75.5},{"X":283.960004,"Y":88.57},{"X":289.03,"Y":102.35},{"X":291.840003,"Y":116.77},{"X":292.340003,"Y":131.44},{"X":290.53999999999996,"Y":144.5},{"X":286.49,"Y":157.05},{"X":280.16999999999996,"Y":168.96},{"X":271.9,"Y":179.61},{"X":261.710004,"Y":188.89},{"X":250.1,"Y":196.31},{"X":237.4,"Y":201.68},{"X":223.99,"Y":204.85},{"X":209.93,"Y":205.76},{"X":197.13,"Y":204.33},{"X":185.11,"Y":200.63},{"X":173.99,"Y":194.74},{"X":164.19,"Y":186.85},{"X":156.22,"Y":177.51},{"X":150.18,"Y":166.82},{"X":146.41,"Y":155.44},{"X":144.89,"Y":143.56},{"X":145.63,"Y":132.5},{"X":148.59,"Y":122.14},{"X":153.8,"Y":112.36},{"X":160.93,"Y":103.88},{"X":169.89,"Y":96.86},{"X":180.09,"Y":91.81},{"X":191.1,"Y":88.95},{"X":202.47,"Y":88.37},{"X":212.8,"Y":90.1},{"X":222.22,"Y":93.95},{"X":230.55,"Y":99.81},{"X":237.37,"Y":107.36},{"X":242.28,"Y":116.28},{"X":244.92,"Y":125.8},{"X":245.3,"Y":135.670002},{"X":243.42,"Y":144.44},{"X":239.27,"Y":152.74},{"X":233.18,"Y":159.73},{"X":225.31,"Y":165.19},{"X":216.34,"Y":168.54},{"X":206.82,"Y":169.6},{"X":198.23,"Y":168.4},{"X":190.26,"Y":164.97},{"X":183.5,"Y":159.55},{"X":178.57999999999998,"Y":152.77},{"X":175.66,"Y":144.92},{"X":175,"Y":136.87},{"X":176.64,"Y":132.74},{"X":180.14,"Y":130},{"X":184.55,"Y":129.420002},{"X":188.66,"Y":131.13},{"X":191.34,"Y":134.670002},{"X":192.98,"Y":143.46},{"X":196.15,"Y":148.15},{"X":201.12,"Y":151.43},{"X":206.97,"Y":152.6},{"X":214.06,"Y":151.54},{"X":220.43,"Y":148.24},{"X":225.15,"Y":143.26},{"X":227.84,"Y":137.26},{"X":228.34,"Y":130.41},{"X":226.58,"Y":122.84},{"X":222.68,"Y":116.12},{"X":216.8,"Y":110.59},{"X":209.61,"Y":106.92},{"X":201.39,"Y":105.35},{"X":191.84,"Y":106.08},{"X":182.76,"Y":109.12},{"X":174.94,"Y":114.11},{"X":168.57999999999998,"Y":120.86},{"X":164.26,"Y":128.730002},{"X":162.09,"Y":137.44},{"X":162.230002,"Y":147.31},{"X":164.64,"Y":157.2},{"X":169.35,"Y":166.56},{"X":176.03,"Y":174.64},{"X":184.51999999999998,"Y":181.28},{"X":194.25,"Y":185.91},{"X":204.75,"Y":188.36},{"X":216.73,"Y":188.57},{"X":228.83,"Y":186.5},{"X":240.34,"Y":182.2},{"X":250.83,"Y":175.81},{"X":259.9,"Y":167.54},{"X":267.03,"Y":157.91},{"X":272.02,"Y":147.35},{"X":274.830004,"Y":136.01},{"X":275.340003,"Y":123.74},{"X":273.580004,"Y":110.68},{"X":269.580004,"Y":98.12},{"X":263.3,"Y":86.19},{"X":255.05,"Y":75.52},{"X":244.91,"Y":66.18},{"X":233.35,"Y":58.67},{"X":220.7,"Y":53.21},{"X":207.02,"Y":49.86},{"X":192.67,"Y":48.8},{"X":176.82999999999998,"Y":50},{"X":161.34,"Y":53.51},{"X":146.540002,"Y":59.27},{"X":132.76,"Y":67.17},{"X":120.53999999999999,"Y":76.85},{"X":109.92,"Y":88.25},{"X":101.32,"Y":100.89},{"X":94.75999999999999,"Y":114.7},{"X":90.52,"Y":129.07},{"X":88.57,"Y":143.93},{"X":88.92,"Y":160.71},{"X":91.63,"Y":177.58},{"X":96.66,"Y":193.91},{"X":103.9,"Y":209.38},{"X":113.37,"Y":223.97},{"X":124.72,"Y":237.14},{"X":137.93,"Y":248.89},{"X":152.59,"Y":258.78999999999996},{"X":168.44,"Y":266.65},{"X":185.18,"Y":272.34},{"X":202.83,"Y":275.85},{"X":221.07,"Y":277.07},{"X":241.43,"Y":276},{"X":261.24,"Y":272.63},{"X":280.52,"Y":266.98},{"X":299,"Y":259.12},{"X":316.43,"Y":249.14},{"X":332.56,"Y":237.17},{"X":346.94,"Y":223.57},{"X":359.59,"Y":208.35},{"X":370.15,"Y":191.97},{"X":378.51,"Y":174.7},{"X":384.69,"Y":156.54},{"X":388.59,"Y":137.75},{"X":390.64,"Y":115.98},{"X":393.21,"Y":112.35},{"X":397.26,"Y":110.5},{"X":401.4,"Y":110.83},{"X":405.08,"Y":113.33},{"X":407,"Y":117.34},{"X":406.15,"Y":134.39},{"X":402.81,"Y":154.2},{"X":397.18,"Y":173.48},{"X":389.22,"Y":192.25},{"X":379.12,"Y":209.96},{"X":366.88,"Y":226.64},{"X":352.8,"Y":241.8},{"X":336.88,"Y":255.47},{"X":319.51,"Y":267.26},{"X":300.93,"Y":277.02},{"X":281.38,"Y":284.66},{"X":260.81,"Y":290.13},{"X":239.75,"Y":293.26},{"X":222.07,"Y":294.08},{"X":222.02,"Y":294.08}]]';
var rectangle1 = '[[{"X":210.91,"Y":210.89},{"X":107.65,"Y":210.65},{"X":107.65,"Y":27.58},{"X":210.91,"Y":27.75},{"X":210.91,"Y":36.45},{"X":210.91,"Y":210.89}],[{"X":304.24,"Y":323.13},{"X":210.91,"Y":322.86},{"X":210.91,"Y":70.9},{"X":304.24,"Y":70.91},{"X":304.24,"Y":148.61},{"X":304.24,"Y":323.13}],[{"X":407.5,"Y":254.59},{"X":304.24,"Y":254.35},{"X":304.24,"Y":71.28},{"X":407.5,"Y":71.28},{"X":407.5,"Y":71.28},{"X":407.5,"Y":254.59}]]';
return {
"ss": deserialize_clipper_poly(spiral),
"cc": deserialize_clipper_poly(rectangle1)
};
}
function get_rounded_grid_and_star() {
var rounded_grid = '[[{"X":346.2,"Y":66.31},{"X":332.04,"Y":53.81},{"X":316.43,"Y":43.18},{"X":299.87,"Y":34.73995},{"X":282.39,"Y":28.44},{"X":263.95,"Y":24.33},{"X":245.15,"Y":22.56002},{"X":226.27,"Y":23.13},{"X":207.6,"Y":26.04},{"X":189.74,"Y":31.18},{"X":172.6702,"Y":38.53},{"X":156.65,"Y":47.95},{"X":141.69,"Y":59.49},{"X":128.2902,"Y":72.8},{"X":116.65,"Y":87.67},{"X":107.11,"Y":103.62},{"X":99.64,"Y":120.65},{"X":94.37,"Y":138.47},{"X":91.33,"Y":157.11},{"X":90.63,"Y":175.99},{"X":92.2599,"Y":194.8},{"X":96.2401,"Y":213.27},{"X":102.42,"Y":230.8},{"X":110.74,"Y":247.42},{"X":121.26,"Y":263.1},{"X":133.6702,"Y":277.35},{"X":147.74,"Y":289.95},{"X":163.2698,"Y":300.69},{"X":179.77,"Y":309.25},{"X":197.21,"Y":315.68},{"X":215.62,"Y":319.92},{"X":234.41,"Y":321.81},{"X":253.29,"Y":321.38},{"X":271.98,"Y":318.6},{"X":289.87,"Y":313.58},{"X":307,"Y":306.36},{"X":323.09,"Y":297.05},{"X":338.12,"Y":285.62},{"X":351.62,"Y":272.4},{"X":363.37,"Y":257.61},{"X":373.02,"Y":241.73},{"X":380.61,"Y":224.76},{"X":386,"Y":206.97},{"X":389.18,"Y":188.35},{"X":390.01,"Y":169.48},{"X":388.5,"Y":150.65},{"X":384.65,"Y":132.16},{"X":378.61,"Y":114.59},{"X":370.4,"Y":97.91},{"X":359.99,"Y":82.15},{"X":347.69,"Y":67.82},{"X":346.21,"Y":66.32},{"X":346.2,"Y":66.31}],[{"X":375.03,"Y":172.21},{"X":373.83,"Y":190.15},{"X":370.54,"Y":206.57},{"X":362.32,"Y":203.86},{"X":365.47,"Y":187.07},{"X":366.32,"Y":169.7},{"X":364.79,"Y":152.38},{"X":360.96,"Y":135.73},{"X":354.9,"Y":119.76},{"X":346.72,"Y":104.76},{"X":336.39,"Y":90.77},{"X":324.26,"Y":78.32},{"X":320.42,"Y":74.77},{"X":325.57,"Y":67.97},{"X":338.69,"Y":80.27},{"X":350.08,"Y":94.19},{"X":359.52,"Y":109.5},{"X":366.74,"Y":125.65},{"X":371.75,"Y":142.61},{"X":374.49,"Y":160.09},{"X":375.03,"Y":172.07},{"X":375.03,"Y":172.21}],[{"X":129.2698,"Y":172.21},{"X":130.44,"Y":156.06},{"X":133.9598,"Y":140.26},{"X":139.75,"Y":125.15},{"X":147.69,"Y":111.05},{"X":157.6,"Y":98.25},{"X":169.15,"Y":87.09},{"X":174.15,"Y":94.2},{"X":163.0798,"Y":105.17},{"X":153.78,"Y":117.67},{"X":146.4802,"Y":131.44},{"X":141.37,"Y":146.17},{"X":138.57,"Y":161.5},{"X":138.13,"Y":177.08},{"X":140.05,"Y":192.54},{"X":140.87,"Y":196.58},{"X":132.61,"Y":199.24},{"X":129.82,"Y":183.3},{"X":129.2698,"Y":172.22},{"X":129.2698,"Y":172.21}],[{"X":327.6,"Y":172.21},{"X":326.42,"Y":186.54},{"X":325.15,"Y":191.85},{"X":317.08,"Y":189.04},{"X":318.84,"Y":175.67},{"X":318.28,"Y":162.2},{"X":315.43,"Y":149.02},{"X":310.34,"Y":136.54},{"X":303.19,"Y":125.11},{"X":293.99,"Y":114.85},{"X":292.48,"Y":113.21},{"X":297.61,"Y":106.36},{"X":307.64,"Y":116.67},{"X":315.86,"Y":128.4802},{"X":322.02,"Y":141.48},{"X":325.96,"Y":155.31},{"X":327.56,"Y":169.6},{"X":327.6,"Y":172},{"X":327.6,"Y":172.21}],[{"X":153.0198,"Y":172.21},{"X":154.19,"Y":157.87},{"X":157.7098,"Y":143.92},{"X":163.49,"Y":130.75},{"X":171.35,"Y":118.7},{"X":181.07,"Y":108.1},{"X":183.26,"Y":106.49},{"X":188.11,"Y":113.5},{"X":178.65,"Y":123.52},{"X":171.2098,"Y":134.7698},{"X":165.8,"Y":147.12},{"X":162.61,"Y":160.22},{"X":161.7098,"Y":173.67},{"X":163.13,"Y":187.08},{"X":163.36,"Y":189.28},{"X":155.22,"Y":191.74},{"X":153.18,"Y":177.51},{"X":153.0198,"Y":172.41},{"X":153.0198,"Y":172.21}],[{"X":303.92,"Y":172.21},{"X":302.73,"Y":184.43},{"X":294.39,"Y":181.76},{"X":295.2104,"Y":170.41},{"X":293.6597,"Y":159.14},{"X":289.8,"Y":148.43},{"X":283.81,"Y":138.76},{"X":278.44,"Y":132.51},{"X":283.57,"Y":125.63},{"X":291.73,"Y":134.82},{"X":297.98,"Y":145.39},{"X":302.07,"Y":156.97},{"X":303.84,"Y":169.12},{"X":303.92,"Y":172.12},{"X":303.92,"Y":172.21}],[{"X":200.38,"Y":172.21},{"X":201.53,"Y":162.7},{"X":204.93,"Y":153.75},{"X":210.37,"Y":145.87},{"X":211.36,"Y":145.15},{"X":216.39,"Y":152.18},{"X":211.93,"Y":159.27},{"X":209.51,"Y":167.29},{"X":209.16,"Y":174.43},{"X":200.68,"Y":177.07},{"X":200.38,"Y":172.28},{"X":200.38,"Y":172.21}],[{"X":279.9196,"Y":177.17},{"X":271.4,"Y":174.36},{"X":270.87,"Y":166.01},{"X":268.12,"Y":158.1},{"X":264.19,"Y":152.1},{"X":269.5,"Y":145.05},{"X":275.1597,"Y":152.78},{"X":278.81,"Y":161.64},{"X":280.22,"Y":171.11},{"X":279.93,"Y":177.09},{"X":279.9196,"Y":177.17}],[{"X":251.65,"Y":183.77},{"X":247.81,"Y":186.59},{"X":247.94,"Y":182.57},{"X":251.37,"Y":183.68},{"X":251.65,"Y":183.77}],[{"X":252.45,"Y":168.26},{"X":254.89,"Y":165.21},{"X":256.24,"Y":169.49},{"X":252.54,"Y":168.29},{"X":252.45,"Y":168.26}],[{"X":240.31,"Y":159.45},{"X":238.03,"Y":156.14},{"X":242.64,"Y":156.24},{"X":240.35,"Y":159.39},{"X":240.31,"Y":159.45}],[{"X":228.17,"Y":168.26},{"X":224.41,"Y":169.28},{"X":225.81,"Y":165.03},{"X":228.11,"Y":168.18},{"X":228.17,"Y":168.26}],[{"X":232.81,"Y":182.52},{"X":232.69,"Y":186.53},{"X":229.1,"Y":183.73},{"X":232.52,"Y":182.62},{"X":232.81,"Y":182.52}],[{"X":247.81,"Y":202.54},{"X":255.59,"Y":199.43},{"X":262.26,"Y":194.37},{"X":266.99,"Y":188.75},{"X":275.13,"Y":191.69},{"X":269.4604,"Y":199.41},{"X":262.13,"Y":205.57},{"X":253.54,"Y":209.81},{"X":247.81,"Y":211.35},{"X":247.81,"Y":211.35},{"X":247.81,"Y":202.54}],[{"X":252.07,"Y":143.27},{"X":243.96,"Y":141.17},{"X":235.59,"Y":141.32},{"X":228.44,"Y":143.14},{"X":223.38,"Y":136.11},{"X":232.48,"Y":133.12},{"X":242.03,"Y":132.38},{"X":251.47,"Y":133.94},{"X":257.2,"Y":136.21},{"X":252.09,"Y":143.25},{"X":252.07,"Y":143.27}],[{"X":213.83,"Y":188.68},{"X":219.16,"Y":195.14},{"X":226,"Y":199.96},{"X":232.81,"Y":202.54},{"X":232.71,"Y":211.33},{"X":223.6,"Y":208.4},{"X":215.45,"Y":203.37},{"X":208.72,"Y":196.55},{"X":205.5,"Y":191.38},{"X":213.77,"Y":188.7},{"X":213.83,"Y":188.68}],[{"X":232.81,"Y":226.54},{"X":232.63,"Y":235.34},{"X":220.63,"Y":232.69},{"X":209.38,"Y":227.76},{"X":199.29,"Y":220.77},{"X":190.72,"Y":211.98},{"X":183.98,"Y":201.71},{"X":182.64,"Y":198.8},{"X":190.9,"Y":196.16},{"X":196.92,"Y":205.81},{"X":204.8,"Y":214.02},{"X":214.19,"Y":220.45},{"X":224.71,"Y":224.8},{"X":232.61,"Y":226.52},{"X":232.81,"Y":226.54}],[{"X":247.81,"Y":226.54},{"X":258.86,"Y":223.83},{"X":269.11,"Y":218.89},{"X":278.13,"Y":211.95},{"X":285.53,"Y":203.31},{"X":289.8,"Y":196.15},{"X":298.05,"Y":198.84},{"X":291.8403,"Y":209.44},{"X":283.7104,"Y":218.65},{"X":273.98,"Y":226.14},{"X":262.99,"Y":231.62},{"X":251.15,"Y":234.87},{"X":247.81,"Y":235.36},{"X":247.81,"Y":235.36},{"X":247.81,"Y":226.54}],[{"X":266.2,"Y":123.84},{"X":255.67,"Y":119.53},{"X":244.47,"Y":117.5},{"X":233.1,"Y":117.82},{"X":222.03,"Y":120.47},{"X":214.38,"Y":123.79},{"X":209.48,"Y":116.59},{"X":220.74,"Y":111.68},{"X":232.73,"Y":109.05},{"X":245.01,"Y":108.77},{"X":257.11,"Y":110.86},{"X":268.5804,"Y":115.25},{"X":271.2896,"Y":116.84},{"X":266.35,"Y":123.63},{"X":266.2,"Y":123.84}],[{"X":202.29,"Y":132.6702},{"X":194.93,"Y":141.35},{"X":189.51,"Y":151.35},{"X":186.28,"Y":162.26},{"X":185.4,"Y":173.61},{"X":186.15,"Y":181.89},{"X":177.9,"Y":184.53},{"X":176.7,"Y":172.31},{"X":177.86,"Y":160.08},{"X":181.36,"Y":148.31},{"X":187.07,"Y":137.44},{"X":194.76,"Y":127.86},{"X":197.29,"Y":125.8},{"X":202.24,"Y":132.6},{"X":202.29,"Y":132.6702}],[{"X":168.2,"Y":203.48},{"X":174.63,"Y":215.34},{"X":182.98,"Y":225.92},{"X":193.01,"Y":234.94},{"X":204.42,"Y":242.12},{"X":216.89,"Y":247.24},{"X":230.06,"Y":250.14},{"X":232.81,"Y":250.45},{"X":232.66,"Y":259.23},{"X":218.49,"Y":256.8},{"X":204.91,"Y":252.05},{"X":192.3,"Y":245.13},{"X":180.99,"Y":236.24},{"X":171.2902,"Y":225.63},{"X":163.44,"Y":213.57},{"X":160.07,"Y":206.12},{"X":168.06,"Y":203.53},{"X":168.2,"Y":203.48}],[{"X":247.81,"Y":250.45},{"X":261.07,"Y":248.02},{"X":273.72,"Y":243.35},{"X":285.37,"Y":236.57},{"X":295.7104,"Y":227.91},{"X":304.42,"Y":217.62},{"X":311.26,"Y":206},{"X":312.63,"Y":203.56},{"X":320.65,"Y":206.36},{"X":313.94,"Y":219.08},{"X":305.24,"Y":230.53},{"X":294.7896,"Y":240.41},{"X":282.86,"Y":248.46},{"X":269.78,"Y":254.44},{"X":255.89,"Y":258.1696},{"X":247.81,"Y":259.24},{"X":247.81,"Y":259.24},{"X":247.81,"Y":250.45}],[{"X":280.24,"Y":104.53},{"X":268.0903,"Y":98.68},{"X":255.12,"Y":95},{"X":241.71,"Y":93.61},{"X":228.26,"Y":94.52},{"X":215.16,"Y":97.73},{"X":202.81,"Y":103.15},{"X":200.26,"Y":104.37},{"X":195.31,"Y":97.37},{"X":208.19,"Y":90.97},{"X":221.96,"Y":86.79},{"X":236.22,"Y":84.94},{"X":250.6,"Y":85.45},{"X":264.69,"Y":88.31},{"X":278.12,"Y":93.47},{"X":285.3304,"Y":97.52},{"X":280.4,"Y":104.31},{"X":280.24,"Y":104.53}],[{"X":240.31,"Y":69.85},{"X":224.77,"Y":71.03},{"X":209.88,"Y":74.47},{"X":195.4,"Y":80.25},{"X":186.27,"Y":85.13},{"X":181.44,"Y":78.15},{"X":195.74,"Y":70.57},{"X":211,"Y":65.16},{"X":226.88,"Y":62.04},{"X":243.04,"Y":61.27},{"X":259.15,"Y":62.84},{"X":274.86,"Y":66.7401},{"X":289.8304,"Y":72.9},{"X":299.23,"Y":78.4},{"X":294.06,"Y":85.16},{"X":280.23,"Y":77.97},{"X":265.47,"Y":72.98},{"X":250.12,"Y":70.31},{"X":240.53,"Y":69.85},{"X":240.31,"Y":69.85}],[{"X":145.5798,"Y":210.82},{"X":152.56,"Y":224.76},{"X":161.57,"Y":237.47},{"X":172.4,"Y":248.69},{"X":184.79,"Y":258.14},{"X":198.47,"Y":265.6},{"X":213.13,"Y":270.89},{"X":228.42,"Y":273.88},{"X":232.81,"Y":274.29},{"X":232.64,"Y":282.91},{"X":216.62,"Y":280.63},{"X":201.1,"Y":276.03},{"X":186.42,"Y":269.2},{"X":172.89,"Y":260.32},{"X":160.7902,"Y":249.58},{"X":150.36,"Y":237.2},{"X":141.8298,"Y":223.44},{"X":137.4202,"Y":213.47},{"X":145.41,"Y":210.88},{"X":145.5798,"Y":210.82}],[{"X":247.81,"Y":274.29},{"X":263.22,"Y":271.97},{"X":277.81,"Y":267.44},{"X":291.82,"Y":260.61},{"X":304.63,"Y":251.73},{"X":315.96,"Y":241.02},{"X":325.54,"Y":228.73},{"X":333.15,"Y":215.13},{"X":335.12,"Y":210.85},{"X":343.34,"Y":213.56},{"X":336.21,"Y":228.09},{"X":327.03,"Y":241.42},{"X":316.03,"Y":253.29},{"X":303.43,"Y":263.45},{"X":289.49,"Y":271.68},{"X":274.5,"Y":277.79},{"X":258.78,"Y":281.63},{"X":247.81,"Y":282.92},{"X":247.81,"Y":282.92},{"X":247.81,"Y":274.29}],[{"X":339.67,"Y":196.56},{"X":342.21,"Y":181.18},{"X":342.39,"Y":165.6},{"X":340.21,"Y":150.16},{"X":335.7,"Y":135.25},{"X":328.96,"Y":121.19},{"X":320.17,"Y":108.32},{"X":309.54,"Y":96.93},{"X":306.51,"Y":93.91},{"X":311.67,"Y":87.25},{"X":323.27,"Y":98.54},{"X":333.13,"Y":111.37},{"X":341.03,"Y":125.5},{"X":346.77,"Y":140.63},{"X":350.23,"Y":156.45},{"X":351.35,"Y":172.59},{"X":350.12,"Y":188.73},{"X":347.95,"Y":199.24},{"X":339.68,"Y":196.56},{"X":339.67,"Y":196.56}],[{"X":313.32,"Y":59.03},{"X":307.96,"Y":65.9601},{"X":292.9604,"Y":57.76},{"X":277,"Y":51.68},{"X":260.35,"Y":47.83},{"X":243.33,"Y":46.27},{"X":225.96,"Y":47.04},{"X":209.16,"Y":50.13},{"X":192.93,"Y":55.47},{"X":177.58,"Y":62.97},{"X":172.2698,"Y":65.87},{"X":167.5,"Y":58.9},{"X":182.96,"Y":50.31},{"X":199.42,"Y":43.83},{"X":216.59,"Y":39.58},{"X":234.17,"Y":37.62005},{"X":252.15,"Y":38},{"X":269.63,"Y":40.7},{"X":286.6,"Y":45.68},{"X":302.76,"Y":52.86},{"X":313.12,"Y":58.91},{"X":313.32,"Y":59.03}],[{"X":155.18,"Y":67.87},{"X":160.1,"Y":75.1},{"X":147.49,"Y":87.07},{"X":136.63,"Y":100.65},{"X":127.73,"Y":115.58},{"X":121.08,"Y":131.32},{"X":116.64,"Y":147.82},{"X":114.48,"Y":164.77},{"X":114.66,"Y":182.16},{"X":117.15,"Y":199.06},{"X":118.19,"Y":203.94},{"X":110.01,"Y":206.46},{"X":106.64,"Y":189.1},{"X":105.59,"Y":171.14},{"X":106.92,"Y":153.21},{"X":110.56,"Y":135.9},{"X":116.45,"Y":119.22},{"X":124.48,"Y":103.46},{"X":134.7,"Y":88.66},{"X":146.78,"Y":75.33},{"X":155.07,"Y":67.9601},{"X":155.18,"Y":67.87}],[{"X":114.68,"Y":220.85},{"X":123.06,"Y":218.44},{"X":130.4202,"Y":233.86},{"X":139.99,"Y":248.38},{"X":151.45,"Y":261.4604},{"X":164.59,"Y":272.84},{"X":179.16,"Y":282.33},{"X":194.62,"Y":289.61},{"X":210.92,"Y":294.71},{"X":227.77,"Y":297.55},{"X":232.81,"Y":297.95},{"X":232.52,"Y":306.7},{"X":214.97,"Y":304.53},{"X":197.86,"Y":300.07},{"X":181.48,"Y":293.39},{"X":166.12,"Y":284.61},{"X":151.82,"Y":273.7},{"X":139.0798,"Y":261.01},{"X":128.14,"Y":246.73},{"X":119.31,"Y":231.41},{"X":114.72,"Y":220.97},{"X":114.68,"Y":220.85}],[{"X":247.81,"Y":306.71},{"X":248.04,"Y":297.93},{"X":264.99,"Y":295.74},{"X":281.48,"Y":291.27},{"X":297.2,"Y":284.59},{"X":311.88,"Y":275.84},{"X":325.47,"Y":264.99},{"X":337.44,"Y":252.39},{"X":347.59,"Y":238.27},{"X":355.57,"Y":223.16},{"X":357.93,"Y":218.25},{"X":365.83,"Y":221.11},{"X":358.32,"Y":237.13},{"X":348.6,"Y":252.26},{"X":336.95,"Y":265.97},{"X":323.6,"Y":278.02},{"X":308.77,"Y":288.19},{"X":292.99,"Y":296.19},{"X":276.3,"Y":302.04},{"X":258.98,"Y":305.63},{"X":247.94,"Y":306.71},{"X":247.81,"Y":306.71}]]';
var star = '[[{"X":240.31,"Y":12.26},{"X":292.45,"Y":117.59},{"X":408.23,"Y":134.69},{"X":324.43,"Y":216.63},{"X":344.16,"Y":332.1},{"X":240.06,"Y":277.63},{"X":136.4,"Y":332},{"X":155.99,"Y":216.19},{"X":72.33,"Y":134.4202},{"X":188.37,"Y":117.5},{"X":240.28,"Y":12.30999},{"X":240.31,"Y":12.26}]]';
return {
"ss": deserialize_clipper_poly(rounded_grid),
"cc": deserialize_clipper_poly(star)
};
}
function get_glyph_and_grid() {
var glyph = '[[{"X":105.65,"Y":51.37},{"X":105.65,"Y":295.26},{"X":395.09,"Y":295.26},{"X":395.01,"Y":51.37},{"X":108.51,"Y":51.37},{"X":105.65,"Y":51.37}],[{"X":380.09,"Y":89.4799},{"X":349.33,"Y":89.4799},{"X":349.33,"Y":66.37},{"X":380.09,"Y":66.37},{"X":380.09,"Y":66.37},{"X":380.09,"Y":89.4799}],[{"X":257.87,"Y":127.7},{"X":257.87,"Y":104.48},{"X":288.63,"Y":104.48},{"X":288.44,"Y":127.7},{"X":283.0396,"Y":127.7},{"X":257.87,"Y":127.7}],[{"X":288.63,"Y":142.7},{"X":288.63,"Y":165.81},{"X":257.87,"Y":165.81},{"X":257.87,"Y":142.7},{"X":257.87,"Y":142.7},{"X":288.63,"Y":142.7}],[{"X":242.87,"Y":127.7},{"X":212.11,"Y":127.7},{"X":212.11,"Y":104.48},{"X":242.87,"Y":104.48},{"X":242.87,"Y":104.48},{"X":242.87,"Y":127.7}],[{"X":242.87,"Y":142.7},{"X":242.87,"Y":165.81},{"X":212.11,"Y":165.81},{"X":212.11,"Y":142.7},{"X":212.11,"Y":142.7},{"X":242.87,"Y":142.7}],[{"X":197.11,"Y":165.81},{"X":166.41,"Y":165.81},{"X":166.41,"Y":142.7},{"X":197.11,"Y":142.7},{"X":197.11,"Y":142.7},{"X":197.11,"Y":165.81}],[{"X":197.11,"Y":180.81},{"X":197.11,"Y":203.92},{"X":166.41,"Y":203.92},{"X":166.41,"Y":180.81},{"X":166.41,"Y":180.81},{"X":197.11,"Y":180.81}],[{"X":212.11,"Y":180.81},{"X":242.87,"Y":180.81},{"X":242.87,"Y":203.92},{"X":212.11,"Y":203.92},{"X":212.11,"Y":203.92},{"X":212.11,"Y":180.81}],[{"X":242.87,"Y":218.92},{"X":242.87,"Y":242.14},{"X":212.11,"Y":242.14},{"X":212.11,"Y":218.92},{"X":212.11,"Y":218.92},{"X":242.87,"Y":218.92}],[{"X":257.87,"Y":218.92},{"X":288.63,"Y":218.92},{"X":288.63,"Y":242.14},{"X":257.87,"Y":242.14},{"X":257.87,"Y":242.14},{"X":257.87,"Y":218.92}],[{"X":257.87,"Y":203.92},{"X":257.87,"Y":180.81},{"X":288.63,"Y":180.81},{"X":288.63,"Y":203.92},{"X":288.63,"Y":203.92},{"X":257.87,"Y":203.92}],[{"X":303.63,"Y":180.81},{"X":334.33004,"Y":180.81},{"X":334.33004,"Y":203.92},{"X":303.63,"Y":203.92},{"X":303.63,"Y":203.92},{"X":303.63,"Y":180.81}],[{"X":303.63,"Y":165.81},{"X":303.63,"Y":142.7},{"X":334.33004,"Y":142.7},{"X":334.33004,"Y":165.81},{"X":334.33004,"Y":165.81},{"X":303.63,"Y":165.81}],[{"X":303.63,"Y":127.7},{"X":303.63,"Y":104.48},{"X":334.33004,"Y":104.48},{"X":334.07,"Y":127.7},{"X":328.6696,"Y":127.7},{"X":303.63,"Y":127.7}],[{"X":303.63,"Y":89.4799},{"X":303.63,"Y":66.37},{"X":334.33004,"Y":66.37},{"X":334.15,"Y":89.4799},{"X":316.15,"Y":89.4799},{"X":303.63,"Y":89.4799}],[{"X":288.63,"Y":89.4799},{"X":257.87,"Y":89.4799},{"X":257.87,"Y":66.37},{"X":288.63,"Y":66.37},{"X":288.63,"Y":89.4799},{"X":288.63,"Y":89.4799}],[{"X":242.87,"Y":89.4799},{"X":212.11,"Y":89.4799},{"X":212.11,"Y":66.37},{"X":242.87,"Y":66.37},{"X":242.87,"Y":66.37},{"X":242.87,"Y":89.4799}],[{"X":197.11,"Y":89.4799},{"X":166.41,"Y":89.4799},{"X":166.41,"Y":66.37},{"X":197.11,"Y":66.37},{"X":197.11,"Y":66.37},{"X":197.11,"Y":89.4799}],[{"X":197.11,"Y":104.48},{"X":196.93,"Y":127.7},{"X":166.41,"Y":127.7},{"X":166.41,"Y":104.48},{"X":166.41,"Y":104.48},{"X":197.11,"Y":104.48}],[{"X":151.41,"Y":127.7},{"X":120.65,"Y":127.7},{"X":120.65,"Y":104.48},{"X":151.41,"Y":104.48},{"X":151.41,"Y":104.48},{"X":151.41,"Y":127.7}],[{"X":151.41,"Y":142.7},{"X":151.41,"Y":165.81},{"X":120.65,"Y":165.81},{"X":120.65,"Y":142.7},{"X":120.65,"Y":142.7},{"X":151.41,"Y":142.7}],[{"X":151.41,"Y":180.81},{"X":151.41,"Y":203.92},{"X":120.65,"Y":203.92},{"X":120.65,"Y":180.81},{"X":120.65,"Y":180.81},{"X":151.41,"Y":180.81}],[{"X":151.41,"Y":218.92},{"X":151.41,"Y":242.14},{"X":120.65,"Y":242.14},{"X":120.65,"Y":218.92},{"X":120.65,"Y":218.92},{"X":151.41,"Y":218.92}],[{"X":166.41,"Y":218.92},{"X":197.11,"Y":218.92},{"X":197.11,"Y":242.14},{"X":166.41,"Y":242.14},{"X":166.41,"Y":242.14},{"X":166.41,"Y":218.92}],[{"X":197.11,"Y":257.14},{"X":197.11,"Y":280.26},{"X":166.41,"Y":280.26},{"X":166.41,"Y":257.14},{"X":166.41,"Y":257.14},{"X":197.11,"Y":257.14}],[{"X":212.11,"Y":257.14},{"X":242.87,"Y":257.14},{"X":242.87,"Y":280.26},{"X":212.11,"Y":280.26},{"X":212.11,"Y":280.26},{"X":212.11,"Y":257.14}],[{"X":257.87,"Y":257.14},{"X":288.63,"Y":257.14},{"X":288.63,"Y":280.26},{"X":257.87,"Y":280.26},{"X":257.87,"Y":280.26},{"X":257.87,"Y":257.14}],[{"X":303.63,"Y":257.14},{"X":334.33004,"Y":257.14},{"X":334.33004,"Y":280.26},{"X":303.63,"Y":280.26},{"X":303.63,"Y":280.26},{"X":303.63,"Y":257.14}],[{"X":303.63,"Y":242.14},{"X":303.63,"Y":218.92},{"X":334.33004,"Y":218.92},{"X":334.33004,"Y":242.14},{"X":334.33004,"Y":242.14},{"X":303.63,"Y":242.14}],[{"X":349.33,"Y":218.92},{"X":380.09,"Y":218.92},{"X":380.09,"Y":242.14},{"X":349.33,"Y":242.14},{"X":349.33,"Y":242.14},{"X":349.33,"Y":218.92}],[{"X":349.33,"Y":203.92},{"X":349.33,"Y":180.81},{"X":380.09,"Y":180.81},{"X":380.09,"Y":203.92},{"X":380.09,"Y":203.92},{"X":349.33,"Y":203.92}],[{"X":349.33,"Y":165.81},{"X":349.33,"Y":142.7},{"X":380.09,"Y":142.7},{"X":380.09,"Y":165.81},{"X":380.09,"Y":165.81},{"X":349.33,"Y":165.81}],[{"X":349.33,"Y":127.7},{"X":349.33,"Y":104.48},{"X":380.09,"Y":104.48},{"X":379.9,"Y":127.7},{"X":374.5,"Y":127.7},{"X":349.33,"Y":127.7}],[{"X":151.41,"Y":66.37},{"X":151.12,"Y":89.4799},{"X":120.65,"Y":89.4799},{"X":120.65,"Y":66.37},{"X":120.65,"Y":66.37},{"X":151.41,"Y":66.37}],[{"X":120.65,"Y":257.14},{"X":151.41,"Y":257.14},{"X":151.41,"Y":280.26},{"X":120.65,"Y":280.26},{"X":120.65,"Y":280.26},{"X":120.65,"Y":257.14}],[{"X":349.33,"Y":280.26},{"X":349.33,"Y":257.14},{"X":380.09,"Y":257.14},{"X":380.09,"Y":280.26},{"X":380.09,"Y":280.26},{"X":349.33,"Y":280.26}]]';
var grid = '[[{"X":305.08004,"Y":241.97},{"X":306,"Y":251.51},{"X":308.18,"Y":256.4},{"X":311.72,"Y":259.09003},{"X":317.31,"Y":260.01},{"X":324.71004,"Y":259.01},{"X":332.45,"Y":255.86},{"X":335.57,"Y":257.5396},{"X":337.6,"Y":260.44},{"X":336.94,"Y":262.33004},{"X":328.27,"Y":268.74},{"X":317.9,"Y":273.4196},{"X":307.94,"Y":275.49},{"X":296.26,"Y":275.23},{"X":286.64,"Y":272.99},{"X":279.7896,"Y":269.31},{"X":274.14,"Y":263.55},{"X":271.6597,"Y":260.21004},{"X":269.2,"Y":261.06},{"X":254.83,"Y":268.51},{"X":242.11,"Y":272.97},{"X":227.59,"Y":275.23},{"X":209.91,"Y":275.48},{"X":197.47,"Y":273.63},{"X":187.91,"Y":270.13},{"X":180.48002,"Y":265.09003},{"X":175.32,"Y":258.88},{"X":172.2098,"Y":251.44},{"X":171.1,"Y":242.23},{"X":172.24,"Y":233.63},{"X":175.49,"Y":226.24},{"X":181,"Y":219.54},{"X":189.42002,"Y":213.3},{"X":201.36,"Y":207.73},{"X":217.23,"Y":203.25},{"X":238.28,"Y":200.1},{"X":265.24,"Y":198.78},{"X":269.37,"Y":198.47},{"X":269.98,"Y":182.93},{"X":268.74,"Y":171.32},{"X":266.05,"Y":163.7098},{"X":261.58004,"Y":157.72},{"X":255.24,"Y":153.24},{"X":247.06,"Y":150.3298},{"X":235.44,"Y":149.13},{"X":224.71,"Y":150.05},{"X":215.91,"Y":153},{"X":210.23002,"Y":156.86},{"X":207.64,"Y":160.85},{"X":207.19,"Y":165.28},{"X":209.34,"Y":169.86},{"X":212.0198,"Y":174.15},{"X":212.14,"Y":177.99},{"X":209.8,"Y":181.78},{"X":204.22,"Y":185.79},{"X":197.62,"Y":187.68},{"X":188.65,"Y":187.43},{"X":182.41,"Y":185.39},{"X":178.45,"Y":181.77},{"X":176.2098,"Y":176.9},{"X":176.03,"Y":170.64},{"X":178.2,"Y":164.13},{"X":183.09,"Y":157.7},{"X":191.04002,"Y":151.36},{"X":202.01,"Y":145.8298},{"X":216.09,"Y":141.57},{"X":232.08,"Y":139.24},{"X":250.07,"Y":139.18},{"X":266.13,"Y":141.23002},{"X":279.05,"Y":145.06},{"X":289.1597,"Y":150.3},{"X":295.9196,"Y":156.19},{"X":300.73,"Y":163.41},{"X":303.85,"Y":172.47},{"X":305.07,"Y":183.78},{"X":305.07,"Y":241.97},{"X":305.08004,"Y":241.97}],[{"X":243.99,"Y":64.95},{"X":255.92,"Y":66.07},{"X":266.21004,"Y":69.28},{"X":274.98,"Y":74.44},{"X":280.65,"Y":80.19},{"X":284.03,"Y":86.85},{"X":285.26,"Y":94.52001},{"X":284.28,"Y":102.84},{"X":281.24,"Y":109.66},{"X":276.0396,"Y":115.43},{"X":267.89,"Y":120.46},{"X":257.68,"Y":123.93},{"X":245.79,"Y":125.33},{"X":232.93,"Y":124.53},{"X":222.21,"Y":121.74},{"X":213.14,"Y":117.11},{"X":207.36,"Y":111.92},{"X":203.7,"Y":105.75},{"X":201.94,"Y":98.18},{"X":202.34,"Y":90.12},{"X":204.86,"Y":83.4},{"X":210.01,"Y":76.81},{"X":217.49,"Y":71.33},{"X":227.17,"Y":67.31},{"X":238.35,"Y":65.2},{"X":243.75,"Y":64.95},{"X":243.99,"Y":64.95}],[{"X":269.99,"Y":212.88},{"X":269.48,"Y":208.76},{"X":266.59003,"Y":208.36},{"X":245.76,"Y":210.86},{"X":230.95,"Y":214.67},{"X":220.9,"Y":219.34},{"X":213.82,"Y":224.85},{"X":209.69,"Y":230.71},{"X":207.92002,"Y":237.03},{"X":208.4,"Y":244.49},{"X":210.86,"Y":250.57},{"X":215.2,"Y":255.08},{"X":221.69,"Y":258.13},{"X":230.57,"Y":259.43},{"X":242.52,"Y":258.58004},{"X":255.27,"Y":255.23},{"X":266.07,"Y":250.04},{"X":269.34003,"Y":247.02},{"X":269.99,"Y":244.81},{"X":269.99,"Y":212.88},{"X":269.99,"Y":212.88}],[{"X":243.63,"Y":73.34},{"X":235.93,"Y":74.4},{"X":230.07,"Y":77.36},{"X":225.65,"Y":82.21001},{"X":223.05,"Y":88.57},{"X":222.41,"Y":96.92},{"X":223.94,"Y":104.53},{"X":227.23,"Y":110.22},{"X":231.99,"Y":114.29},{"X":238.44,"Y":116.65},{"X":246.81,"Y":116.94},{"X":253.73,"Y":115.1},{"X":258.87,"Y":111.5},{"X":262.63,"Y":106.12},{"X":264.65,"Y":98.93},{"X":264.59003,"Y":90.25},{"X":262.47,"Y":83.41},{"X":258.6597,"Y":78.43},{"X":253.37,"Y":75.08},{"X":246.08,"Y":73.43},{"X":243.68,"Y":73.34},{"X":243.63,"Y":73.34}]]';
return {
"ss": deserialize_clipper_poly(glyph),
"cc": deserialize_clipper_poly(grid)
};
}
function get_custom_poly() {
var selected_value = $("#custom_polygons_select").val();
var arr = $.totalStorage('custom_polygons');
if (selected_value !== "") {
//selected_value = parseInt(selected_value,10);
return {
"ss": deserialize_clipper_poly(arr[selected_value].subj),
"cc": deserialize_clipper_poly(arr[selected_value].clip)
};
}
//else return { "ss": '[[{"X":0,"Y":0},{"X":1,"Y":1}]]', "cc": '[[{"X":0,"Y":0},{"X":1,"Y":1}]]' };
else return {
"ss": deserialize_clipper_poly(default_custom_subject_polygon),
"cc": deserialize_clipper_poly(default_custom_clip_polygon)
};
}
function get_random_polys(which, polygon) {
var point_count, polygon_count;
if (which == "subj") {
point_count = rnd_sett.subj_point_count;
polygon_count = rnd_sett.subj_polygon_count;
}
else if (which == "clip") {
point_count = rnd_sett.clip_point_count;
polygon_count = rnd_sett.clip_polygon_count;
}
if (arguments.length === 1) polygon = parseInt($('input[type="radio"][name="polygons"]:checked').val(), 10);
if (polygon != 4 && polygon != 5) return new ClipperLib.Polygons();
var svg = $("#p");
var margin = 10;
rnd_sett.rand_min_x = 0 + margin;
rnd_sett.rand_max_x = parseFloat(svg.attr("width"), 10) - margin;
rnd_sett.rand_min_y = 0 + margin;
rnd_sett.rand_max_y = parseFloat(svg.attr("height"), 10) - margin;
var i, j, pp, np = new ClipperLib.Polygons(),
prev_x = null,
prev_y = null,
horiz_or_vertic = null,
prev_horiz_or_vertic = null;
for (i = 0; i < polygon_count; i++) {
np[i] = new ClipperLib.Polygon();
for (j = 0; j < point_count; j++) {
pp = new ClipperLib.IntPoint();
if (polygon == 4) {
horiz_or_vertic = rnd("int", 0, 1); // 0 = horiz, 1 = vertic
if (prev_horiz_or_vertic === horiz_or_vertic) {
if (horiz_or_vertic === 0) horiz_or_vertic = 1;
else if (horiz_or_vertic === 1) horiz_or_vertic = 0;
else horiz_or_vertic = 0;
}
if (horiz_or_vertic === 0) // horiz => y remains same
{
pp.X = round(rnd("float", rnd_sett.rand_min_x, rnd_sett.rand_max_x));
if (prev_y == null) pp.Y = round(rnd("float", rnd_sett.rand_min_y, rnd_sett.rand_max_y));
else pp.Y = prev_y;
prev_x = pp.X;
prev_y = pp.Y;
prev_horiz_or_vertic = horiz_or_vertic;
}
else // vertic => x remains same
{
pp.Y = round(rnd("float", rnd_sett.rand_min_y, rnd_sett.rand_max_y));
if (prev_x == null) pp.X = round(rnd("float", rnd_sett.rand_min_x, rnd_sett.rand_max_x));
else pp.X = prev_x;
prev_x = pp.X;
prev_y = pp.Y;
prev_horiz_or_vertic = horiz_or_vertic;
}
// last point fix
if (j == point_count - 1 && point_count !== 1) {
if (horiz_or_vertic === 0) // horiz => y remains same
{
pp.X = np[i][0].X;
}
else // vertic => x remains same
{
pp.Y = np[i][0].Y;
}
np[i].push(pp);
}
else np[i].push(pp);
}
else if (polygon == 5) {
pp.X = round(rnd("float", rnd_sett.rand_min_x, rnd_sett.rand_max_x));
pp.Y = round(rnd("float", rnd_sett.rand_min_y, rnd_sett.rand_max_y));
np[i].push(pp);
}
}
prev_x = null;
prev_y = null;
horiz_or_vertic = null;
prev_horiz_or_vertic = null;
}
rnd_sett.scale = scale;
return np;
}
function scale_again_random_poly(poly) {
var i, j;
for (i = 0; i < poly.length; i++) {
for (j = 0; j < poly[i].length; j++) {
poly[i][j].X = round(poly[i][j].X / rnd_sett.scale);
poly[i][j].Y = round(poly[i][j].Y / rnd_sett.scale);
}
}
return poly;
}
function rnd(intfloat, Amin, Amax) {
var num;
if (intfloat == "float") num = (Amin + (Amax - Amin) * Math.random()).toFixed(2);
else if (intfloat == "int") num = Math.floor(Amin + (1 + Amax - Amin) * Math.random());
return num;
}
function get_rect_poly() {
var rect_poly = '[[{"X":100,"Y":100},{"X":200,"Y":100},{"X":200,"Y":200},{"X":100,"Y":200},{"X":100,"Y":100}]]';
rect_poly = '[[{"X":100,"Y":100},{"X":100,"Y":100},{"X":200,"Y":100},{"X":200,"Y":200},{"X":100,"Y":200},{"X":100,"Y":100}]]';
return deserialize_clipper_poly(rect_poly);
}
function round(a) {
if (global_do_not_round_and_scale) return a;
else
return Math.floor(a * scale);
}
window.lsk = 0;
function deserialize_clipper_poly(polystr) {
window.lsk++;
var poly = JSON.parse(polystr);
var i, j, pp, n = [
[]
], m, pm;
var np = new ClipperLib.Polygons();
for (i = 0, m = poly.length; i < m; i++) {
np[i] = new ClipperLib.Polygon();
for (j = 0, pm = poly[i].length; j < pm; j++) {
pp = new ClipperLib.IntPoint();
if (!isNaN(Number(poly[i][j].X)) && !isNaN(Number(poly[i][j].Y))) {
pp.X = round(Number(poly[i][j].X));
pp.Y = round(Number(poly[i][j].Y));
if (benchmark_running) {
if (pp.X > bench.max_point_x) bench.max_point_x = pp.X;
if (pp.Y > bench.max_point_y) bench.max_point_y = pp.Y;
if (pp.X < bench.min_point_x) bench.min_point_x = pp.X;
if (pp.Y < bench.min_point_y) bench.min_point_y = pp.Y;
if (typeof (bench.points["L" + pp.X]) == "undefined") bench.points["L" + pp.X] = scale + ":" + pp.X + ":" + poly[i][j].X;
if (typeof (bench.points["L" + pp.Y]) == "undefined") bench.points["L" + pp.Y] = scale + ":" + pp.Y + ":" + poly[i][j].Y;
}
np[i].push(pp);
}
else return n;
}
}
return np;
}
(function () {
"use strict";
SVG.create = function () {
p = Raphael("svgcontainer", 500, 350);
p.canvas.setAttribute("id", "p");
var str = "<filter id='innerbewel' x0='-50%' y0='-50%' width='200%' height='200%' >";
str += "<feGaussianBlur in='SourceAlpha' stdDeviation='2' result='blur'/>";
str += "<feOffset dy='3' dx='3'/>";
str += "<feComposite in2='SourceAlpha' operator='arithmetic'";
str += " k2='-1' k3='1' result='hlDiff'/>";
str += "<feFlood flood-color='white' flood-opacity='0.8'/>"; // changed to 1.0 for speed
str += "<feComposite in2='hlDiff' operator='in'/>";
str += "<feComposite in2='SourceGraphic' operator='over' result='withGlow'/>";
str += "<feOffset in='blur' dy='-3' dx='-3'/>";
str += "<feComposite in2='SourceAlpha' operator='arithmetic'";
str += " k2='-1' k3='1' result='shadowDiff'/>";
str += "<feFlood flood-color='black' flood-opacity='0.5'/>"; // changed to 1.0 for speed
str += "<feComposite in2='shadowDiff' operator='in'/>";
str += "<feComposite in2='withGlow' operator='over'/>";
str += "</filter>";
var markers = '';
// Markers to show start, mid and end points of path
markers += '<marker id="StartMarker" viewBox="0 0 10 10" refX="0" refY="5" markerUnits="strokeWidth" markerWidth="10" markerHeight="10" stroke="red" stroke-width="1" fill="none" orient="auto">';
markers += '<path d="M0,5L10,5M5,0L10,5M5,10L10,5">';
markers += '</marker>';
markers += '<marker id="MidMarker" stroke-opacity="1.0" viewBox="-1 -1 12 12" refX="5" refY="5" markerWidth="4" markerHeight="4" stroke="red" stroke-width="1" fill="yellow" stroke-opacity="0.5" fill-opacity="0.5" orient="0" markerUnits="strokeWidth">';
markers += '<circle cx="5" cy="5" r="5"></circle>';
markers += '</marker>';
markers += '<marker id="EndMarker" viewBox="0 0 10 10" refX="5" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="8" stroke="blue" stroke-width="1" fill="none" orient="auto">';
markers += '<rect x="0" y="0" width="10" height="10"></rect>';
markers += '</marker>';
// This second invisible marker is needed to avoid noisy after images in Chrome:
markers += '<marker id="StartMarker2" viewBox="0 0 10 10" refX="0" refY="5" markerUnits="strokeWidth" markerWidth="10" markerHeight="10" stroke="none" fill="none" orient="auto">';
markers += '<path d="M0,5L10,5M5,0L10,5M5,10L10,5">';
markers += '</marker>';
markers += '<marker id="MidMarker2" stroke-opacity="1.0" viewBox="-1 -1 12 12" refX="5" refY="5" markerWidth="4" markerHeight="4" stroke="none" fill="none" orient="0" markerUnits="strokeWidth">';
markers += '<circle cx="5" cy="5" r="5"></circle>';
markers += '</marker>';
markers += '<marker id="EndMarker2" viewBox="0 0 10 10" refX="5" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="8" stroke="none" fill="none" orient="auto">';
markers += '<rect x="0" y="0" width="10" height="10"></rect>';
markers += '</marker>';
$("body").append("<svg id='dummy' style='display:none'><defs>" + str + markers + "</defs></svg>");
$("#p defs").append($("#innerbewel"));
$("#p defs").append($("#dummy marker"));
$("#dummy").remove();
return p;
};
SVG.addpaths = function (a, b, c, a1, b1) {
if (a) {
subj_subpolygons = a.length;
a = this.polys2path(a, "1");
if (sub_poly_links_update) {
if (typeof(subj_subpolygons) == "undefined") subj_subpolygons = 0;
$("#subj_subpolygons").html(subj_subpolygons);
$("#subj_points_in_subpolygons").html(this.sub_poly_links);
$("#subj_points_total").html(this.total.toString());
subj_points_total = this.total;
}
}
if (b) {
clip_subpolygons = b.length;
b = this.polys2path(b, "2");
if (sub_poly_links_update) {
if (typeof(clip_subpolygons) == "undefined") clip_subpolygons = 0;
$("#clip_subpolygons").html(clip_subpolygons);
$("#clip_points_in_subpolygons").html(this.sub_poly_links);
$("#clip_points_total").html(this.total.toString());
clip_points_total = this.total;
}
}
if (typeof(c) != "undefined" && typeof(c.length) != "undefined") solution_subpolygons = c.length;
else solution_subpolygons = 0;
if (c) {
c = this.polys2path(c, "3");
if (sub_poly_links_update) {
$("#solution_subpolygons").html(solution_subpolygons);
$("#solution_points_in_subpolygons").html(this.sub_poly_links);
$("#solution_points_total").html(this.total.toString());
solution_points_total = this.total;
}
}
if (sub_poly_links_update) {
$("#points_total").html((subj_points_total + clip_points_total + solution_points_total).toString());
if (isNaN(subj_subpolygons)) subj_subpolygons = 0;
else if (isNaN(clip_subpolygons)) clip_subpolygons = 0;
else if (isNaN(solution_subpolygons)) solution_subpolygons = 0;
$("#all_subpolygons").html(subj_subpolygons + clip_subpolygons + solution_subpolygons);
}
if (a) p1 = p.path(a);
if (b) p2 = p.path(b);
if (c) p3 = p.path(c);
if (a) p1.node.setAttribute("id", "p1");
if (b) p2.node.setAttribute("id", "p2");
if (c) p3.node.setAttribute("id", "p3");
if (c && bevel) $("#p3").attr("filter", "url(#innerbewel)");
else if (c && !bevel) $("#p3").removeAttr("filter");
if (a) $("#p1").removeAttr("fill stroke");
if (b) $("#p2").removeAttr("fill stroke");
if (c) $("#p3").removeAttr("fill stroke");
var PolyFillType = {
pftEvenOdd: 0,
pftNonZero: 1,
pftPositive: 2,
pftNegative: 3
};
if (a) {
if (a1 == PolyFillType.pftEvenOdd) $("#p1").attr("fill-rule", "evenodd");
else $("#p1").attr("fill-rule", "nonzero");
}
if (b) {
if (b1 == PolyFillType.pftEvenOdd) $("#p2").attr("fill-rule", "evenodd");
else $("#p2").attr("fill-rule", "nonzero");
}
// p.setViewBox(bbox.x, bbox.y, bbox.width, bbox.height, true);
// p.setSize(bbox.width, bbox.height);
};
SVG.sub_poly_counts = null;
SVG.sub_poly_links = null;
SVG.total = null;
SVG.polys2path = function (a, fr) {
scaled_paths[fr] = [];
var path = "",
i, j, link, d;
this.sub_poly_counts = [];
this.sub_poly_links = "";
this.total = 0;
if (!scale) scale = 1;
var a_length = a.length;
var a_i_length;
var classi = "subpolylinks";
for (i = 0; i < a_length; i++) {
this.sub_poly_counts.push(a[i].length);
d = "";
a_i_length = a[i].length;
for (j = 0; j < a_i_length; j++) {
this.total++;
if (j == 0) {
d += "M";
}
else {
d += "L";
}
d += (a[i][j].X / scale) + ", " + (a[i][j].Y / scale);
}
d += "Z";
path += d;
if (sub_poly_links_update) {
if (d.trim() == "Z") d = "";
scaled_paths[fr].push(d);
if (benchmark_running) classi = "subpolylinks_disabled";
link = '<span class="' + classi + '" onclick="popup_path(' + i + ',' + fr + ')" onmouseover="show_path(' + i + ',' + fr + ')" onmouseout="hide_path()" >' + a_i_length + '</span>, ';
this.sub_poly_links += link;
}
}
if (sub_poly_links_update) this.sub_poly_links = this.sub_poly_links.substring(0, this.sub_poly_links.length - 2);
if (path.trim() == "Z") path = "";
if (benchmark_running) $(".subpolylinks").removeClass("subpolylinks subpolylinks_disabled").addClass("subpolylinks_disabled");
else $(".subpolylinks_disabled").removeClass("subpolylinks subpolylinks_disabled").addClass("subpolylinks");
return path;
}
window.SVG = SVG;
})()
function toint(a) {
a = parseInt(a, 10);
return a;
}
// Englarges mypath ( = black partially transparent path)
// when clicked
function popup_path(i, fr) {
if (benchmark_running) return false;
if (typeof(i) == "undefined") d = scaled_paths[fr].join(" ");
else d = scaled_paths[fr][i];
var points_string = normalize_clipper_poly(d, true); // quiet
var area;
SolutionPolygonClicked = false;
SolutionPolygonString = "";
if (points_string !== false) {
if (typeof(i) == "undefined") // which means that "Points" column is clicked
{
if (fr == 3) {
$("#polygon_explorer_string_inp").val(format_output(points_string, "ExPolygons"));
SolutionPolygonClicked = true;
SolutionPolygonString = points_string;
}
else {
$("#polygon_explorer_string_inp").val(format_output(points_string));
}
var scaled_paths_length = scaled_paths.length;
var points_str;
area = 0;
for (var j = 0; j < scaled_paths_length; j++) {
points_str = normalize_clipper_poly(scaled_paths[fr][j], true);
if (points_str !== false) {
var polygon = JSON.parse(points_str.replace(/^\[\[/, "[").replace(/\]\]$/, "]"));
area += ClipperLib.Clipper.Area(polygon);
}
}
$("#area").html("Area: " + area);
}
else {
$("#polygon_explorer_string_inp").val(format_output(points_string).replace(/^\[\[/, "[").replace(/\]\]$/, "]"));
area = ClipperLib.Clipper.Area(JSON.parse(points_string.replace(/^\[\[/, "[").replace(/\]\]$/, "]")));
$("#area").html("Area: " + area);
}
}
else {
$("#polygon_explorer_string_inp").val("Some error occurred when parsing polygon points!");
}
$(mypath.node).removeAttr('fill stroke').attr('class', 'svg_mypath');
$(mypath.node).attr('fill-rule', $('#p' + fr).attr('fill-rule'));
$(mypath.node).attr('vector-effect', 'non-scaling-stroke'); // This is not supported in IE 9 and IE10 pre!
var bb = mypath.node.getBBox();
var svg_w = toint($('#p').attr('width'));
var svg_h = toint($('#p').attr('height'));
var x_scale = (svg_w - 20) / bb.width;
var y_scale = (svg_h - 20) / bb.height;
var scal = Math.min(x_scale, y_scale);
var x_trans = -(bb.x + bb.width / 2) + svg_w / 2;
var y_trans = -(bb.y + bb.height / 2) + svg_h / 2;
$('#StartMarker')[0].setAttribute('markerWidth', 10 / scal * 2);
$('#StartMarker')[0].setAttribute('markerHeight', 10 / scal * 2);
$('#MidMarker')[0].setAttribute('markerWidth', 4 / scal * 2);
$('#MidMarker')[0].setAttribute('markerHeight', 4 / scal * 2);
$('#EndMarker')[0].setAttribute('markerWidth', 4 / scal * 2);
$('#EndMarker')[0].setAttribute('markerHeight', 4 / scal * 2);
mypath.animate({
'transform': 't' + x_trans + ' ' + y_trans + 's' + scal + ' ' + scal
}, 500, function () {
$(mypath.node).attr({
'marker-start': 'url(#StartMarker)',
'marker-mid': 'url(#MidMarker)',
'marker-end': 'url(#EndMarker)'
});
});
}
// Shows mypath ( = black partially transparent path)
// when hovered
function show_path(i, fr) {
if (benchmark_running) return false;
var d;
if (typeof(i) == "undefined")
d = scaled_paths[fr].join(" ");
else
d = scaled_paths[fr][i];
mypath = p.path(d);
$(mypath.node).removeAttr('fill stroke').attr('class', 'svg_mypath');
$(mypath.node).attr('fill-rule', $('#p' + fr).attr('fill-rule'));
$(mypath.node).attr('vector-effect', 'non-scaling-stroke');
}
// Hides mypath ( = black partially transparent path)
function hide_path() {
if (mypath == null) return;
$(mypath.node).attr({
'marker-start': 'url(#StartMarker2)',
'marker-mid': 'url(#MidMarker2)',
'marker-end': 'url(#EndMarker2)'
});
if ($(mypath.node).attr('transform')) mypath.animate({
'transform': 's1 1'
}, 500, function () {
this.animate({
'opacity': '0'
}, 500, function () {
this.remove();
});
});
else mypath.animate({
'opacity': '0'
}, 300, function () {
this.remove();
});
}
function set_default_custom_polygon() {
var subj = default_custom_subject_polygon;
var clip = default_custom_clip_polygon;
var def_obj = {
"subj": subj,
"clip": clip
};
var arr = $.totalStorage('custom_polygons');
if (typeof (arr) == "undefined" || arr === null || !isArray(arr) || arr.length == 0) arr = [];
arr[0] = def_obj;
$.totalStorage('custom_polygons', arr);
}
function update_custom_polygons_select() {
var arr = $.totalStorage('custom_polygons');
var selected_value = $("#custom_polygons_select").val();
if (!selected_value && selected_value + "" !== "0") selected_value = 0;
$("#custom_polygons_select option").remove();
var arr_length = 0;
if (isArray(arr)) arr_length = arr.length;
var selected_txt, i;
if (arr_length > 0)
for (i = 0; i < arr_length; i++) {
selected_txt = "";
if (i == selected_value) selected_txt = "selected";
if (arr[i] !== null) $("#custom_polygons_select").append('<option ' + selected_txt + ' value="' + i + '">Poly ' + i + '</option>');
}
else set_default_custom_polygon();
if ($("#custom_polygons_select option").length === 0) set_default_custom_polygon();
// If previously selected value is removed, select the next one
if (arr_length > 0)
if (arr[selected_value] === null) {
for (i = parseInt(selected_value, 10); i < arr_length; i++) {
if (arr[i] !== null) {
$('#custom_polygons_select').val(i);
break;
}
}
}
$('#custom_polygons_select').change();
}
function show_alert(e, obj, txt) {
var x = e.pageX - obj.offsetLeft + $(obj).width() / 2;
var y = e.pageY - obj.offsetTop;
$("#custom_poly_message_box_green")
.stop()
.css({
height: '0px',
opacity: '0',
left: x + 1,
top: y - 1,
width: '0px',
display: 'block'
})
.html("")
.animate({
left: (x + 200),
top: (y - 40),
height: '40px',
width: '200px',
opacity: '0.9'
}, 300, function () {
$(this).html(txt).animate({
opacity: '0.9'
}, 1200, function () {
$(this).animate({
opacity: 0
}, 600, function () {
$(this).css("display", "none");
})
})
});
}
function update_fieldset_heights() {
$("#misc_fieldset").css("min-height", "").css("height", "");
$("#polygon_explorer_fieldset").css("min-height", "").css("height", "");
$("#benchmark_fieldset_f").css("min-height", "").css("height", "");
var td_heights = [];
td_heights.push($("#td0").innerHeight());
td_heights.push($("#td1").innerHeight());
td_heights.push($("#td2").innerHeight());
td_heights.push($("#td3").innerHeight());
var max_td_height = Array_max(td_heights);
var max_index;
var children_heights;
var children_heights_arr = [];
for (var i = 0; i < td_heights.length; i++) {
children_heights = 0;
if (td_heights != max_td_height || 1 == 1) {
$("#td" + i).children().each(function () {
if ($(this).css("display") != "none") {
children_heights += $(this).outerHeight(true);
//console.log(this.type + ":" + $(this).attr("id") + ":" + $(this).outerHeight(true) + ":" + $(this).outerWidth(true));
}
else {
//console.log("----- NOT COUNTED:"+this.type + ":" + $(this).attr("id") + ":" + $(this).outerHeight(true) + ":" + $(this).outerWidth(true));
}
});
}
children_heights_arr.push(children_heights);
//console.log("-------------");
}
var misc_fieldset_height = $("#misc_fieldset").height();
var polygon_explorer_fieldset_height = $("#polygon_explorer_fieldset").height();
var benchmark_fieldset_f_height = $("#benchmark_fieldset_f").height();
misc_fieldset_height += (max_td_height - children_heights_arr[0]);
polygon_explorer_fieldset_height += (max_td_height - children_heights_arr[1]);
benchmark_fieldset_f_height += (max_td_height - children_heights_arr[2]);
$("#misc_fieldset").css("min-height", misc_fieldset_height + "px");
$("#polygon_explorer_fieldset").css("min-height", polygon_explorer_fieldset_height + "px");
$("#benchmark_fieldset_f").css("min-height", benchmark_fieldset_f_height + "px");
}
function resize() {
myresize();
}
function myresize() {
var freeheight = $(window).height();
var polygon_explorer_div_max_height = freeheight - 360;
if (polygon_explorer_div_max_height < 170) polygon_explorer_div_max_height = 170;
// This causes problems when resizing polygon explorer textarea, so have to comment:
//$("#polygon_explorer_div").css("max-height", polygon_explorer_div_max_height +"px");
}
// ADDITIONAL SVG WINDOW STARTS
window.update_enlarged_SVG = false;
window.update_enlarged_SVG_source = false;
window.svg_source_place = "svg_source_container";
function svg_source_enlarge() {
var source = $("#svg_source_textarea").val().replace(/ id=\"/g, ' id="_');
$("#enlarged_svg").html(source);
var original_height = $("#_p").attr("height");
var original_width = $("#_p").attr("width");
// get bbox of all children of svg
$("body").append('<div id="dummy" style="display:block;visibility:hidden"><svg><g id="g123"></g></svg></div>');
$("#g123").append($("#_p").children().clone());
$("#dummy").html($("#dummy").html());
var bb = $("#g123")[0].getBBox();
var g_width = bb.width + 10;
var g_height = bb.height + 10;
var g_x = bb.x - 5;
var g_y = bb.y - 5;
$("#dummy").remove();
/*
$("body").append('<img id="width_img" width="100%" src="">');
var window_width = parseInt(window.getComputedStyle($("#width_img")[0],null).getPropertyValue("width"));
$("#width_img").remove();
*/
$("#_p").attr("viewBox", g_x + " " + g_y + " " + g_width + " " + g_height);
$("#enlarged_svg").html($("#enlarged_svg").html());
$("#enlarged_svg").css("display", "block");
if (svg_source_place == "svg_source_container") // at the bottom
{
$("#_p").attr("width", window_width);
$("#_p").attr("height", parseInt((window_width / original_width) * original_height));
$("#_p1,#_p2,#_p3").css("stroke-width", 0.8 * (original_width / window_width));
}
else // at the right
{
$("#_p").attr("height", window_height);
$("#_p").attr("width", parseInt((window_height / original_height) * original_width));
$("#_p1,#_p2,#_p3").css("stroke-width", 0.8 * (original_height / window_height));
}
$("#svg_source_textarea").css("display", "none");
if (benchmark_running) var disabled = "disabled";
else disabled = "";
$("#svg_source_enlarge_button").html('<button ' + disabled + ' class="textarea_hide_buttons" onClick="show_svg_source_f()" title="Show SVG source">Show SVG source</button>');
update_enlarged_SVG_source = true;
update_enlarged_SVG = true;
}
function show_svg_source_f() {
$("#svg_source_textarea").css("display", "block");
show_svg_source_click("non_click");
$("#enlarged_svg").html("");
if (benchmark_running) var disabled = "disabled";
else disabled = "";
$("#svg_source_enlarge_button").html('<button ' + disabled + ' class="textarea_hide_buttons" onClick="svg_source_enlarge()" title="Show SVG">Show SVG</button>');
update_enlarged_SVG_source = true;
update_enlarged_SVG = false;
}
window_height = $(document).height() * 0.9;
window_width = $(document).width() * 0.9;
function show_svg_source_click(non_click) {
update_enlarged_SVG_source = true;
if (!update_enlarged_SVG) update_enlarged_SVG = false;
if (non_click !== "non_click") {
if ($("#show_svg_source").html() == "Show SVG source B") {
$("#show_svg_source").html("Show SVG source R");
$("#show_svg_source").attr("title", "Show current SVG source in textarea at the right side of the page");
svg_source_place = "svg_source_container";
$("#svg_source_container2").html("");
}
else {
$("#show_svg_source").html("Show SVG source B");
$("#show_svg_source").attr("title", "Show current SVG source in textarea at the bottom of the page");
svg_source_place = "svg_source_container2";
$("#svg_source_container").html("");
}
var textarea_str = '<div id="svg_source_textarea_div">';
textarea_str += '<button class="textarea_hide_buttons" onClick="$(\'#svg_source_textarea_div\').remove();update_enlarged_SVG_source=false;update_enlarged_SVG=false" title="Hide SVG source">Hide</button>';
if (benchmark_running) var disabled = "disabled";
else disabled = "";
textarea_str += '<span id="svg_source_enlarge_button"><button ' + disabled + ' class="textarea_hide_buttons" onClick="svg_source_enlarge()" title="Show SVG">Show SVG</button></span><br>';
textarea_str += '<div style="display:none" id="enlarged_svg"></div>';
textarea_str += '<textarea id="svg_source_textarea"></textarea>';
textarea_str += '</div>';
$("#" + svg_source_place).append(textarea_str);
}
var svg_source = $("#svgcontainer").html().replace(/\>/g, ">\n");
$("#svg_source_textarea").val(svg_source);
}
// ADDITIONAL SVG WINDOW ENDS
// BENCHMARKING STARTS
function benchmark2(i) {
var start_time = new Date().getTime();
var obj = bench_glob[i];
$("#clean").removeAttr('checked');
clean = false;
$("#lighten").removeAttr('checked');
lighten = false;
joinType = obj.joinType;
$('input[type="radio"][name="joinType"][value="' + joinType + '"]').attr('checked', 'checked');
offsettable_poly = obj.offsettable_poly;
$('input[type="radio"][name="offsettable_poly"][value="' + offsettable_poly + '"]').attr('checked', 'checked');
delta = obj.delta;
$('#delta').val(delta);
miterLimit = obj.miterLimit;
$('#miterLimit').val(miterLimit);
AutoFix = obj.AutoFix;
if (AutoFix) $('#AutoFix').attr('checked', 'checked');
else $('#AutoFix').removeAttr('checked');
ExPolygons = obj.ExPolygons;
if (ExPolygons) $('#ExPolygons').attr('checked', 'checked');
else $('#ExPolygons').removeAttr('checked');
Simplify = obj.Simplify;
if (Simplify) $('#Simplify').attr('checked', 'checked');
else $('#Simplify').removeAttr('Simplify');
subject_fillType = obj.subject_fillType;
$("input[name='subject_fillType'][value='" + subject_fillType + "']").attr("checked", "checked");
clip_fillType = obj.clip_fillType;
$("input[name='clip_fillType'][value='" + clip_fillType + "']").attr("checked", "checked");
clipType = obj.clipType;
$("input[name='clipType'][value='" + clipType + "']").attr("checked", "checked");
scale = obj.scale;
$('#scale').val(scale);
if (obj.polygon_id == 4 || obj.polygon_id == 5) rnd_sett = obj.rnd_sett;
$('input[type="radio"][name="polygons"][value="' + obj.polygon_id + '"]').attr('checked', 'checked').trigger("change");
obj = null;
window.last_completed_bench = i;
var end_time = new Date().getTime();
var time = end_time - start_time;
bench_glob[i].measured_time = time;
bench_elapsed_time += time;
bench_glob[i].elapsed_time = bench_elapsed_time;
// update next timeouts
for (var lsk = i + 1; lsk < bench_glob.length; lsk++) {
clearTimeout(bench_glob[lsk].setTimeout);
bench_glob[lsk].setTimeout = setTimeout("benchmark2(" + (lsk) + ")",
bench_glob[lsk].elapsed_time + lsk * bench_glob[lsk].time);
}
var results = '';
var elapsed_time = end_time - bench.list[0].start;
results += Math.floor((i + 1) / bench_glob.length * 100) + " % (";
results += (elapsed_time / 1000).toFixed(1) + " s) of ";
results += "benchmark " + (repeat + 1) + " / " + repeat_times + ". ";
results += "Remaining: ";
results += Math.floor((((elapsed_time / (i + 1)) * (bench_glob.length - i + 1)) / 1000)) + " s.";
$("#benchmark_multiple_status").html(results);
$("#benchmark_multiple_status").css("display", "table-cell");
if (i == 0) {
var multiple_runs_table = bench.print_multiple_runs();
if ($("#benchmark_multiple_table").length) $("#benchmark_multiple_table").remove();
$("#benchmark_multiple_table_cont").append(multiple_runs_table);
}
else if (i == bench_glob.length - 1) {
if (!$("#benchmark_exports_textarea").length) {
var textarea_str = '<div id="benchmark_exports_textarea_div">';
textarea_str += '<button class="textarea_hide_buttons" onClick="$(\'#benchmark_exports_textarea_div\').remove()" title="Hide Benchmark exports">Hide</button><br>';
textarea_str += '<textarea id="benchmark_exports_textarea"></textarea>';
textarea_str += '</div>';
$("#benchmark_exports_container").append(textarea_str);
}
if (benchmark_exports.indexOf("max_point_x") == -1) {
benchmark_exports += "bench.max_point_x:" + bench.max_point_x + "\n";
benchmark_exports += "bench.max_point_y:" + bench.max_point_y + "\n";
benchmark_exports += "bench.min_point_x:" + bench.min_point_x + "\n";
benchmark_exports += "bench.min_point_y:" + bench.min_point_y + "\n";
}
benchmark_exports += bench.totals + ";" + browserg.browser + ";" + browserg.version + "\n";
$("#benchmark_exports_textarea").val(benchmark_exports);
bench.totals_arr_multiple.push(bench.totals_arr[0]);
var multiple_runs_table = bench.print_multiple_runs();
if ($("#benchmark_multiple_table").length) $("#benchmark_multiple_table").remove();
$("#benchmark_multiple_table_cont").append(multiple_runs_table);
repeat++;
if (repeat < repeat_times) {
benchmark_automatic_click = 1;
$("#" + clicked_benchmark_button_id).trigger("click");
}
else {
bench_glob.length = 0;
repeat = 0;
benchmark_running = 0;
ClipperLib.MaxSteps = ClipperLib_MaxSteps_original;
$("#" + clicked_benchmark_button_id).html($("#" + clicked_benchmark_button_id).html().replace("Stop", "Run"));
$("#" + clicked_benchmark_button_id).attr("title", $("#" + clicked_benchmark_button_id).attr("title").replace("Stop", "Execute"));
$("button,input,select").removeAttr('disabled');
$('#sub_poly_links_update').trigger("change");
}
}
}
(function (window) {
var benchmark = function (varname) {
if (typeof (varname) == "string") this.varname = varname;
else this.varname = "";
this.list = [];
this.cats = [];
this.cats.arr = [];
this.type = "benchmark";
this.includeSVG = true;
this.totals = "";
this.totals_arr = [];
this.totals_arr_multiple = [];
this.max_point_x = Number.NEGATIVE_INFINITY;
this.min_point_x = Number.POSITIVE_INFINITY;
this.max_point_y = Number.NEGATIVE_INFINITY;
this.min_point_y = Number.POSITIVE_INFINITY;
this.points = [];
};
// returns index
// cat = category, which name belongs to
// name = code region name or function, which is measured
benchmark.prototype.start = function (cat, name) {
if (cat == "") return;
if (name == "") return;
var b = {};
b.start = new Date().getTime();
b.name = name;
b.cat = cat;
this.list.push(b);
return this.list.length - 1;
}
benchmark.prototype.end = function (index) {
this.list[index].end = new Date().getTime();
this.list[index].time = this.list[index].end - this.list[index].start;
var this_list_cat = this.list[index].cat;
var this_list_cat_counts = this_list_cat + "_counts";
var this_list_cat_time_sum = this_list_cat + "_time_sum"
if (typeof (this.cats[this_list_cat_counts]) == "undefined") this.cats.arr.push(this_list_cat);
if (typeof (this.cats[this_list_cat_time_sum]) == "undefined") this.cats[this_list_cat_time_sum] = 0;
if (typeof (this.cats[this_list_cat_counts]) == "undefined") this.cats[this_list_cat_counts] = 0;
this.cats[this_list_cat_time_sum] += this.list[index].time;
this.cats[this_list_cat_counts]++;
if (typeof (bench_glob) != "undefined" && bench_glob.length > 0) {
this.list[index].bench_glob_index = window.last_completed_bench;
}
}
benchmark.prototype.clear = function () {
this.list = [];
this.cats = [];
this.cats.arr = [];
this.includeSVG = true;
return true;
}
benchmark.prototype.print = function (all) {
var tbl = '<style>';
tbl += '.bench {width:317px;border-collapse:collapse;white-space:nowrap;}';
tbl += '.bench td, .bench th{font-size:12px;text-align:left;border:1px solid #444444; padding:2px}';
tbl += '.bench th{background-color:#DDDDDD;}';
tbl += '.bench tfoot td{background-color:#DDDDDD;}';
tbl += '.bench_foot{font-weight:bold;}';
tbl += '</style>';
tbl += '<table class="bench"><thead><tr>';
tbl += '<th>Num</th>';
tbl += '<th>Name</th>';
tbl += '<th>Category</th>';
tbl += '<th>Time</th>';
tbl += '</tr></thead>';
tbl += '<tbody>';
var time, totaltime = 0,
i, m;
if (this.list && this.list.length) {
m = this.list.length;
// print all
var start_index = 0;
if (!all) {
if (bench_glob.length) start_index = m - 3;
else start_index = m - 16;
if (start_index < 0) start_index = 0;
}
for (i = start_index; i < m; i++) {
time = (this.list[i].time);
tbl += '<tr><td>';
tbl += (i + 1);
tbl += '</td><td>';
tbl += this.list[i].name;
tbl += '</td><td>';
tbl += this.list[i].cat;
tbl += '</td><td>';
tbl += time;
tbl += '</td></tr>';
}
}
tbl += '</tbody>';
tbl += '</table>';
var tbl2 = "";
tbl2 += '<table style="margin-top:10px;margin-bottom:10px" class="bench"><thead><tr>';
tbl2 += '<th>Num</th>';
tbl2 += '<th>Category</th>';
tbl2 += '<th>Calls</th>';
tbl2 += '<th>Sum</th>';
tbl2 += '<th>Avg</th>';
tbl2 += '</tr></thead>';
tbl2 += '<tbody>';
m = this.cats.arr.length;
var item;
this.totals = "";
this.totals_arr = [];
var totals_arr_item = [];
var counts_sum = 0,
cat_time_sum = 0,
cat_time_avg = 0,
this_list_i_cat,
this_list_i_cat_time_sum, this_list_i_cat_counts;
if (m > 0) for (i = 0; i < m; i++) {
tbl2 += '<tr><td>';
item = (i + 1);
tbl2 += item;
tbl2 += '</td><td>';
this_list_i_cat = this.cats.arr[i];
item = this_list_i_cat;
tbl2 += item;
tbl2 += '</td><td>';
this_list_i_cat_counts = this.cats[this_list_i_cat + "_counts"];
item = this_list_i_cat_counts;
tbl2 += item;
counts_sum += this_list_i_cat_counts;
tbl2 += '</td><td>';
this_list_i_cat_time_sum = this.cats[this_list_i_cat + "_time_sum"];
item = this_list_i_cat_time_sum;
tbl2 += item;
cat_time_sum += this_list_i_cat_time_sum;
tbl2 += '</td><td>';
item = (this_list_i_cat_time_sum / this_list_i_cat_counts).toFixed(4);
tbl2 += item;
tbl2 += '</td></tr>';
}
tbl2 += '</tbody>';
if (m > 0) {
tbl2 += '<tfoot><tr><td class="bench_foot" colspan="2">Total</td>';
item = (counts_sum);
totals_arr_item.push(item);
this.totals += item + ";";
tbl2 += '<td>' + item + '</td>';
item = (cat_time_sum);
totals_arr_item.push(item);
this.totals += item + ";";
tbl2 += '<td>' + item + '</td>';
item = (cat_time_sum / counts_sum).toFixed(4);
totals_arr_item.push(item);
this.totals += item;
tbl2 += '<td style="padding: 4px; background-color: hsl(125, 73%, 80%); border: 3px solid black; font-weight: bold; font-size: 16px;">' + item + '</td>';
tbl2 += '</tr>';
tbl2 += '</tfoot>';
}
tbl2 += '</table>';
this.totals_arr.push(totals_arr_item);
tbl += tbl2;
if (benchmark_running) var disabled = "disabled";
else disabled = "";
tbl += '<button ' + disabled + ' onClick="try { ' + this.varname + '.clear();$(\'#benchmark_div\').html(' + this.varname + '.print()); return true; } catch (e) {return false}">Clear Benchmarks</button>';
tbl += '<button ' + disabled + ' onClick="try { $(\'#benchmark_div\').html(' + this.varname + '.print(1)); } catch (e) {return false}">Show all</button>';
return tbl;
}
benchmark.prototype.print_multiple_runs = function (str) {
var tbl2 = "";
tbl2 += '<table id="benchmark_multiple_table" style="margin-top:10px;margin-bottom:10px" class="bench"><thead><tr>';
tbl2 += '<th>Num</th>';
tbl2 += '<th>Calls</th>';
tbl2 += '<th>Sum</th>';
tbl2 += '<th>Avg</th>';
tbl2 += '</tr></thead>';
tbl2 += '<tbody>';
var item, counts_sum = 0,
time_sum = 0;
var times_array = [];
for (var i = 0, m = this.totals_arr_multiple.length; i < m; i++) {
times_array.push(this.totals_arr_multiple[i][1]);
}
var max = Array_max(times_array);
var min = Array_min(times_array);
var range = max - min;
var average = getAverageFromNumArr(times_array, 4);
var stdev = getStandardDeviation(times_array, 4);
var minus_range = min - average;
var plus_range = max - average;
for (var i = 0, m = this.totals_arr_multiple.length; i < m; i++) {
tbl2 += '<tr><td>';
item = (i + 1);
tbl2 += item;
tbl2 += '</td><td>';
item = this.totals_arr_multiple[i][0];
tbl2 += item;
tbl2 += '</td><td>';
item = this.totals_arr_multiple[i][1];
tbl2 += item;
time_sum += item;
tbl2 += '</td><td>';
item = this.totals_arr_multiple[i][2];
tbl2 += item;
tbl2 += '</td></tr>';
}
tbl2 += '<tr><td colspan="4" id="benchmark_multiple_status" style="display:none"></td></tr>';
if (!isNaN(average)) {
tbl2 += '<tr><td colspan="4">';
tbl2 += '<b>Average:</b> ' + average + " ms<br>";
tbl2 += '<b>Min:</b> ' + min + " ms<br>";
tbl2 += '<b>Max:</b> ' + max + " ms<br>";
tbl2 += '<b>Range:</b> ' + range.toFixed(4) + " ms<br>";
tbl2 += '<b>Minus-Range:</b> ' + minus_range.toFixed(4) + " ms<br>";
tbl2 += '<b>Plus-Range:</b> ' + plus_range.toFixed(4) + " ms<br>";
tbl2 += '<b>Stdev:</b> ' + stdev + " ms<br>";
tbl2 += '<b>Range/Average %:</b> ' + (range / average * 100).toFixed(4) + "<br>";
tbl2 += '<b>Minus-Range/Average %:</b> ' + (minus_range / average * 100).toFixed(4) + "<br>";
tbl2 += '<b>Plus-Range/Average %:</b> ' + (plus_range / average * 100).toFixed(4) + "<br>";
tbl2 += '<b>Stdev/Average %:</b> ' + (stdev / average * 100).toFixed(4) + "<br>";
tbl2 += '</td></tr>';
}
tbl2 += '</tbody>';
return tbl2;
}
window.benchmark = benchmark;
})(window);
// Programmer: Larry Battle
// Date: Mar 06, 2011
// Purpose: Calculate standard deviation, variance, and average among an array of numbers.
function isArray(obj) {
return Object.prototype.toString.call(obj) === "[object Array]";
}
;
function getNumWithSetDec(num, numOfDec) {
var pow10s = Math.pow(10, numOfDec || 0);
return (numOfDec) ? Math.round(pow10s * num) / pow10s : num;
}
;
function getAverageFromNumArr(numArr, numOfDec) {
if (!isArray(numArr)) {
return false;
}
var i = numArr.length,
sum = 0;
while (i--) {
sum += numArr[i];
}
return getNumWithSetDec((sum / numArr.length), numOfDec);
}
;
function getVariance(numArr, numOfDec) {
if (!isArray(numArr)) {
return false;
}
var avg = getAverageFromNumArr(numArr, numOfDec),
i = numArr.length,
v = 0;
while (i--) {
v += Math.pow((numArr[i] - avg), 2);
}
v /= numArr.length;
return getNumWithSetDec(v, numOfDec);
}
;
function getStandardDeviation(numArr, numOfDec) {
if (!isArray(numArr)) {
return false;
}
var stdDev = Math.sqrt(getVariance(numArr, numOfDec));
return getNumWithSetDec(stdDev, numOfDec);
}
;
function Array_max(array) {
return Math.max.apply(Math, array);
}
;
function Array_min(array) {
return Math.min.apply(Math, array);
}
;
// BENCHMARKING ENDS
function colorize_boxes_like_in_svg() {
var bg_color = $("#p").css("background-color");
bg_color = new RGBColor(bg_color);
var fill_color, stroke_color, fill_opacity, stroke_opacity, box;
for (var i = 1; i <= 3; i++) {
fill_color = $("#p" + i).css("fill");
fill_color = new RGBColor(fill_color);
fill_opacity = $("#p" + i).css("fill-opacity");
fill_color = fill_color.flattenRGBA(fill_opacity, bg_color);
stroke_color = $("#p" + i).css("stroke");
stroke_color = new RGBColor(stroke_color);
stroke_opacity = $("#p" + i).css("stroke-opacity");
stroke_color = stroke_color.flattenRGBA(stroke_opacity, bg_color);
if (i == 1) box = "#subject_box";
else if (i == 2) box = "#clip_box";
else if (i == 3) box = "#solution_box";
$(box).css("background-color", fill_color);
}
}
// The following class exists only for transferring color values from SVG to html table
// It could be got without this, but wanted to try this.
// May be I remove this and make precalculated values
/**
* A class to parse color values
* @author Stoyan Stefanov <[email protected]>
* @link http://www.phpied.com/rgb-color-parser-in-javascript/
* @license Use it if you like it
*/
function RGBColor(color_string) {
if (typeof(color_string) == "undefined") color_string = "";
this.ok = false;
// strip any leading #
if (color_string.charAt(0) == '#') { // remove # if any
color_string = color_string.substr(1, 6);
}
color_string = color_string.replace(/ /g, '');
color_string = color_string.toLowerCase();
// before getting into regexps, try simple matches
// and overwrite the input
var simple_colors = { aliceblue: 'f0f8ff', antiquewhite: 'faebd7', aqua: '00ffff', aquamarine: '7fffd4', azure: 'f0ffff', beige: 'f5f5dc', bisque: 'ffe4c4', black: '000000', blanchedalmond: 'ffebcd', blue: '0000ff', blueviolet: '8a2be2', brown: 'a52a2a', burlywood: 'deb887', cadetblue: '5f9ea0', chartreuse: '7fff00', chocolate: 'd2691e', coral: 'ff7f50', cornflowerblue: '6495ed', cornsilk: 'fff8dc', crimson: 'dc143c', cyan: '00ffff', darkblue: '00008b', darkcyan: '008b8b', darkgoldenrod: 'b8860b', darkgray: 'a9a9a9', darkgreen: '006400', darkkhaki: 'bdb76b', darkmagenta: '8b008b', darkolivegreen: '556b2f', darkorange: 'ff8c00', darkorchid: '9932cc', darkred: '8b0000', darksalmon: 'e9967a', darkseagreen: '8fbc8f', darkslateblue: '483d8b', darkslategray: '2f4f4f', darkturquoise: '00ced1', darkviolet: '9400d3', deeppink: 'ff1493', deepskyblue: '00bfff', dimgray: '696969', dodgerblue: '1e90ff', feldspar: 'd19275', firebrick: 'b22222', floralwhite: 'fffaf0', forestgreen: '228b22', fuchsia: 'ff00ff', gainsboro: 'dcdcdc', ghostwhite: 'f8f8ff', gold: 'ffd700', goldenrod: 'daa520', gray: '808080', green: '008000', greenyellow: 'adff2f', honeydew: 'f0fff0', hotpink: 'ff69b4', indianred: 'cd5c5c', indigo: '4b0082', ivory: 'fffff0', khaki: 'f0e68c', lavender: 'e6e6fa', lavenderblush: 'fff0f5', lawngreen: '7cfc00', lemonchiffon: 'fffacd', lightblue: 'add8e6', lightcoral: 'f08080', lightcyan: 'e0ffff', lightgoldenrodyellow: 'fafad2', lightgrey: 'd3d3d3', lightgreen: '90ee90', lightpink: 'ffb6c1', lightsalmon: 'ffa07a', lightseagreen: '20b2aa', lightskyblue: '87cefa', lightslateblue: '8470ff', lightslategray: '778899', lightsteelblue: 'b0c4de', lightyellow: 'ffffe0', lime: '00ff00', limegreen: '32cd32', linen: 'faf0e6', magenta: 'ff00ff', maroon: '800000', mediumaquamarine: '66cdaa', mediumblue: '0000cd', mediumorchid: 'ba55d3', mediumpurple: '9370d8', mediumseagreen: '3cb371', mediumslateblue: '7b68ee', mediumspringgreen: '00fa9a', mediumturquoise: '48d1cc', mediumvioletred: 'c71585', midnightblue: '191970', mintcream: 'f5fffa', mistyrose: 'ffe4e1', moccasin: 'ffe4b5', navajowhite: 'ffdead', navy: '000080', oldlace: 'fdf5e6', olive: '808000', olivedrab: '6b8e23', orange: 'ffa500', orangered: 'ff4500', orchid: 'da70d6', palegoldenrod: 'eee8aa', palegreen: '98fb98', paleturquoise: 'afeeee', palevioletred: 'd87093', papayawhip: 'ffefd5', peachpuff: 'ffdab9', peru: 'cd853f', pink: 'ffc0cb', plum: 'dda0dd', powderblue: 'b0e0e6', purple: '800080', red: 'ff0000', rosybrown: 'bc8f8f', royalblue: '4169e1', saddlebrown: '8b4513', salmon: 'fa8072', sandybrown: 'f4a460', seagreen: '2e8b57', seashell: 'fff5ee', sienna: 'a0522d', silver: 'c0c0c0', skyblue: '87ceeb', slateblue: '6a5acd', slategray: '708090', snow: 'fffafa', springgreen: '00ff7f', steelblue: '4682b4', tan: 'd2b48c', teal: '008080', thistle: 'd8bfd8', tomato: 'ff6347', turquoise: '40e0d0', violet: 'ee82ee', violetred: 'd02090', wheat: 'f5deb3', white: 'ffffff', whitesmoke: 'f5f5f5', yellow: 'ffff00', yellowgreen: '9acd32' };
for (var key in simple_colors) {
if (color_string == key) {
color_string = simple_colors[key];
}
}
// emd of simple type-in colors
// array of color definition objects
var color_defs = [
{
re: /^rgb\((\d{1,3}),\s*(\d{1,3}),\s*(\d{1,3})\)$/,
example: ['rgb(123, 234, 45)', 'rgb(255,234,245)'],
process: function (bits) {
return [
parseInt(bits[1], 10),
parseInt(bits[2], 10),
parseInt(bits[3], 10)];
}
},
{
re: /^(\w{2})(\w{2})(\w{2})$/,
example: ['#00ff00', '336699'],
process: function (bits) {
return [
parseInt(bits[1], 16),
parseInt(bits[2], 16),
parseInt(bits[3], 16)];
}
},
{
re: /^(\w{1})(\w{1})(\w{1})$/,
example: ['#fb0', 'f0f'],
process: function (bits) {
return [
parseInt(bits[1] + bits[1], 16),
parseInt(bits[2] + bits[2], 16),
parseInt(bits[3] + bits[3], 16)];
}
}
];
// search through the definitions to find a match
for (var i = 0; i < color_defs.length; i++) {
var re = color_defs[i].re;
var processor = color_defs[i].process;
var bits = re.exec(color_string);
if (bits) {
channels = processor(bits);
this.r = channels[0];
this.g = channels[1];
this.b = channels[2];
this.ok = true;
}
}
// validate/cleanup values
this.r = (this.r < 0 || isNaN(this.r)) ? 0 : ((this.r > 255) ? 255 : this.r);
this.g = (this.g < 0 || isNaN(this.g)) ? 0 : ((this.g > 255) ? 255 : this.g);
this.b = (this.b < 0 || isNaN(this.b)) ? 0 : ((this.b > 255) ? 255 : this.b);
// some getters
this.toRGB = function () {
return 'rgb(' + this.r + ', ' + this.g + ', ' + this.b + ')';
};
this.toHex = function () {
var r = this.r.toString(16);
var g = this.g.toString(16);
var b = this.b.toString(16);
if (r.length == 1) r = '0' + r;
if (g.length == 1) g = '0' + g;
if (b.length == 1) b = '0' + b;
return '#' + r + g + b;
};
this.flattenRGBA = function (a, bg) {
var alpha = 1 - parseFloat(a);
this.r = Math.round((a * (this.r / 255) + (alpha * (bg.r / 255))) * 255);
this.g = Math.round((a * (this.g / 255) + (alpha * (bg.g / 255))) * 255);
this.b = Math.round((a * (this.b / 255) + (alpha * (bg.b / 255))) * 255);
return this.toHex();
};
}
// Not used, because using mouse and repeated events is better
function offset_key(e) {
return;
if (e.which == 38) {
make_offset(1);
}
else if (e.which == 40) {
make_offset(-1);
}
e.stopPropagation();
e.preventDefault();
}
function round_to(num, dec) {
if (typeof (num) == "undefined" || typeof (dec) == "undefined" || isNaN(dec)) {
alert("Cannot round other than number");
return false;
}
else return Math.round(num * Math.pow(10, dec)) / Math.pow(10, dec);
}
function update_enlarged_SVG_if_needed() {
if (update_enlarged_SVG) {
show_svg_source_click("non_click");
svg_source_enlarge();
}
else if (update_enlarged_SVG_source && !update_enlarged_SVG) {
show_svg_source_click("non_click");
}
}
function to_printable(a) {
return a.toFixed(1);
}
function get_polys(scale_again) {
var polygon = parseInt($('input[type="radio"][name="polygons"]:checked').val(), 10);
var polys;
if (polygon === 0) {
polys = get_gb_and_arrow();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 1) {
polys = get_text_polys();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 2) {
polys = get_rectangle_polys();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 3) {
polys = get_same_self_intersecting_polys();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 6) {
polys = get_star_and_rect();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 7) {
polys = get_spiral_and_rects();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 8) {
polys = get_rounded_grid_and_star();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 9) {
polys = get_glyph_and_grid();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 10) {
polys = get_custom_poly();
ss = polys.ss;
cc = polys.cc;
}
else if (polygon == 4 || polygon == 5) {
if (!random_subj) random_subj = get_random_polys("subj");
if (!random_clip) random_clip = get_random_polys("clip");
if (scale_again) {
random_subj = scale_again_random_poly(random_subj);
random_clip = scale_again_random_poly(random_clip);
ss = random_subj;
cc = random_clip;
rnd_sett.scale = scale;
}
else {
ss = random_subj;
cc = random_clip;
}
}
sss = [
[]
];
}
// Main function which setups defaults
// and attach events
// and finally draw the default svg image
function main() {
// formats internal representation of polygons to
// specified output format and prints them on input fields
$("#output_format").change(function () {
var selected_val = parseInt($(this).val(), 10);
output_format = selected_val;
if ($("#custom_polygons_fieldset").css("display") != "none") {
var subj = $("#custom_polygon_subj").val();
var clip = $("#custom_polygon_clip").val();
subj = normalize_clipper_poly(subj, true); // quiet
clip = normalize_clipper_poly(clip, true); // quiet
if (subj !== false && clip !== false) {
$("#custom_polygon_subj").val(format_output(subj));
$("#custom_polygon_clip").val(format_output(clip));
}
}
var polygon_explorer_string = $("#polygon_explorer_string_inp").val();
polygon_explorer_string = normalize_clipper_poly(polygon_explorer_string, true); // quiet
if (polygon_explorer_string !== false) {
$("#polygon_explorer_string_inp").val(format_output(polygon_explorer_string));
}
});
$("#sample_custom_polygon").change(function () {
var polygon = parseInt($(this).val(), 10);
var polys;
global_do_not_round_and_scale = true;
var subj = "",
clip = "";
if (polygon === "") {
subj = "";
clip = "";
}
else if (polygon === 0) {
polys = get_gb_and_arrow();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 1) {
polys = get_text_polys();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 2) {
polys = get_rectangle_polys();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 3) {
polys = get_same_self_intersecting_polys();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 4 || polygon == 5) {
subj = get_random_polys("subj", polygon);
clip = get_random_polys("clip", polygon);
}
else if (polygon == 6) {
polys = get_star_and_rect();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 7) {
polys = get_spiral_and_rects();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 8) {
polys = get_rounded_grid_and_star();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 9) {
polys = get_glyph_and_grid();
subj = polys.ss;
clip = polys.cc;
}
else if (polygon == 10) {
polys = get_custom_poly();
subj = polys.ss;
clip = polys.cc;
}
global_do_not_round_and_scale = false;
if (subj != "") subj = JSON.stringify(subj);
if (clip != "") clip = JSON.stringify(clip);
$("#custom_polygon_subj").val(format_output(subj));
$("#custom_polygon_clip").val(format_output(clip));
});
$("#help_custom_polygon").click(function () {
var txt = "";
txt += 'A) You can add your own custom polygons in several formats:\n\n';
txt += '1) The program uses as an inner default the following format: JSON-stringified array of arrays of point objects eg. [[{"X":100,"Y":100},{"X":200,"Y":100},{"X":200,"Y":200},{"X":100,"Y":200}],[{"X":110,"Y":110},{"X":210,"Y":110},{"X":210,"Y":210},{"X":110,"Y":210}]]. This format allows to input sub polygons. Each sub polygon is an array of point objects. This format makes it easy to transfer polygons to other programs that use Clipper library and is suitable for storing polygons in database.\n\n';
txt += '2) JSON-stringified array of point objects eg. [{"X":100,"Y":100},{"X":200,"Y":100},{"X":200,"Y":200},{"X":100,"Y":200}]. This format doesn\'t allow to input sub polygons.\n\n';
txt += '3) JSON-stringified array of arrays of coordinates without "X" and "Y" eg. [[100,100,200,100,200,200,100,200],[110,110,210,110,210,210,110,210]]. This format allows to input sub polygons. Each sub polygon is an array of coordinates so that each x coordinate is followed by an y coordinate. This format makes it easy to transfer polygons to other programs that use Clipper library and is suitable for storing polygons in database.\n\n';
txt += '4) JSON-stringified array of x and y coordinates eg. [100,100,200,100,200,200,100,200] or [100 100 200 100 200 200 100 200] or [100 100,200 100,200 200,100 200] or the same without []:s. This format doesn\'t allow to input sub polygons.\n\n';
txt += '5) SVG path strings with commands MLVHZ or mlvhz eg. M100,100 L200,100 L200,200 L100,200Z M110,110 L210,110 L210,210 L110,210Z. This format allows to input sub polygons. Each subpolygon starts with M (moveto) command.\n\n';
txt += '\n';
txt += 'B) Custom polygons are saved in browser\'s Local Storage, so they should be tolerant for page reload and browser crashes.\n';
txt += '\n';
alert(txt);
});
$("#help_builtin_polygon_sets").click(function () {
var txt = '';
txt += 'Builtin polygon sets\n\n';
txt += 'You can add builtin polygon sets into Subj and Clip input fields to edit copies of them.\n\n';
txt += 'Note! Before saving, nothing happens. After saving, the SVG window is also updated.';
alert(txt);
});
$("#help_output_format").click(function () {
var txt = '';
txt += 'Output format\n\n';
txt += 'To change the polygon coordinate output format please use the above dropdown. The available formats are:\n\n';
txt += '- Clipper: [[{"X":100,"Y":100},{"X":200,"Y":200}]]\n\n';
txt += '- Plain: [[100,100,200,200]]\n\n';
txt += '- SVG: M100,100L200,200Z\n\n';
txt += 'There are two places where this has effect: 1) the above text box 2) the Subj and Clip text boxes in Custom Polygon fieldset.';
alert(txt);
});
$("#remove_custom_polygon").click(function () {
var selected_value = $("#custom_polygons_select").val();
if (selected_value + "" === "0") {
alert("Cannot remove the default polygon.");
}
else if (selected_value) {
if (confirm("Remove custom polygon " + selected_value + "?")) {
var arr = $.totalStorage('custom_polygons');
arr[selected_value] = null;
$.totalStorage('custom_polygons', arr);
update_custom_polygons_select();
}
}
else alert("Nothing removable.");
});
$("#removeall_custom_polygon").click(function () {
var count = $("#custom_polygons_select option").length;
if (count > 1) {
if (confirm("Remove all " + (count - 1) + " custom polygons?")) {
$.totalStorage('custom_polygons', []);
set_default_custom_polygon();
update_custom_polygons_select();
}
}
else alert("Nothing removable.");
});
$("#save_custom_polygon").click(function (e) {
var subj = $("#custom_polygon_subj").val();
var clip = $("#custom_polygon_clip").val();
subj = normalize_clipper_poly(subj);
clip = normalize_clipper_poly(clip);
if (subj === false || clip === false) return false;
if (typeof ($.totalStorage('custom_polygons')) == "undefined" || $.totalStorage('custom_polygons') === null) {
set_default_custom_polygon();
}
var polygon_set = {};
polygon_set.subj = subj;
polygon_set.clip = clip;
var arr2 = $.totalStorage('custom_polygons');
var selected_value = $("#custom_polygons_select").val();
if (selected_value + "" !== "0" && selected_value) {
arr2[selected_value] = polygon_set;
$.totalStorage('custom_polygons', arr2);
update_custom_polygons_select();
show_alert(e, this, "Polygon " + (selected_value) + " updated!");
}
else if (selected_value + "" === "0") {
alert("The default custom polygon cannot be overwrited. If you want to modify it, save it first as a new.");
}
else alert("Polygon update failed!");
});
$("#add_as_new_custom_polygon").click(function (e) {
var subj = $("#custom_polygon_subj").val();
var clip = $("#custom_polygon_clip").val();
subj = normalize_clipper_poly(subj);
clip = normalize_clipper_poly(clip);
if (subj === false || clip === false) return false;
if (typeof ($.totalStorage('custom_polygons')) == "undefined" || $.totalStorage('custom_polygons') === null) {
set_default_custom_polygon();
}
var polygon_set = {};
polygon_set.subj = subj;
polygon_set.clip = clip;
var arr2 = $.totalStorage('custom_polygons');
arr2.push(polygon_set);
$.totalStorage('custom_polygons', arr2);
update_custom_polygons_select();
$('#custom_polygons_select').val(arr2.length - 1).change();
show_alert(e, this, "New polygon " + (arr2.length - 1) + " added!");
});
$("#custom_polygons_select").change(function () {
var selected_value = $("#custom_polygons_select").val();
if (!selected_value && selected_value + "" !== "0") selected_value = 0;
var arr = $.totalStorage('custom_polygons');
if (isArray(arr) && arr.length && typeof (arr[selected_value]) != "undefined") {
$("#custom_polygon_subj").val(format_output(arr[selected_value].subj));
$("#custom_polygon_clip").val(format_output(arr[selected_value].clip));
make_clip();
}
});
$("input[type='radio'][name='polygons']").change(function (e) {
var val = parseInt($(this).val(), 10);
if (val == 10) {
$("#random_polygons_fieldset").css("display", "none");
$("#custom_polygons_fieldset").css("display", "block");
set_default_custom_polygon();
update_custom_polygons_select();
update_fieldset_heights();
$("#custom_polygons_select").change();
make_clip();
return;
}
else $("#custom_polygons_fieldset").css("display", "none");
if (val == 4 || val == 5) {
$("#random_polygons_fieldset").css("display", "block");
if (val == 4) rnd_sett_defaults.current = "rect";
else rnd_sett_defaults.current = "norm";
// Test for ranges
if (rnd_sett.clip_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count;
if (rnd_sett.clip_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count;
$('#clip_point_count').val(rnd_sett.clip_point_count);
if (rnd_sett.clip_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count;
if (rnd_sett.clip_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count;
$('#clip_polygon_count').val(rnd_sett.clip_polygon_count);
if (rnd_sett.subj_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count;
if (rnd_sett.subj_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count;
$('#subj_point_count').val(rnd_sett.subj_point_count);
if (rnd_sett.subj_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count;
if (rnd_sett.subj_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count;
$('#subj_polygon_count').val(rnd_sett.subj_polygon_count);
random_subj = get_random_polys("subj", val);
random_clip = get_random_polys("clip", val);
}
else $("#random_polygons_fieldset").css("display", "none");
make_clip();
update_fieldset_heights();
});
$('#generate_random_polygons').hold(function (e) {
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$("input[name='clipType']").change(function () {
if ($('input[type="radio"][name="clipType"][value=""]').is(":checked")) {
$('input[type="radio"][name="offsettable_poly"][value="1"]').attr('checked', 'checked');
offsettable_poly = 1;
}
else {
$("#offsettable_poly3").attr('checked', 'checked');
offsettable_poly = 3;
}
clipType = $('input[type="radio"][name="clipType"]:checked').val();
if (clipType !== "") clipType = parseInt(clipType, 10);
make_clip();
});
$("input[name='subject_fillType']").change(function () {
subject_fillType = parseInt(this.value, 10);
make_clip();
});
$("input[name='clip_fillType']").change(function () {
clip_fillType = parseInt(this.value, 10);
make_clip();
});
$("input[name='offsettable_poly']").change(function () {
offsettable_poly = parseInt(this.value, 10);
// When offsettable poly is set to Subject or Clip, then boolean operations are not done.
// To show this to user, set clipType to "No"
if (offsettable_poly == 1 || offsettable_poly == 2) {
$('input[type="radio"][name="clipType"][value=""]').attr('checked', 'checked');
clipType = "";
}
make_clip();
});
$('#scale_minus').hold(function () {
var scale_orig = $('#scale').val();
if (scale_orig && !isNaN(scale_orig) && parseInt(scale_orig, 10).toString() !== "0") scale = parseFloat(scale_orig);
scale = scale - scale_addition;
scale = Math.round(scale / scale_addition) * scale_addition;
if (scale <= 0) scale = 1.0;
$('#scale').val(to_printable(scale));
$('#scale').trigger('change');
}, {
duration: 300,
speed: 0,
min: 150
});
$('#scale').change(function () {
if (this.value && !isNaN(this.value) && parseInt(this.value, 10).toString() !== "0") {
scale = parseFloat(this.value);
get_polys(true);
sss = cc;
make_clip();
}
else alert("Scale has to be a number");
});
$('#scale_plus').hold(function () {
var scale_orig = $('#scale').val();
if (scale_orig && !isNaN(scale_orig) && parseInt(scale_orig, 10).toString() !== "0") scale = parseFloat(scale_orig);
scale = scale + scale_addition;
scale = Math.round(scale / scale_addition) * scale_addition;
$('#scale').val(to_printable(scale));
$('#scale').trigger('change');
}, {
duration: 300,
speed: 0,
min: 150
});
$('#delta_minus').hold(function () {
var delta_orig = $('#delta').val();
if (!isNaN(delta_orig)) delta = parseFloat(delta_orig);
delta = delta - 1;
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#delta').change(function (e) {
var delta_orig = $('#delta').val();
if (!isNaN(delta_orig)) delta = parseFloat(delta_orig);
make_clip();
});
$('#delta_plus').hold(function () {
var delta_orig = $('#delta').val();
if (!isNaN(delta_orig)) delta = parseFloat(delta_orig);
delta = delta + 1;
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$("input[name='joinType']").change(function () {
joinType = parseInt(this.value, 10);
//make_offset();
make_clip();
});
$('#miterLimit_minus').hold(function () {
var miterLimit_orig = $('#miterLimit').val();
if (!isNaN(miterLimit_orig)) miterLimit = parseFloat(miterLimit_orig);
miterLimit = miterLimit - 0.1;
if (miterLimit < 1.0) miterLimit = 1.0;
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$("#miterLimit").change(function () {
miterLimit = parseFloat(this.value);
make_clip();
});
$("#cleandelta").change(function () {
var cleandelta_orig = $(this).val();
var this_value = parseFloat(this.value);
if (!isNaN(this_value)) cleandelta = this_value;
//else cleandelta = cleandelta_orig;
make_clip();
});
$("#lighten_distance").change(function () {
var lighten_distance_orig = $(this).val();
var this_value = parseFloat(this.value);
if (!isNaN(this_value)) lighten_distance = this_value;
//else lighten_distance = lighten_distance_orig;
make_clip();
});
$('#miterLimit_plus').hold(function () {
var miterLimit_orig = $('#miterLimit').val();
if (!isNaN(miterLimit_orig)) miterLimit = parseFloat(miterLimit_orig);
miterLimit = miterLimit + 0.1;
if (miterLimit < 1.0) miterLimit = 1.0;
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$("#AutoFix").change(function () {
if ($(this).attr('checked')) AutoFix = true;
else AutoFix = false;
make_clip();
});
$("#ExPolygons").change(function () {
if ($(this).attr('checked')) ExPolygons = true;
else ExPolygons = false;
if (SolutionPolygonClicked) $("#polygon_explorer_string_inp").val(format_output(SolutionPolygonString, "ExPolygons"));
//make_clip();
});
$("#Simplify").change(function () {
if ($(this).attr('checked')) Simplify = true;
else Simplify = false;
make_clip();
});
$("#clean").change(function () {
if ($(this).attr('checked')) {
if ($("#cleandelta").val() + "".trim() == "") {
cleandelta = cleandelta_default;
$("#cleandelta").val(cleandelta);
}
clean = true;
}
else {
clean = false;
}
make_clip();
});
$("#lighten").change(function () {
if ($(this).attr('checked')) lighten = true;
else lighten = false;
make_clip();
});
$("#lighten").change(function () {
if ($(this).attr('checked')) {
if ($("#lighten_distance").val() + "".trim() == "") {
lighten_distance = lighten_distance_default;
$("#lighten_distance").val(lighten_distance);
}
lighten = true;
}
else {
lighten = false;
}
make_clip();
});
$('#subj_polygon_count_minus').hold(function () {
var subj_polygon_count_orig = $('#subj_polygon_count').val();
if (!isNaN(subj_polygon_count_orig)) rnd_sett.subj_polygon_count = parseFloat(subj_polygon_count_orig);
rnd_sett.subj_polygon_count = rnd_sett.subj_polygon_count - 1;
if (rnd_sett.subj_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count;
$('#subj_polygon_count').val(rnd_sett.subj_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#subj_polygon_count').change(function (e) {
var subj_polygon_count_orig = $('#subj_polygon_count').val();
if (!isNaN(subj_polygon_count_orig)) rnd_sett.subj_polygon_count = parseFloat(subj_polygon_count_orig);
if (rnd_sett.subj_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_polygon_count;
if (rnd_sett.subj_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count;
$('#subj_polygon_count').val(rnd_sett.subj_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
});
$('#subj_polygon_count_plus').hold(function () {
var subj_polygon_count_orig = $('#subj_polygon_count').val();
if (!isNaN(subj_polygon_count_orig)) rnd_sett.subj_polygon_count = parseFloat(subj_polygon_count_orig);
rnd_sett.subj_polygon_count = rnd_sett.subj_polygon_count + 1;
if (rnd_sett.subj_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count) rnd_sett.subj_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_polygon_count;
$('#subj_polygon_count').val(rnd_sett.subj_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#subj_point_count_minus').hold(function () {
var subj_point_count_orig = $('#subj_point_count').val();
if (!isNaN(subj_point_count_orig)) rnd_sett.subj_point_count = parseFloat(subj_point_count_orig);
rnd_sett.subj_point_count = rnd_sett.subj_point_count - 1;
if (rnd_sett.subj_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count;
$('#subj_point_count').val(rnd_sett.subj_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#subj_point_count').change(function (e) {
var subj_point_count_orig = $('#subj_point_count').val();
if (!isNaN(subj_point_count_orig)) rnd_sett.subj_point_count = parseFloat(subj_point_count_orig);
if (rnd_sett.subj_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.subj_point_count;
if (rnd_sett.subj_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count;
$('#subj_point_count').val(rnd_sett.subj_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
});
$('#subj_point_count_plus').hold(function () {
var subj_point_count_orig = $('#subj_point_count').val();
if (!isNaN(subj_point_count_orig)) rnd_sett.subj_point_count = parseFloat(subj_point_count_orig);
rnd_sett.subj_point_count = rnd_sett.subj_point_count + 1;
if (rnd_sett.subj_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count) rnd_sett.subj_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.subj_point_count;
$('#subj_point_count').val(rnd_sett.subj_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#clip_polygon_count_minus').hold(function () {
var clip_polygon_count_orig = $('#clip_polygon_count').val();
if (!isNaN(clip_polygon_count_orig)) rnd_sett.clip_polygon_count = parseFloat(clip_polygon_count_orig);
rnd_sett.clip_polygon_count = rnd_sett.clip_polygon_count - 1;
if (rnd_sett.clip_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count;
$('#clip_polygon_count').val(rnd_sett.clip_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#clip_polygon_count').change(function (e) {
var clip_polygon_count_orig = $('#clip_polygon_count').val();
if (!isNaN(clip_polygon_count_orig)) rnd_sett.clip_polygon_count = parseFloat(clip_polygon_count_orig);
if (rnd_sett.clip_polygon_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_polygon_count;
if (rnd_sett.clip_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count;
$('#clip_polygon_count').val(rnd_sett.clip_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
});
$('#clip_polygon_count_plus').hold(function () {
var clip_polygon_count_orig = $('#clip_polygon_count').val();
if (!isNaN(clip_polygon_count_orig)) rnd_sett.clip_polygon_count = parseFloat(clip_polygon_count_orig);
rnd_sett.clip_polygon_count = rnd_sett.clip_polygon_count + 1;
if (rnd_sett.clip_polygon_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count) rnd_sett.clip_polygon_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_polygon_count;
$('#clip_polygon_count').val(rnd_sett.clip_polygon_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#clip_point_count_minus').hold(function () {
var clip_point_count_orig = $('#clip_point_count').val();
if (!isNaN(clip_point_count_orig)) rnd_sett.clip_point_count = parseFloat(clip_point_count_orig);
rnd_sett.clip_point_count = rnd_sett.clip_point_count - 1;
if (rnd_sett.clip_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count;
$('#clip_point_count').val(rnd_sett.clip_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$('#clip_point_count').change(function (e) {
var clip_point_count_orig = $('#clip_point_count').val();
if (!isNaN(clip_point_count_orig)) rnd_sett.clip_point_count = parseFloat(clip_point_count_orig);
if (rnd_sett.clip_point_count < rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].min.clip_point_count;
if (rnd_sett.clip_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count;
$('#clip_point_count').val(rnd_sett.clip_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
});
$('#clip_point_count_plus').hold(function () {
var clip_point_count_orig = $('#clip_point_count').val();
if (!isNaN(clip_point_count_orig)) rnd_sett.clip_point_count = parseFloat(clip_point_count_orig);
rnd_sett.clip_point_count = rnd_sett.clip_point_count + 1;
if (rnd_sett.clip_point_count > rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count) rnd_sett.clip_point_count = rnd_sett_defaults[rnd_sett_defaults.current].max.clip_point_count;
$('#clip_point_count').val(rnd_sett.clip_point_count);
random_subj = get_random_polys("subj");
random_clip = get_random_polys("clip");
make_clip();
}, {
duration: 300,
speed: 0,
min: 150
});
$("#bevel").change(function () {
if ($(this).attr('checked')) {
bevel = true;
$("#p3").attr("filter", "url(#innerbewel)");
}
else {
bevel = false;
$("#p3").removeAttr("filter");
}
update_enlarged_SVG_if_needed();
});
$("#show_svg_source").click(show_svg_source_click);
$('#sub_poly_links_update').change(function () {
if ($(this).is(':checked')) {
sub_poly_links_update = 1;
if (!p) p = SVG.create();
if (p1) p1.remove();
if (p2) p2.remove();
if (p3) p3.remove();
SVG.addpaths(ss, cc, off_result, subject_fillType, clip_fillType);
}
else {
sub_poly_links_update = 0;
$("#subj_subpolygons, #subj_points_in_subpolygons, #subj_points_total, #clip_subpolygons, #clip_points_in_subpolygons, #clip_points_total, #solution_subpolygons, #solution_points_in_subpolygons, #solution_points_total, #points_total, #all_subpolygons").html("");
}
});
$("#output_format").val(output_format);
myresize();
$("document").mousedown(function () {
ismousedown = true;
});
$("document").mouseup(function () {
ismousedown = false;
});
$("#benchmark1,#benchmark2,#benchmark1b,#benchmark2b").click(function () {
clicked_benchmark_button_id = $(this).attr("id");
if (benchmark_running && !benchmark_automatic_click) {
for (var lsk = 0; lsk < bench_glob.length; lsk++) {
clearTimeout(bench_glob[lsk].setTimeout);
}
bench_glob.length = 0;
repeat = 0;
benchmark_running = 0;
ClipperLib.MaxSteps = ClipperLib_MaxSteps_original;
$("#" + clicked_benchmark_button_id).html($("#" + clicked_benchmark_button_id).html().replace("Stop", "Run"));
$("#" + clicked_benchmark_button_id).attr("title", $("#" + clicked_benchmark_button_id).attr("title").replace("Stop", "Execute"));
$("button,input,select").removeAttr('disabled');
$('#sub_poly_links_update').trigger("change");
return;
}
else {
// disable buttons that are not allowed to click when running
$("button,input,select").attr('disabled', 'disabled');
$("#benchmark1,#benchmark2,#benchmark1b,#benchmark2b").each(function () {
var $this = $(this);
if ($this.attr("id") != clicked_benchmark_button_id) $this.attr('disabled', 'disabled');
else $this.removeAttr('disabled');
});
}
benchmark_automatic_click = 0;
benchmark_running = 1;
ClipperLib.MaxSteps = 10;
$("#" + clicked_benchmark_button_id).html($("#" + clicked_benchmark_button_id).html().replace("Run", "Stop"));
$("#" + clicked_benchmark_button_id).attr("title", $("#" + clicked_benchmark_button_id).attr("title").replace("Execute", "Stop"));
bench.clear();
bench.includeSVG = false;
var scaleLocal;
var polygon_id;
var deltaLocal, clipTypeLocal, offsettable_polyLocal_start, offsettable_polyLocal_end;
var offsettable_polyLocal, joinTypeLocal, miterLimitLocal, miterLimitLocal_start, miterLimitLocal_end;
var fillTypeLocal, fillTypeLocal_start, fillTypeLocal_end;
var timeout_time = 0;
var timeout_time_addition = 100;
window.last_completed_bench = "";
bench_glob.length = 0;
timeout_time = 0;
timeout_time_addition = 0;
latest_time = 10;
window.last_completed_bench = "";
var count;
var deltaLocals = [-5, 0, 10, 30];
var deltaLocal_i, m;
var this_id = $(this).attr("id");
if (this_id == "benchmark1b" || this_id == "benchmark2b") repeat_times = 5;
else repeat_times = 1;
for (count = 0; count < 2; count++) {
if (this_id == "benchmark1" || this_id == "benchmark1b") {
scaleLocal = 100;
if (count === 1) continue;
}
else if (this_id == "benchmark2" || this_id == "benchmark2b") {
scaleLocal = 100000000;
if (count === 1) continue;
}
for (polygon_id = 0; polygon_id < 10; polygon_id++) {
if (!(polygon_id == 0 || polygon_id == 1 || polygon_id == 7 || polygon_id == 8 || polygon_id == 9)) continue;
if (polygon_id == 4 || polygon_id == 5) {
fillTypeLocal_start = 0;
fillTypeLocal_end = 1;
}
else {
fillTypeLocal_start = 1;
fillTypeLocal_end = 1
}
for (fillTypeLocal = fillTypeLocal_start; fillTypeLocal < fillTypeLocal_end + 1; fillTypeLocal++) {
for (clipTypeLocal = 0; clipTypeLocal < 5; clipTypeLocal++) {
if (clipTypeLocal === 0) {
offsettable_polyLocal_start = 1;
offsettable_polyLocal_end = 2;
}
else {
offsettable_polyLocal_start = 3;
offsettable_polyLocal_end = 3;
}
for (offsettable_polyLocal = offsettable_polyLocal_start; offsettable_polyLocal < offsettable_polyLocal_end + 1; offsettable_polyLocal++) {
if (polygon_id === 0 && offsettable_polyLocal == 2) continue;
for (joinTypeLocal = 0; joinTypeLocal < 3; joinTypeLocal++) {
for (deltaLocal_i = 0, m = deltaLocals.length; deltaLocal_i < m; deltaLocal_i++) {
deltaLocal = deltaLocals[deltaLocal_i];
if (joinTypeLocal == 2 && deltaLocal !== 0) {
miterLimitLocal_start = 1;
miterLimitLocal_end = 6;
}
else {
miterLimitLocal_start = 1;
miterLimitLocal_end = 1
}
for (miterLimitLocal = miterLimitLocal_start; miterLimitLocal < miterLimitLocal_end + 1; miterLimitLocal += 2) {
bench_glob[bench_glob.length] = {
polygon_id: polygon_id,
joinType: joinTypeLocal, // 0,1
offsettable_poly: offsettable_polyLocal, // 1,2,3
delta: deltaLocal, // -10 - 10
miterLimit: miterLimitLocal, // 1 - 5
AutoFix: true, // false, true
ExPolygons: false, // false, true
Simplify: false, // false, true
subject_fillType: fillTypeLocal, // 0,1
clip_fillType: fillTypeLocal, //0, 1
clipType: (clipTypeLocal === 0) ? "" : clipTypeLocal - 1, // "",0,1,2,3
scale: scaleLocal, // 100, 100 000, 1000 000 000
rnd_sett: {
clip_polygon_count: 4,
clip_point_count: 4,
subj_polygon_count: 4,
subj_point_count: 4
}
};
timeout_time += timeout_time_addition;
bench_glob[bench_glob.length - 1].setTimeout = setTimeout("benchmark2(" + (bench_glob.length - 1) + ")", timeout_time);
bench_glob[bench_glob.length - 1].timeout_time = timeout_time;
}
}
}
}
}
}
}
}
});
$('#scale').val(to_printable(scale));
$('#delta').val(to_printable(delta));
$('#miterLimit').val(to_printable(miterLimit));
if (AutoFix) $('#AutoFix').attr('checked', 'checked');
else $('#AutoFix').removeAttr('checked');
if (ExPolygons) $('#ExPolygons').attr('checked', 'checked');
else $('#ExPolygons').removeAttr('checked');
if (Simplify) $('#Simplify').attr('checked', 'checked');
else $('#Simplify').removeAttr('checked');
if (clean) $('#clean').attr('checked', 'checked');
else $('#clean').removeAttr('checked');
$("#cleandelta").val(cleandelta_default);
if (lighten) $('#lighten').attr('checked', 'checked');
else $('#lighten').removeAttr('checked');
$('#lighten_distance').val(lighten_distance_default);
$('input[type="radio"][name="subject_fillType"][value="' + subject_fillType + '"]').attr('checked', 'checked');
$('input[type="radio"][name="clip_fillType"][value="' + clip_fillType + '"]').attr('checked', 'checked');
$('input[type="radio"][name="polygons"][value="' + polygons_default + '"]').attr('checked', 'checked').change();
var polygon = parseInt(polygons_default, 10);
if (polygon == 4) rnd_sett = rnd_sett_defaults.rects.default;
else rnd_sett = rnd_sett_defaults.norm.default;
$('input[type="radio"][name="offsettable_poly"][value="' + offsettable_poly + '"]').attr('checked', 'checked');
$('#subj_polygon_count').val(rnd_sett.subj_polygon_count);
$('#subj_point_count').val(rnd_sett.subj_point_count);
$('#clip_polygon_count').val(rnd_sett.clip_polygon_count);
$('#clip_point_count').val(rnd_sett.clip_point_count);
if (bevel) $('#bevel').attr('checked', 'checked');
else $('#bevel').removeAttr('checked');
if (sub_poly_links_update) $('#sub_poly_links_update').attr('checked', 'checked');
else $('#sub_poly_links_update').removeAttr('checked');
make_clip();
colorize_boxes_like_in_svg();
} // main()
function make_offset() {
// --------------------------------
// SELECT OFSETTABLE POLYGON STARTS
// --------------------------------
var off_poly, off_poly1, off_poly2, off_poly3;
offsettable_poly = parseInt($('input[type="radio"][name="offsettable_poly"]:checked').val(), 10);
if (offsettable_poly == 1) {
off_poly = ClipperLib.Clone(ss);
}
else if (offsettable_poly == 2) {
off_poly = ClipperLib.Clone(cc);
}
else if (offsettable_poly == 3) {
off_poly = sss;
}
if (typeof(off_poly) == "undefined" || !(sss instanceof Array)) off_poly = [
[]
];
// ------------------------------
// SELECT OFSETTABLE POLYGON ENDS
// ------------------------------
if (ClipperLib.biginteger_used === null) ClipperLib.biginteger_used = 0;
// -------------
// CLEAN STARTS
// -------------
if (clean) {
off_poly = ClipperLib.Clean(off_poly, cleandelta * scale);
}
//if (isArray(off_poly) && off_poly.length)
//console.log(JSON.stringify(off_poly[0]));
// -------------
// CLEAN ENDS
// -------------
// ---------------
// SIMPLIFY STARTS
// ---------------
// Must simplify before offsetting, to get offsetting right in certain cases.
// Other operations (boolean ones) doesn't need this.
// This is needed when offsetting polygons that has selfintersecting parts ( eg. 5-point star needs this )
if (Simplify) {
// Simplifying is only needed when offsetting original polys, because
// results of boolean operations are already simplified.
// Note! if clip polygon is the same as subject polygon
// then it seems that simplifying is needed also for result of boolean operation (ie. solution).
if (offsettable_poly == 1) // subj
{
off_poly = cpr.SimplifyPolygons(off_poly, subject_fillType);
}
if (offsettable_poly == 2) // clip
{
off_poly = cpr.SimplifyPolygons(off_poly, clip_fillType);
}
if (offsettable_poly == 3) // solution
{
off_poly = cpr.SimplifyPolygons(off_poly, clip_fillType);
if (subject_fillType !== clip_fillType) {
console.log("Subject filltype and Clip filltype are different. We used Clip filltype in SimplifyPolygons().");
}
}
}
// -------------
// SIMPLIFY ENDS
// -------------
// ------------------------------
// ACTUAL OFFSET OPERATION STARTS
// ------------------------------
if (delta) {
for (var i = 0; i < 10; i++) {
cpr.Clear();
cpr.alternateAlgo = false;
var param_delta = round_to(delta * scale, 3);
var param_miterLimit = round_to(miterLimit, 3);
var B0 = bench.start("Offset", "Offset(" + param_delta + ", " + joinType + ", " + param_miterLimit + ", " + AutoFix + ")");
off_result = cpr.OffsetPolygons(off_poly, param_delta, joinType, param_miterLimit, AutoFix);
bench.end(B0);
cpr.Clear();
cpr.alternateAlgo = true;
var param_delta = round_to(delta * scale, 3);
var param_miterLimit = round_to(miterLimit, 3);
var B0 = bench.start("Offset bulk insert", "Offset(" + param_delta + ", " + joinType + ", " + param_miterLimit + ", " + AutoFix + ")");
off_result = cpr.OffsetPolygons(off_poly, param_delta, joinType, param_miterLimit, AutoFix);
bench.end(B0);
}
}
else {
off_result = off_poly;
}
// ----------------------------
// ACTUAL OFFSET OPERATION ENDS
// ------------------------------
// ----------------------------
// LIGHTENING STARTS
// ------------------------------
if (lighten) {
off_result = ClipperLib.Lighten(off_result, lighten_distance * scale);
// Because lighten may produce self-intersections,
// must Simplify to be sure that result is free of them,
// but only if user wants
if (Simplify) {
off_result = cpr.SimplifyPolygons(off_result, subject_fillType);
}
}
// ----------------------------
// LIGHTENING ENDS
// ------------------------------
// --------------------------------
// SVG UPDATE STARTS
// --------------------------------
if (!p) p = SVG.create();
if (p1) p1.remove();
if (p2) p2.remove();
if (p3) p3.remove();
if (bench.includeSVG) var B3 = bench.start("SVG", "addpaths(ss, cc, off_result, " + subject_fillType + ", " + clip_fillType + ")");
SVG.addpaths(ss, cc, off_result, subject_fillType, clip_fillType);
if (bench.includeSVG) bench.end(B3);
// --------------------------------
// SVG UPDATE ENDS
// --------------------------------
// --------------------------------
// UPDATE BIGINTEGERS TOGGLE STARTS
// --------------------------------
if (ClipperLib.biginteger_used !== null) {
if (ClipperLib.biginteger_used) $("#biginteger_used").html("true");
else $("#biginteger_used").html("false");
}
else $("#biginteger_used").html("unknown");
ClipperLib.biginteger_used = null;
// ------------------------------
// UPDATE BIGINTEGERS TOGGLE ENDS
// ------------------------------
// UPDATE DELTA TO FORM
$('#delta').val(to_printable(delta));
// UPDATE MITERLIMIT TO FORM
$('#miterLimit').val(to_printable(miterLimit));
// PRINT BENCHMARKS
$("#benchmark_div").html(bench.print());
update_enlarged_SVG_if_needed();
update_fieldset_heights();
}
function make_clip() {
ClipperLib.biginteger_used = null;
if (!cpr) {
cpr = new ClipperLib.Clipper();
}
else cpr.Clear();
if (!cpr2) {
cpr2 = new ClipperLib.Clipper();
}
else cpr2.Clear();
get_polys();
if (clipType !== "" && offsettable_poly == 3) {
for (var i = 0; i < 10; i++) {
cpr.Clear()
cpr.AddPolygons(ss, ClipperLib.PolyType.ptSubject);
cpr.AddPolygons(cc, ClipperLib.PolyType.ptClip);
sss = new ClipperLib.Polygons();
cpr.alternateAlgo = false;
var B1 = bench.start("Boolean", "Execute(" + clipType + ", sss, " + subject_fillType + ", " + clip_fillType + ")");
cpr.Execute(clipType, sss, subject_fillType, clip_fillType);
bench.end(B1);
cpr2.Clear();
cpr2.AddPolygons(ss, ClipperLib.PolyType.ptSubject);
cpr2.AddPolygons(cc, ClipperLib.PolyType.ptClip);
sss2 = new ClipperLib.Polygons();
cpr2.alternateAlgo = true;
var B2 = bench.start("Boolean bulk insert", "Execute(" + clipType + ", sss2, " + subject_fillType + ", " + clip_fillType + ")");
cpr2.Execute(clipType, sss2, subject_fillType, clip_fillType);
bench.end(B2);
}
}
make_offset();
}
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment