diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..545482d
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+original/
+respacks/
diff --git a/css/hues-m.css b/css/hues-m.css
new file mode 100644
index 0000000..42e62d6
--- /dev/null
+++ b/css/hues-m.css
@@ -0,0 +1,313 @@
+/* Heavily based on Kepstin's wonderful CSS work
+ https://github.com/kepstin/0x40hues-html5/blob/master/hues-m.css */
+
+.ModernUI {
+ font-family: 'PetMe64Web';
+}
+
+.hues-m-beatbar {
+ position: absolute;
+ top: 0;
+ max-width: 992px;
+ height: 30px;
+ margin: 0 auto;
+ overflow: hidden;
+ left: 8px;
+ right: 8px;
+ color: white;
+ background: rgba(127,127,127,0.5);
+ border-color: rgba(0,0,0,0.5);
+ border-width: 0 4px 4px;
+ border-style: solid;
+}
+
+.hues-m-beatleft, .hues-m-beatright, .hues-m-songtitle, .hues-m-imagename, .hues-m-huename {
+ color: white;
+ background: rgba(0,0,0,0.7);
+ height: 20px;
+ line-height: 20px;
+ font-size: 12px;
+ overflow: hidden;
+ white-space: nowrap;
+ border-radius: 10px;
+}
+
+.hues-m-huename {
+ font-size: 8px;
+ height: 12px;
+ line-height: 12px;
+ border-radius: 10px;
+}
+
+.hues-m-beatleft, .hues-m-beatright {
+ position: absolute;
+ padding: 0 0 0 20px;
+ top: 5px;
+ overflow: hidden;
+ border-radius: 0 10px 10px 0;
+}
+.hues-m-beatleft {
+ transform: scaleX(-1);
+ -webkit-transform: scaleX(-1);
+ left: 8px;
+ right: 50%;
+}
+.hues-m-beatright {
+ left: 50%;
+ right: 8px;
+}
+
+.hues-m-beatcenter {
+ position: absolute;
+ top: -6px;
+ left: 0;
+ right: 0;
+ margin: 0 auto;
+ height: 40px;
+ width: 40px;
+ color: white;
+ background: rgb(80,80,80);
+ font-size: 20px;
+ line-height: 40px;
+ border-radius: 20px;
+ text-align: center;
+ box-shadow: inset 0 0 12px rgba(0,0,0,0.5);
+}
+
+.hues-m-beatcenter > span {
+ -moz-animation-duration: 150ms;
+ -webkit-animation-duration: 150ms;
+ animation-duration: 150ms;
+ -moz-animation-name: hues-m-beatcenter;
+ -webkit-animation-name: hues-m-beatcenter;
+ animation-name: hues-m-beatcenter;
+ -moz-animation-fill-mode: forwards;
+ -webkit-animation-fill-mode: forwards;
+ animation-fill-mode: forwards;
+}
+@-moz-keyframes hues-m-beatcenter {
+ from {
+ opacity: 1;
+ }
+ 50% {
+ opacity: 1;
+ }
+ to {
+ opacity: 0;
+ }
+}
+@-webkit-keyframes hues-m-beatcenter {
+ from {
+ opacity: 1;
+ }
+ 50% {
+ opacity: 1;
+ }
+ to {
+ opacity: 0;
+ }
+}
+@keyframes hues-m-beatcenter {
+ from {
+ opacity: 1;
+ }
+ 50% {
+ opacity: 1;
+ }
+ to {
+ opacity: 0;
+ }
+}
+
+.hues-m-controls {
+ position: absolute;
+ bottom: 0;
+ max-width: 992px;
+ height: 104px;
+ margin: 0 auto;
+ overflow: hidden;
+ left: 8px;
+ right: 8px;
+ color: rgba(255,255,255,0.7);
+ background: rgba(127,127,127,0.5);
+ border-color: rgba(0,0,0,0.5);
+ border-width: 4px 4px 0;
+ border-style: solid;
+}
+
+.hues-m-songtitle, .hues-m-imagename, .hues-m-huename {
+ position: absolute;
+ left: 8px;
+ right: 8px;
+ text-align: center;
+ padding: 0 4px;
+}
+.hues-m-songtitle {
+ bottom: 5px;
+}
+.hues-m-imagename {
+ bottom: 29px;
+}
+
+.hues-m-songtitle > a:link, .hues-m-songtitle > a:visited, .hues-m-imagename > a:link, .hues-m-imagename > a:visited {
+ display: block;
+ color: inherit;
+ text-decoration: none;
+ overflow: hidden;
+}
+.hues-m-songtitle > a.small, .hues-m-imagename > a.small {
+ font-size: 10px;
+}
+.hues-m-songtitle > a.x-small, .hues-m-imagename > a.x-small {
+ font-size: 8px;
+}
+
+.hues-m-leftbox {
+ position: absolute;
+ bottom: 50px;
+ left: 0;
+ right: 50%;
+ height: 54px;
+}
+
+.hues-m-rightbox {
+ position: absolute;
+ bottom: 50px;
+ left: 50%;
+ right: 0;
+ height: 54px;
+}
+
+.hues-m-huename {
+ bottom: 5px;
+}
+
+.hues-m-vol-bar {
+ position: absolute;
+ height: 20px;
+ bottom: 21px;
+ left: 32px;
+ right: 32px;
+}
+
+.hues-m-vol-bar > button {
+ display: block;
+ position: absolute;
+ left: 0;
+ bottom: 14px;
+ right: 0;
+ height: 12px;
+ color: inherit;
+ font: inherit;
+ font-size: 12px;
+ line-height: 12px;
+ text-align: center;
+ padding: 0;
+ width: 100%;
+ background: transparent;
+ border: none;
+}
+.hues-m-vol-bar > input {
+ display: block;
+ position: absolute;
+ left: 0;
+ bottom: 0;
+ right: 0;
+ height: 12px;
+}
+
+/* Fun slider stuff! */
+
+input[type=range] {
+ width: 100%;
+ margin: 0;
+ padding: 0;
+ height: 12px;
+ background: transparent;
+ -webkit-appearance: none;
+}
+
+input[type=range]::-webkit-slider-runnable-track {
+ width: 100%;
+ height: 4px;
+ background: rgba(255,255,255,0.7);
+ border: none;
+ border-radius: 0;
+}
+
+input[type=range]::-webkit-slider-thumb {
+ -webkit-appearance: none;
+ box-shadow: none;
+ border: none;
+ height: 12px;
+ width: 4px;
+ border-radius: 0;
+ background: rgb(255,255,255);
+ margin-top: -4px; /* You need to specify a margin in Chrome, but in Firefox and IE it is automatic */
+}
+
+input[type=range]::-moz-range-track {
+ width: 100%;
+ height: 4px;
+ background: rgba(255,255,255,0.7);
+ border: none;
+ border-radius: 0;
+}
+
+input[type=range]::-moz-range-thumb {
+ box-shadow: none;
+ border: none;
+ height: 12px;
+ width: 4px;
+ border-radius: 0;
+ background: rgb(255,255,255);
+}
+
+input[type=range]::-ms-track {
+ width: 100%;
+ background: transparent; /* Hides the slider so custom styles can be added */
+ border-color: transparent;
+ color: transparent;
+ height: 4px;
+ border-width: 4px 0;
+}
+
+input[type=range]::-ms-fill-lower {
+ background: rgba(255,255,255,0.7);
+}
+input[type=range]::-ms-fill-upper {
+ background: rgba(0,0,0,0.7);
+}
+
+input[type=range]::-ms-thumb {
+ box-shadow: none;
+ border: none;
+ height: 12px;
+ width: 4px;
+ border-radius: 0;
+ background: rgb(255,255,255);
+}
+
+@media (min-width: 768px) {
+ .hues-m-controls {
+ height: 54px;
+ }
+ .hues-m-songtitle, .hues-m-imagename {
+ left: 192px;
+ right: 192px;
+ }
+ .hues-m-leftbox {
+ bottom: 0;
+ left: 0;
+ right: auto;
+ width: 192px;
+ height: 54px;
+ }
+ .hues-m-rightbox {
+ bottom: 0;
+ left: auto;
+ right: 0;
+ width: 192px;
+ height: 54px;
+ }
+}
diff --git a/css/hues-r.css b/css/hues-r.css
new file mode 100644
index 0000000..95fc8b8
--- /dev/null
+++ b/css/hues-r.css
@@ -0,0 +1,20 @@
+/* TODO: Fix scrollbar */
+
+.RetroUI {
+}
+
+.hues-r-container {
+ position: absolute;
+ bottom: 0px;
+
+ white-space: nowrap;
+ font-family: 'PetMe64Web';
+ font-size: 5pt;
+}
+
+.hues-r-container a:link, .hues-r-container a:visited {
+ display: block;
+ color: inherit;
+ text-decoration: none;
+ overflow: hidden;
+}
\ No newline at end of file
diff --git a/css/hues-w.css b/css/hues-w.css
new file mode 100644
index 0000000..b7f8517
--- /dev/null
+++ b/css/hues-w.css
@@ -0,0 +1,53 @@
+/* TODO: Fix scrollbar */
+
+.WeedUI {
+ font-family: 'PetMe64Web';
+}
+
+.hues-w-beatleft, .hues-w-beatright {
+ font-size: 13px;
+ position: absolute;
+ padding: 0 0 0 5px;
+ top: 5px;
+ overflow: hidden;
+ border-radius: 0 10px 10px 0;
+}
+.hues-w-beatleft {
+ transform: scaleX(-1);
+ -webkit-transform: scaleX(-1);
+ left: 8px;
+ right: 50%;
+}
+.hues-w-beatright {
+ left: 50%;
+ right: 8px;
+}
+
+.hues-w-beataccent {
+ position: absolute;
+ left: 0; right: 0;
+ margin-left:auto;
+ margin-right:auto;
+ margin-top: 15px;
+ text-align:center;
+ font-size: 35px;
+ opacity: 0;
+ text-shadow: -2px 2px 0px #666;
+
+ animation-name: fallspin;
+ animation-timing-function: cubic-bezier(0.215, 0.61, 0.355, 1);
+ animation-duration: 0.5s;
+
+ -webkit-animation-name: fallspin;
+ -webkit-animation-timing-function: cubic-bezier(0.215, 0.61, 0.355, 1);
+ -webkit-animation-duration: 0.5s;
+}
+
+@keyframes fallspin {
+ from {transform: rotate(0deg) translate(0px, 0px);
+ opacity: 1;}
+}
+@-webkit-keyframes fallspin {
+ from {-webkit-transform: rotate(0deg) translate(0px, 0px);
+ opacity: 1;}
+}
\ No newline at end of file
diff --git a/css/hues-x.css b/css/hues-x.css
new file mode 100644
index 0000000..8689e21
--- /dev/null
+++ b/css/hues-x.css
@@ -0,0 +1,30 @@
+/* Heavily based on Kepstin's wonderful CSS work
+ https://github.com/kepstin/0x40hues-html5/blob/master/hues-m.css */
+
+.XmasUI {
+ font-family: 'PetMe64Web';
+}
+
+.hues-x-controls {
+ position: absolute;
+ bottom: 0;
+ max-width: 992px;
+ height: 50px;
+ margin: 0 auto;
+ overflow: hidden;
+ left: 8px;
+ right: 8px;
+ color: rgba(255,255,255,0.7);
+}
+
+.hues-x-beatbar {
+ position: absolute;
+ top: 0;
+ max-width: 992px;
+ height: 30px;
+ margin: 0 auto;
+ overflow: hidden;
+ left: 8px;
+ right: 8px;
+ color: white;
+}
\ No newline at end of file
diff --git a/css/style.css b/css/style.css
new file mode 100644
index 0000000..98da097
--- /dev/null
+++ b/css/style.css
@@ -0,0 +1,74 @@
+@font-face {
+ font-family: 'PetMe64Web';
+ font-style: normal;
+ font-weight: normal;
+ -webkit-font-smoothing: none;
+ font-smooth: never;
+ src: local("PetMe64Web");
+ src: local('Pet Me 64'), local('Pet Me 64'), url("PetMe64.woff") format('woff');
+}
+
+html, body {
+ height: 100%;
+ margin: 0; padding: 0;
+ overflow: hidden;
+ background-color:#ffffff;
+}
+
+#waifu {
+ display: block;
+ height: 100%;
+ padding: 0;
+
+ z-index: -10;
+}
+
+#preloadHelper {
+ background-color: #FFF;
+ width: 100%;
+ height: 100%;
+ display:flex;
+ justify-content:center;
+ align-items:center;
+ flex-direction: column;
+ font-family: 'PetMe64Web';
+ font-size: 25pt;
+
+ position: absolute;
+ top: 0;
+ left: 0;
+ z-index: 10;
+}
+
+#preloadHelper.loaded {
+ opacity: 0;
+ animation-name: fadeout;
+ animation-duration: 1s;
+ -webkit-animation-name: fadeout;
+ -webkit-animation-duration: 3s;
+}
+
+#preloader {
+ display: block;
+ text-align: center;
+}
+
+#preSub {
+ font-size: 12pt;
+}
+
+#huesSettings {
+ background: rgba(127,127,127,0.5);
+ border-color: rgba(0,0,0,0.5);
+ border-width: 4px;
+ border-style: solid;
+}
+
+@keyframes fadeout {
+ from {opacity: 1;}
+ to {opacity: 0;}
+}
+@-webkit-keyframes fadeout {
+ from {opacity: 1;}
+ to {opacity: 0;}
+}
\ No newline at end of file
diff --git a/index.html b/index.html
index ee1aa9f..4c8728a 100644
--- a/index.html
+++ b/index.html
@@ -1,385 +1,45 @@
-
+
-
-
-
X
-
+
+
TODO
+
-
-
stop
-
play
-
prev
-
next
-
B=0x0000
-
T=0x0000
-
Madeon - Finale
-
Megumi
-
white
-
\ No newline at end of file
diff --git a/js/HuesCanvas.js b/js/HuesCanvas.js
new file mode 100644
index 0000000..e212429
--- /dev/null
+++ b/js/HuesCanvas.js
@@ -0,0 +1,303 @@
+
+/* Takes an element name to attach to, and an audio context element for
+ getting the current time with reasonable accuracy */
+function HuesCanvas(element, aContext, core) {
+ 'use strict';
+ this.aContext = aContext;
+ this.core = core;
+
+ this.needsRedraw = false;
+ this.colour = 0xFFFFFF;
+ this.image = null;
+
+ this.animTimeout;
+ this.animFrame;
+
+ // set later
+ this.blurDecay;
+ this.blurAmount;
+ this.blurIterations;
+ this.blurMin;
+ this.blurMax;
+ this.blurDelta;
+ this.blurAlpha;
+ // dynamic
+ this.blurStart = 0;
+ this.blurDistance = 0;
+ this.xBlur = false;
+ this.yBlur = false;
+
+ this.blackout = false;
+ this.blackoutColour = "#000"; // for the whiteout case we must store this
+ this.blackoutTimeout;
+
+ this.colourFade = false;
+ this.colourFadeStart=0;
+ this.colourFadeLength=0;
+ this.oldColour=0xFFFFFF;
+ this.newColour=0xFFFFFF;
+
+ this.blendMode = "hard-light";
+ // Chosen because they look decent
+ this.setBlurAmount(15);
+ this.setBlurIterations(5);
+ this.setBlurDecay(25);
+ this.canvas = document.getElementById(element).getContext("2d");
+ window.addEventListener('resize', this.resizeHandler(this));
+ this.resize();
+
+ this.animating = true;
+ requestAnimationFrame(this.getAnimLoop());
+}
+
+HuesCanvas.prototype.resizeHandler = function(that) {
+ return function() {that.resize();};
+}
+
+HuesCanvas.prototype.resize = function() {
+ // height is constant 720px, we expand width to suit
+ var ratio = window.innerWidth / window.innerHeight;
+ this.canvas.canvas.width = 720 * ratio + 1;
+ this.needsRedraw = true;
+}
+
+HuesCanvas.prototype.redraw = function() {
+ var offset; // for centering/right/left align
+ var bOpacity;
+ var width = this.canvas.canvas.width;
+
+ var cTime = this.aContext.currentTime;
+ // white BG for the hard light filter
+ this.canvas.globalAlpha = 1;
+ this.canvas.globalCompositeOperation = "source-over";
+ if(this.blackout) {
+ // original is 3 frames at 30fps, this is close
+ bOpacity = (cTime - this.blackoutStart)*10;
+ if(bOpacity > 1) { // optimise the draw
+ this.canvas.fillStyle = this.blackoutColour;
+ this.canvas.fillRect(0,0,width,720);
+ this.needsRedraw = false;
+ return;
+ }
+ } else {
+ this.canvas.fillStyle = "#FFF";
+ this.canvas.fillRect(0,0,width,720);
+ }
+
+ if(this.image) {
+ var bitmap = this.image.animated ?
+ this.image.bitmaps[this.animFrame] : this.image.bitmap;
+ switch(this.image.align) {
+ case "left":
+ offset = 0;
+ break;
+ case "right":
+ offset = width - bitmap.width;
+ break;
+ default:
+ offset = width/2 - bitmap.width/2;
+ break;
+ }
+ if(this.xBlur || this.yBlur) {
+ this.canvas.globalAlpha = this.blurAlpha;
+ var delta = cTime - this.blurStart;
+ this.blurDistance = this.blurAmount * Math.exp(-this.blurDecay * delta);
+ }
+ if(this.xBlur) {
+ for(var i=this.blurMin; i<=this.blurMax; i+= this.blurDelta) {
+ this.canvas.drawImage(bitmap, Math.floor(this.blurDistance * i) + offset, 0);
+ }
+ } else if(this.yBlur) {
+ for(var i=this.blurMin; i<=this.blurMax; i+= this.blurDelta) {
+ this.canvas.drawImage(bitmap, offset, Math.floor(this.blurDistance * i));
+ }
+ } else {
+ this.canvas.globalAlpha = 1;
+ this.canvas.drawImage(bitmap, offset, 0);
+ }
+ }
+ this.canvas.globalAlpha = 0.7;
+ this.canvas.fillStyle = this.intToHex(this.colour);
+ this.canvas.globalCompositeOperation = this.blendMode;
+ this.canvas.fillRect(0,0,width,720);
+ if(this.blackout) {
+ this.canvas.globalAlpha = bOpacity;
+ this.canvas.fillStyle = this.blackoutColour;
+ this.canvas.fillRect(0,0,width,720);
+ this.needsRedraw = true;
+ } else {
+ this.needsRedraw = false;
+ }
+}
+
+/* Second fastest method from
+ http://stackoverflow.com/questions/10073699/pad-a-number-with-leading-zeros-in-javascript
+ It stil does millions of ops per second, and isn't ugly like the integer if/else */
+HuesCanvas.prototype.intToHex = function(num) {
+ return '#' + ("00000"+num.toString(16)).slice(-6);
+}
+
+HuesCanvas.prototype.getAnimLoop = function() {
+ var that = this;
+ return function() {that.animationLoop()};
+}
+
+HuesCanvas.prototype.animationLoop = function() {
+ if (this.colourFade) {
+ var delta = this.aContext.currentTime - this.colourFadeStart;
+ var fadeVal = delta / this.colourFadeLength;
+ if (fadeVal >= 1) {
+ this.stopFade();
+ this.colour = this.newColour;
+ } else {
+ this.mixColours(fadeVal);
+ }
+ this.needsRedraw = true;
+ }
+ if(this.blackoutTimeout && this.aContext.currentTime > this.blackoutTimeout) {
+ this.clearBlackout();
+ }
+ if(this.image && this.image.animated
+ && this.animTimeout < this.aContext.currentTime) {
+ this.animFrame++;
+ this.animFrame %= this.image.frameDurations.length;
+ this.animTimeout = this.aContext.currentTime +
+ this.image.frameDurations[this.animFrame]/1000;
+ this.needsRedraw = true;
+ }
+ if(this.blurStart) {
+ var dist = this.blurDistance / this.blurAmount;
+ if(this.xBlur)
+ this.core.blurUpdated(dist, 0);
+ else
+ this.core.blurUpdated(0, dist);
+ }
+ if(this.blurStart && this.blurDistance < 0.3) {
+ this.core.blurUpdated(0, 0);
+ this.blurDistance = 0;
+ this.blurStart = 0;
+ this.xBlur = this.yBlur = false;
+ this.redraw();
+ } else if(this.blurStart) {
+ this.redraw();
+ } else if(this.needsRedraw){
+ this.redraw();
+ }
+ if(this.animating) {
+ requestAnimationFrame(this.getAnimLoop());
+ }
+}
+
+HuesCanvas.prototype.setImage = function(image) {
+ this.needsRedraw = true;
+ this.image = image;
+ // Null images don't need anything interesting done to them
+ if(!image || (!image.bitmap && !image.bitmaps)) {
+ return;
+ }
+ if(image.animated) {
+ this.animFrame = 0;
+ this.animTimeout = this.aContext.currentTime + image.frameDurations[0]/1000;
+ }
+}
+
+HuesCanvas.prototype.setColour = function(colour) {
+ this.stopFade();
+ this.colour = colour;
+ this.needsRedraw = true;
+}
+
+HuesCanvas.prototype.doBlackout = function(whiteout) {
+ if (typeof(whiteout)==='undefined') whiteout = false;
+ if(whiteout) {
+ this.blackoutColour = "#FFF";
+ } else {
+ this.blackoutColour = "#000";
+ }
+ this.blackoutTimeout = 0; // indefinite
+ this.blackoutStart = this.aContext.currentTime;
+ this.blackout = true;
+ this.needsRedraw = true;
+}
+
+// for song changes
+HuesCanvas.prototype.clearBlackout = function() {
+ this.blackout = false;
+ this.blackoutTimeout = 0;
+ this.needsRedraw = true;
+}
+
+HuesCanvas.prototype.doShortBlackout = function(beatTime) {
+ this.doBlackout();
+ // GetRandomWaifu(); TODO IMPLEMENT IN CORE INSTEAD
+ this.blackoutTimeout = this.aContext.currentTime + beatTime / 1.7;
+ // looks better if we go right to black
+ this.blackoutStart = 0;
+}
+
+HuesCanvas.prototype.doColourFade = function(length, newColour) {
+ this.colourFade = true;
+ this.colourFadeLength = length;
+ this.colourFadeStart = this.aContext.currentTime;
+ this.oldColour = this.colour;
+ this.newColour = newColour;
+}
+
+HuesCanvas.prototype.stopFade = function() {
+ this.colourFade = false;
+ this.colourFadeStart = 0;
+ this.colourFadeLength = 0;
+}
+
+HuesCanvas.prototype.mixColours = function(percent) {
+ percent = Math.min(1, percent);
+ var oldR = this.oldColour >> 16 & 0xFF;
+ var oldG = this.oldColour >> 8 & 0xFF;
+ var oldB = this.oldColour & 0xFF;
+ var newR = this.newColour >> 16 & 0xFF;
+ var newG = this.newColour >> 8 & 0xFF;
+ var newB = this.newColour & 0xFF;
+ var mixR = oldR * (1 - percent) + newR * percent;
+ var mixG = oldG * (1 - percent) + newG * percent;
+ var mixB = oldB * (1 - percent) + newB * percent;
+ this.colour = mixR << 16 | mixG << 8 | mixB;
+}
+
+HuesCanvas.prototype.doXBlur = function() {
+ this.blurStart = this.aContext.currentTime;
+ this.blurDistance = this.blurAmount;
+ this.xBlur = true;
+ this.yBlur = false;
+ this.needsRedraw = true;
+}
+
+HuesCanvas.prototype.doYBlur = function() {
+ this.blurStart = this.aContext.currentTime;
+ this.blurDistance = this.blurAmount;
+ this.xBlur = false;
+ this.yBlur = true;
+ this.needsRedraw = true;
+}
+
+HuesCanvas.prototype.setBlurDecay = function(decay) {
+ this.blurDecay = decay;
+}
+
+HuesCanvas.prototype.setBlurIterations = function(iterations) {
+ this.blurIterations = iterations;
+ this.blurDelta = this.blurAmount / this.blurIterations;
+ this.blurAlpha = 1/(this.blurIterations/2);
+}
+
+HuesCanvas.prototype.setBlurAmount = function(amount) {
+ this.blurAmount = amount;
+ this.blurMin = -this.blurAmount/2;
+ this.blurMax = this.blurAmount/2;
+}
+
+HuesCanvas.prototype.setAnimating = function(anim) {
+ if(!this.animating && anim) {
+ requestAnimationFrame(this.animationLoop);
+ }
+ this.animating = anim;
+}
\ No newline at end of file
diff --git a/js/HuesCore.js b/js/HuesCore.js
new file mode 100644
index 0000000..d340dcc
--- /dev/null
+++ b/js/HuesCore.js
@@ -0,0 +1,843 @@
+var defaultSettings = {
+ load : true, // Debugging var, for loading zips or not
+ autoplay : true, // Debug, play first song automatically?
+ blurQuality: 2, // low/med/high/extreme 0-3
+
+ // UI accessible config
+ // Autosong stuff is a todo, becuase why even implement that
+ smartAlign: true,
+ blurAmount: 1, // 0,1,2,3 = off,low,med,high
+ blurDecay: 2, // 0,1,2,3 = slow,med,fast,faster!
+ colourSet: "normal", // normal, pastel, 420
+ blackoutUI: false,
+ playBuildups: "on", // off, once, on
+ volume : 0.7
+}
+
+HuesCore = function(defaults) {
+ // Bunch-o-initialisers
+ this.version = "0x01";
+ this.beatIndex = 0;
+ this.beatLength=-1;
+ this.currentSong;
+ this.currentImage;
+ this.songIndex=-1;
+ this.colourIndex=0;
+ this.imageIndex=-1;
+ this.isFullAuto = true;
+ this.currentVolume=70;
+ this.volumeMuted=false;
+ this.loopCount=0;
+ this.doRandom = false;
+ this.lastSC=0;
+ this.buildupDiff=0;
+ this.animTimeoutID;
+ this.fadeOut=false;
+ this.fadeDirection=false;
+ this.loadedFiles=0;
+ this.doBuildup=true;
+ this.userInterface = null;
+
+ for(var attr in defaultSettings) {
+ if(defaults[attr] == undefined)
+ defaults[attr] = defaultSettings[attr];
+ }
+
+ console.log("0x40 Hues - start your engines!");
+ this.colours = this.oldColours;
+ this.uiArray = [];
+ this.lastSongArray = [];
+ this.lastImageArray = [];
+ this.settings = new HuesSettings(defaults);
+ this.autoSong = this.settings.autosong;
+ this.resourceManager = new Resources();
+ this.soundManager = new SoundManager();
+ if(!this.soundManager.canUse) {
+ this.error("Web Audio API not supported in this browser.");
+ return;
+ }
+ this.renderer = new HuesCanvas("waifu", this.soundManager.context, this);
+
+ this.uiArray.push(new RetroUI(), new WeedUI(), new ModernUI(), new XmasUI());
+ this.changeUI(1);
+ this.settings.connectCore(this);
+
+ // todo: only after respacks loaded?
+ var that = this;
+ if(defaults.load) {
+ this.resourceManager.addAll(defaults.respacks, function() {
+ document.getElementById("preloadHelper").className = "loaded";
+ window.setTimeout(function() {
+ document.getElementById("preloadHelper").style.display = "none";
+ }, 1500);
+ that.setImage(0);
+ if(defaults.autoplay) {
+ that.setSong(0);
+ // TODO delete me
+ that.previousSong();
+ that.previousSong();
+ }
+ }, function(progress) {
+ var prog = document.getElementById("preMain");
+ var scale = Math.floor(progress * 0x40);
+ prog.textContent = '0x' + ("00"+scale.toString(16)).slice(-2);
+ });
+ } else {
+ document.getElementById("preloadHelper").style.display = "none";
+ }
+
+ document.onkeydown = function(e){
+ e = e || window.event;
+ var key = e.keyCode || e.which;
+ return that.keyHandler(key);
+ };
+
+ window.onerror = function(msg, url, line, col, error) {
+ that.error(msg);
+
+ // Get more info in console
+ return false;
+ };
+
+ this.animationLoop();
+}
+
+HuesCore.prototype.animationLoop = function() {
+ var that = this;
+ if(!this.soundManager.playing) {
+ requestAnimationFrame(function() {that.animationLoop()});
+ return;
+ }
+ var now = this.soundManager.currentTime();
+ if(now < 0) {
+ this.userInterface.updateTime(0);
+ } else {
+ this.userInterface.updateTime(this.soundManager.displayableTime());
+ }
+ for(var beatTime = this.beatIndex * this.beatLength; beatTime < now;
+ beatTime = ++this.beatIndex * this.beatLength) {
+ var beat = this.getBeat(this.beatIndex);
+ this.beater(beat);
+ }
+ requestAnimationFrame(function() {that.animationLoop()});
+}
+
+HuesCore.prototype.getCurrentMode = function() {
+ return this.isFullAuto ? "FULL AUTO" : "NORMAL";
+}
+
+HuesCore.prototype.getSafeBeatIndex = function() {
+ if(!this.soundManager.playing) {
+ return 0;
+ }
+ if(this.beatIndex < 0) {
+ return 0;
+ } else {
+ return this.beatIndex % this.currentSong.rhythm.length;
+ }
+}
+
+HuesCore.prototype.blurUpdated = function(x, y) {
+ this.userInterface.blurUpdated(x, y);
+}
+
+HuesCore.prototype.nextSong = function() {
+ this.lastSongArray = [];
+ var index = (this.songIndex + 1) % this.resourceManager.enabledSongs.length;
+ this.setSong(index);
+}
+
+HuesCore.prototype.previousSong = function() {
+ this.lastSongArray = [];
+ var index = ((this.songIndex - 1) + this.resourceManager.enabledSongs.length) % this.resourceManager.enabledSongs.length;
+ this.setSong(index);
+}
+
+HuesCore.prototype.setSong = function(index) {
+ this.soundManager.stop();
+ this.songIndex = index;
+ this.currentSong = this.resourceManager.enabledSongs[this.songIndex];
+ if (this.currentSong == undefined) {
+ this.currentSong = {"name":"None", "title":"None", "rhythm":".", "source":null, "crc":"none", "sound":null, "enabled":true, "filename":"none"};
+ }
+ console.log("Next song:", this.songIndex, this.currentSong);
+ this.userInterface.setSongText();
+ this.loopCount = 0;
+ if (this.currentSong.buildup) {
+ if (this.currentSong.force) {
+ if (this.currentSong.force != "song") {
+ this.currentSong.buildupPlayed = false;
+ this.doBuildup = true;
+ } else {
+ this.currentSong.buildupPlayed = true;
+ this.doBuildup = false;
+ }
+ this.currentSong.force = null;
+ } else {
+ switch (this.settings.buildups) {
+ case "off":
+ this.currentSong.buildupPlayed = true;
+ this.doBuildup = false;
+ break;
+ case "on":
+ this.currentSong.buildupPlayed = false;
+ this.doBuildup = true;
+ break;
+ case "once":
+ this.doBuildup = !this.currentSong.buildupPlayed;
+ break;
+ }
+ }
+ }
+ this.resetAudio();
+ var that = this;
+ this.soundManager.playSong(this.currentSong, function() {
+ that.fillBuildup();
+ });
+}
+
+HuesCore.prototype.fillBuildup = function() {
+ this.beatLength = this.soundManager.loopLength / this.currentSong.rhythm.length;
+ var buildBeats = Math.floor(this.soundManager.loopStart / this.beatLength) + 1;
+ if (this.currentSong.buildupRhythm == null) {
+ this.currentSong.buildupRhythm = "";
+ }
+ if (this.currentSong.buildupRhythm.length < buildBeats) {
+ console.log("Filling buildup beatmap");
+ while (this.currentSong.buildupRhythm.length < buildBeats) {
+ this.currentSong.buildupRhythm = this.currentSong.buildupRhythm + ".";
+ }
+ }
+ console.log("Buildup length:", buildBeats);
+ this.beatIndex = this.doBuildup ? -this.currentSong.buildupRhythm.length : 0;
+}
+
+HuesCore.prototype.randomSong = function() {
+ var index=Math.floor((Math.random() * this.resourceManager.enabledSongs.length));
+ if (index == this.songIndex && this.resourceManager.enabledSongs.length > 1 || !(this.lastSongArray.indexOf(index) == -1)) {
+ this.randomSong();
+ } else {
+ console.log("Randoming a song!");
+ this.setSong(index);
+ this.lastSongArray.push(index);
+ noRepeat = Math.min(5, Math.floor((this.resourceManager.enabledSongs.length / 2)));
+ while (this.lastSongArray.length > noRepeat && noRepeat >= 0) {
+ this.lastSongArray.shift();
+ }
+ }
+}
+/*
+HuesCore.prototype.onLoop = function() {
+ this.loopCount = this.loopCount + 1;
+ switch (this.settings.autosong) {
+ case "loop":
+ console.log("Checking loops");
+ if (this.loopCount >= this.settings.autosongDelay) {
+ this.startSongChangeFade();
+ }
+ break;
+ case "time":
+ console.log("Checking times");
+ if (this.currentSong.sound && this.calculateSongLength(this.currentSong.sound) / 1000 * this.loopCount >= this.settings.autosongDelay * 60) {
+ this.startSongChangeFade();
+ }
+ break;
+ }
+}
+
+HuesCore.prototype.startSongChangeFade = function() {
+ this.fadeDirection = true;
+ this.fadeOut = true;
+}
+*/
+HuesCore.prototype.songDataUpdated = function() {
+ if (this.currentSong) {
+ this.beatLength = 0;
+ this.userInterface.updateLists();
+ this.userInterface.updateTexts();
+ this.userInterface.setSongText();
+ this.userInterface.setImageText();
+ } else {
+ this.beatLength = -1;
+ }
+}
+
+HuesCore.prototype.resetAudio = function() {
+ this.samplePosition = 0;
+ this.beatIndex = 0;
+ this.position = 0;
+ this.lastSC = 0;
+ this.buildupDiff = 0;
+ this.songDataUpdated();
+}
+
+HuesCore.prototype.randomImage = function() {
+ var len = this.resourceManager.enabledImages.length;
+ var index=Math.floor(Math.random() * len);
+ if ((index == this.imageIndex || this.lastImageArray.indexOf(index) != -1) && len > 1) {
+ this.randomImage();
+ } else {
+ this.setImage(index);
+ this.lastImageArray.push(index);
+ var cull = Math.min(20, Math.floor((len / 2)));
+ while (this.lastImageArray.length > cull && cull >= 0) {
+ this.lastImageArray.shift();
+ }
+ }
+}
+
+HuesCore.prototype.getImageIndex = function() {
+ return this.imageIndex;
+}
+
+HuesCore.prototype.setImage = function(index) {
+ this.imageIndex = index;
+ var img=this.resourceManager.enabledImages[this.imageIndex];
+ if (img == this.currentImage && !(img == null)) {
+ return;
+ }
+ if (img) {
+ this.currentImage = img;
+ } else if (!this.currentImage) {
+ this.currentImage = {"name":"None", "fullname":"None", "align":"center", "bitmap":null, "source":null, "enabled":true};
+ this.imageIndex = -1;
+ this.lastImageArray = [];
+ }
+ this.renderer.setImage(this.currentImage);
+ this.userInterface.setImageText();
+}
+
+HuesCore.prototype.nextImage = function() {
+ this.setIsFullAuto(false);
+ var img=(this.imageIndex + 1) % this.resourceManager.enabledImages.length;
+ this.setImage(img);
+ this.lastImageArray = [];
+}
+
+HuesCore.prototype.previousImage = function() {
+ this.setIsFullAuto(false);
+ var img=((this.imageIndex - 1) + this.resourceManager.enabledImages.length) % this.resourceManager.enabledImages.length;
+ this.setImage(img);
+ this.lastImageArray = [];
+}
+
+HuesCore.prototype.randomColourIndex = function() {
+ var index=Math.floor((Math.random() * 64));
+ if (index == this.colourIndex) {
+ return this.randomColourIndex();
+ }
+ return index;
+}
+
+HuesCore.prototype.randomColour = function() {
+ var index=this.randomColourIndex();
+ this.setColour(index);
+}
+
+HuesCore.prototype.setColour = function(index) {
+ this.colourIndex = index;
+ this.renderer.setColour(this.colours[this.colourIndex].c);
+ this.userInterface.setColourText();
+}
+
+HuesCore.prototype.getBeat = function(index) {
+ if(index < 0) {
+ return this.currentSong.buildupRhythm[this.currentSong.buildupRhythm.length+index];
+ } else {
+ return this.currentSong.rhythm[index % this.currentSong.rhythm.length];
+ }
+}
+
+HuesCore.prototype.beater = function(beat) {
+ this.userInterface.beat();
+ switch(beat) {
+ case 'X':
+ case 'x':
+ this.renderer.doYBlur();
+ break;
+ case 'O':
+ case 'o':
+ this.renderer.doXBlur();
+ break;
+ case '+':
+ this.renderer.doXBlur();
+ this.renderer.doBlackout();
+ break;
+ case '¤':
+ this.renderer.doXBlur();
+ this.renderer.doBlackout(true);
+ break;
+ case '|':
+ this.renderer.doShortBlackout(this.beatLength);
+ this.randomColour();
+ break;
+ case ':':
+ this.randomColour();
+ break;
+ case '*':
+ if(this.isFullAuto) {
+ this.randomImage();
+ }
+ break;
+ case '=':
+ if(this.isFullAuto) {
+ this.randomImage();
+ }
+ case '~':
+ // case: fade in build, not in rhythm. Must max out fade timer.
+ var maxSearch = this.currentSong.rhythm.length;
+ if(this.beatIndex < 0) {
+ maxSearch -= this.beatIndex;
+ }
+ var fadeLen;
+ for (fadeLen = 1; fadeLen <= maxSearch; fadeLen++) {
+ if (this.getBeat(fadeLen + this.beatIndex) != ".") {
+ break;
+ }
+ }
+ this.stopFade();
+ this.startFade(fadeLen * this.beatLength);
+ break;
+ }
+ if ([".", "+", "|", "¤"].indexOf(beat) == -1) {
+ this.renderer.clearBlackout();
+ }
+ if([".", "+", ":", "*", "X", "O", "~", "="].indexOf(beat) == -1) {
+ this.randomColour();
+ if (this.isFullAuto) {
+ this.randomImage();
+ }
+ }
+}
+
+HuesCore.prototype.getBeatString = function(length) {
+ length = length ? length : 256;
+
+ var beatString = "";
+ var song = this.currentSong;
+ if (song) {
+ if(this.beatIndex < 0) {
+ beatString = song.buildupRhythm.slice(
+ song.buildupRhythm.length + this.beatIndex);
+ } else {
+ beatString = song.rhythm.slice(this.beatIndex % song.rhythm.length);
+ }
+ while (beatString.length < length) {
+ beatString += song.rhythm;
+ }
+ }
+
+ return beatString;
+}
+
+HuesCore.prototype.setIsFullAuto = function(auto) {
+ this.isFullAuto = auto;
+ if (this.userInterface) {
+ this.userInterface.modeUpdated();
+ }
+}
+
+/*HuesCore.prototype.enterFrame = function() {
+ this.setTexts();
+ if (this.fadeOut) {
+ // 30fps frame locked, TODO
+ delta = this.fadeDirection ? -2 : 2;
+ if (this.soundChannel) {
+ fadeTo = Math.max(0, Math.min(this.currentVolume, soundManager.volume + delta));
+ this.soundManager.setVolume(fadeTo);
+ if (fadeTo == 0) {
+ this.fadeOut = false;
+ this.fadeDirection = false;
+ if (this.settings.autosongShuffle) {
+ this.randomSong();
+ } else {
+ this.nextSong();
+ }
+ }
+ }
+ }
+}*/
+
+HuesCore.prototype.respackLoaded = function() {
+ this.init();
+ this.userInterface.updateLists();
+}
+
+/*HuesCore.prototype.rightClickListener = function(event) {
+ switch (event) {
+ case flash.events.MouseEvent.RIGHT_CLICK:
+ this.toggleSettingsWindow();
+ break;
+ case flash.events.MouseEvent.MOUSE_WHEEL:
+ if (event.delta > 0) {
+ this.soundManager.increaseVolume();
+ } else {
+ this.soundManager.decreaseVolume();
+ }
+ break;
+ }
+}*/
+
+HuesCore.prototype.changeUI = function(index) {
+ if (index >= 0 && this.uiArray.length > index && !(this.userInterface == this.uiArray[index])) {
+ if(this.userInterface)
+ this.userInterface.disconnect();
+ this.userInterface = this.uiArray[index];
+ this.userInterface.connectCore(this);
+ this.userInterface.updateLists();
+ this.userInterface.updateTexts();
+ this.userInterface.setSongText();
+ this.userInterface.setImageText();
+ this.userInterface.setColourText(this.colourIndex);
+ this.userInterface.beat();
+ }
+}
+
+HuesCore.prototype.settingsUpdated = function() {
+ console.log("Updating according to this.settings");
+ this.blurMultiplier = {"low":0.5, "medium":1, "high":4}[this.settings.blurAmount];
+ this.blurDecayMultiplier = {"low":0.3, "medium":0.6, "high":1, "vhigh":1.6}[this.settings.blurDecay];
+ //this.settings.blackoutUI;
+ // todo: blackoutUI
+ switch (this.settings.colours) {
+ case "normal":
+ this.colours = this.oldColours;
+ break;
+ case "pastel":
+ this.colours = this.pastelColours;
+ break;
+ case "gp":
+ this.colours = this.weedColours;
+ break;
+ }
+ switch (this.settings.ui) {
+ case "retro":
+ this.changeUI(0);
+ break;
+ case "weed":
+ this.changeUI(1);
+ break;
+ case "modern":
+ this.changeUI(2);
+ break;
+ case "xmas":
+ this.changeUI(3);
+ break;
+ }
+ /*if (this.autoSong == "off" && !(this.settings.autosong == "off")) {
+ console.log("Resetting loopCount since AutoSong was enabled");
+ this.loopCount = 0;
+ this.autoSong = this.settings.autosong;
+ }*/
+}
+
+HuesCore.prototype.enabledChanged = function() {
+ this.resourceManager.rebuildEnabled();
+ this.userInterface.updateLists();
+}
+
+HuesCore.prototype.hideLists = function() {
+ this.userInterface.songList.hide();
+ this.userInterface.imageList.hide();
+}
+
+HuesCore.prototype.toggleSongList = function() {
+ this.userInterface.songList.toggleHide();
+ this.userInterface.imageList.hide();
+}
+
+HuesCore.prototype.toggleImageList = function() {
+ this.userInterface.imageList.toggleHide();
+ this.userInterface.songList.hide();
+}
+
+HuesCore.prototype.openSongSource = function() {
+ if (this.currentSong && this.currentSong.source) {
+ window.open(this.currentSong.source,'_blank');
+ }
+}
+
+HuesCore.prototype.openImageSource = function() {
+ if (this.currentImage && this.currentImage.source) {
+ window.open(this.currentImage.source,'_blank');
+ }
+}
+
+HuesCore.prototype.keyHandler = function(key) {
+ switch (key) {
+ case 37: // LEFT
+ this.previousImage();
+ break;
+ case 39: // RIGHT
+ this.nextImage();
+ break;
+ case 38: // UP
+ this.nextSong();
+ break;
+ case 40: // DOWN
+ this.previousSong();
+ break;
+ case 70: // F
+ this.setIsFullAuto(!this.isFullAuto);
+ break;
+ case 109: // NUMPAD_SUBTRACT
+ case 189: // MINUS
+ case 173: // MINUS, legacy
+ this.soundManager.decreaseVolume();
+ break;
+ case 107: // NUMPAD_ADD
+ case 187: // EQUAL
+ case 61: // EQUAL, legacy
+ this.soundManager.increaseVolume();
+ break;
+ case 77: // M
+ this.soundManager.toggleMute();
+ break;
+ case 72: // H
+ this.userInterface.toggleHide();
+ break;
+ case 82: // R
+ this.window.showRespacks();
+ break;
+ case 69: // E
+ this.window.showEditor();
+ break;
+ case 79: // O
+ this.window.showOptions();
+ break;
+ case 73: // I
+ this.window.showInfo();
+ break;
+ case 49: // NUMBER_1
+ this.changeUI(0);
+ break;
+ case 50: // NUMBER_2
+ this.changeUI(1);
+ break;
+ case 51: // NUMBER_3
+ this.changeUI(2);
+ break;
+ case 52: // NUMBER_4
+ this.changeUI(3);
+ break;
+ case 76: // L
+ this.loadLocal();
+ break;
+ case 67: // C
+ this.toggleImageList();
+ break;
+ case 83: // S
+ this.toggleSongList();
+ break;
+ case 87: // W
+ this.toggleSettingsWindow();
+ break;
+ case 16: // SHIFT
+ this.randomSong();
+ break;
+ default:
+ return true;
+ }
+ return false;
+}
+
+HuesCore.prototype.error = function(message) {
+ document.getElementById("preSub").textContent = "Error: " + message;
+ document.getElementById("preMain").style.color = "#F00";
+}
+
+HuesCore.prototype.oldColours =
+ [{'c': 0x000000, 'n': 'black'},
+ {'c': 0x550000, 'n': 'brick'},
+ {'c': 0xAA0000, 'n': 'crimson'},
+ {'c': 0xFF0000, 'n': 'red'},
+ {'c': 0x005500, 'n': 'turtle'},
+ {'c': 0x555500, 'n': 'sludge'},
+ {'c': 0xAA5500, 'n': 'brown'},
+ {'c': 0xFF5500, 'n': 'orange'},
+ {'c': 0x00AA00, 'n': 'green'},
+ {'c': 0x55AA00, 'n': 'grass'},
+ {'c': 0xAAAA00, 'n': 'maize'},
+ {'c': 0xFFAA00, 'n': 'citrus'},
+ {'c': 0x00FF00, 'n': 'lime'},
+ {'c': 0x55FF00, 'n': 'leaf'},
+ {'c': 0xAAFF00, 'n': 'chartreuse'},
+ {'c': 0xFFFF00, 'n': 'yellow'},
+ {'c': 0x000055, 'n': 'midnight'},
+ {'c': 0x550055, 'n': 'plum'},
+ {'c': 0xAA0055, 'n': 'pomegranate'},
+ {'c': 0xFF0055, 'n': 'rose'},
+ {'c': 0x005555, 'n': 'swamp'},
+ {'c': 0x555555, 'n': 'dust'},
+ {'c': 0xAA5555, 'n': 'dirt'},
+ {'c': 0xFF5555, 'n': 'blossom'},
+ {'c': 0x00AA55, 'n': 'sea'},
+ {'c': 0x55AA55, 'n': 'ill'},
+ {'c': 0xAAAA55, 'n': 'haze'},
+ {'c': 0xFFAA55, 'n': 'peach'},
+ {'c': 0x00FF55, 'n': 'spring'},
+ {'c': 0x55FF55, 'n': 'mantis'},
+ {'c': 0xAAFF55, 'n': 'brilliant'},
+ {'c': 0xFFFF55, 'n': 'canary'},
+ {'c': 0x0000AA, 'n': 'navy'},
+ {'c': 0x5500AA, 'n': 'grape'},
+ {'c': 0xAA00AA, 'n': 'mauve'},
+ {'c': 0xFF00AA, 'n': 'purple'},
+ {'c': 0x0055AA, 'n': 'cornflower'},
+ {'c': 0x5555AA, 'n': 'deep'},
+ {'c': 0xAA55AA, 'n': 'lilac'},
+ {'c': 0xFF55AA, 'n': 'lavender'},
+ {'c': 0x00AAAA, 'n': 'aqua'},
+ {'c': 0x55AAAA, 'n': 'steel'},
+ {'c': 0xAAAAAA, 'n': 'grey'},
+ {'c': 0xFFAAAA, 'n': 'pink'},
+ {'c': 0x00FFAA, 'n': 'bay'},
+ {'c': 0x55FFAA, 'n': 'marina'},
+ {'c': 0xAAFFAA, 'n': 'tornado'},
+ {'c': 0xFFFFAA, 'n': 'saltine'},
+ {'c': 0x0000FF, 'n': 'blue'},
+ {'c': 0x5500FF, 'n': 'twilight'},
+ {'c': 0xAA00FF, 'n': 'orchid'},
+ {'c': 0xFF00FF, 'n': 'magenta'},
+ {'c': 0x0055FF, 'n': 'azure'},
+ {'c': 0x5555FF, 'n': 'liberty'},
+ {'c': 0xAA55FF, 'n': 'royalty'},
+ {'c': 0xFF55FF, 'n': 'thistle'},
+ {'c': 0x00AAFF, 'n': 'ocean'},
+ {'c': 0x55AAFF, 'n': 'sky'},
+ {'c': 0xAAAAFF, 'n': 'periwinkle'},
+ {'c': 0xFFAAFF, 'n': 'carnation'},
+ {'c': 0x00FFFF, 'n': 'cyan'},
+ {'c': 0x55FFFF, 'n': 'turquoise'},
+ {'c': 0xAAFFFF, 'n': 'powder'},
+ {'c': 0xFFFFFF, 'n': 'white'}];
+HuesCore.prototype.pastelColours =
+ [{'c': 0xCD4A4A, 'n': 'Mahogany'},
+ {'c': 0xFAE7B5, 'n': 'Banana Mania'},
+ {'c': 0x9F8170, 'n': 'Beaver'},
+ {'c': 0x232323, 'n': 'Black'},
+ {'c': 0xBC5D58, 'n': 'Chestnut'},
+ {'c': 0xDD9475, 'n': 'Copper'},
+ {'c': 0x9ACEEB, 'n': 'Cornflower'},
+ {'c': 0x2B6CC4, 'n': 'Denim'},
+ {'c': 0xEFCDB8, 'n': 'Desert Sand'},
+ {'c': 0x6E5160, 'n': 'Eggplant'},
+ {'c': 0x1DF914, 'n': 'Electric Lime'},
+ {'c': 0x71BC78, 'n': 'Fern'},
+ {'c': 0xFCD975, 'n': 'Goldenrod'},
+ {'c': 0xA8E4A0, 'n': 'Granny Smith Apple'},
+ {'c': 0x95918C, 'n': 'Gray'},
+ {'c': 0x1CAC78, 'n': 'Green'},
+ {'c': 0xFF1DCE, 'n': 'Hot Magenta'},
+ {'c': 0xB2EC5D, 'n': 'Inch Worm'},
+ {'c': 0x5D76CB, 'n': 'Indigo'},
+ {'c': 0xFDFC74, 'n': 'Laser Lemon'},
+ {'c': 0xFCB4D5, 'n': 'Lavender'},
+ {'c': 0xFFBD88, 'n': 'Macaroni and Cheese'},
+ {'c': 0x979AAA, 'n': 'Manatee'},
+ {'c': 0xFF8243, 'n': 'Mango Tango'},
+ {'c': 0xFDBCB4, 'n': 'Melon'},
+ {'c': 0x1A4876, 'n': 'Midnight Blue'},
+ {'c': 0xFFA343, 'n': 'Neon Carrot'},
+ {'c': 0xBAB86C, 'n': 'Olive Green'},
+ {'c': 0xFF7538, 'n': 'Orange'},
+ {'c': 0xE6A8D7, 'n': 'Orchid'},
+ {'c': 0x414A4C, 'n': 'Outer Space'},
+ {'c': 0xFF6E4A, 'n': 'Outrageous Orange'},
+ {'c': 0x1CA9C9, 'n': 'Pacific Blue'},
+ {'c': 0xC5D0E6, 'n': 'Periwinkle'},
+ {'c': 0x8E4585, 'n': 'Plum'},
+ {'c': 0x7442C8, 'n': 'Purple Heart'},
+ {'c': 0xD68A59, 'n': 'Raw Sienna'},
+ {'c': 0xE3256B, 'n': 'Razzmatazz'},
+ {'c': 0xEE204D, 'n': 'Red'},
+ {'c': 0x1FCECB, 'n': 'Robin Egg Blue'},
+ {'c': 0x7851A9, 'n': 'Royal Purple'},
+ {'c': 0xFF9BAA, 'n': 'Salmon'},
+ {'c': 0xFC2847, 'n': 'Scarlet'},
+ {'c': 0x9FE2BF, 'n': 'Sea Green'},
+ {'c': 0xA5694F, 'n': 'Sepia'},
+ {'c': 0x8A795D, 'n': 'Shadow'},
+ {'c': 0x45CEA2, 'n': 'Shamrock'},
+ {'c': 0xFB7EFD, 'n': 'Shocking Pink'},
+ {'c': 0xECEABE, 'n': 'Spring Green'},
+ {'c': 0xFD5E53, 'n': 'Sunset Orange'},
+ {'c': 0xFAA76C, 'n': 'Tan'},
+ {'c': 0xFC89AC, 'n': 'Tickle Me Pink'},
+ {'c': 0xDBD7D2, 'n': 'Timberwolf'},
+ {'c': 0x17806D, 'n': 'Tropical Rain Forest'},
+ {'c': 0x77DDE7, 'n': 'Turquoise Blue'},
+ {'c': 0xFFA089, 'n': 'Vivid Tangerine'},
+ {'c': 0x8F509D, 'n': 'Vivid Violet'},
+ {'c': 0xEDEDED, 'n': 'White'},
+ {'c': 0xFF43A4, 'n': 'Wild Strawberry'},
+ {'c': 0xFC6C85, 'n': 'Wild Watermelon'},
+ {'c': 0xCDA4DE, 'n': 'Wisteria'},
+ {'c': 0xFCE883, 'n': 'Yellow'},
+ {'c': 0xC5E384, 'n': 'Yellow Green'},
+ {'c': 0xFFB653, 'n': 'Yellow Orange'}];
+HuesCore.prototype.weedColours =
+ [{'c': 0x00FF00, 'n': 'Green'},
+ {'c': 0x5A6351, 'n': 'Lizard'},
+ {'c': 0x636F57, 'n': 'Cactus'},
+ {'c': 0x4A7023, 'n': 'Kakapo'},
+ {'c': 0x3D5229, 'n': 'Wet Moss'},
+ {'c': 0x659D32, 'n': 'Tree Moss'},
+ {'c': 0x324F17, 'n': 'Lime Rind'},
+ {'c': 0x7F8778, 'n': 'Flight Jacket'},
+ {'c': 0xBCED91, 'n': 'Green Mist'},
+ {'c': 0x488214, 'n': 'Holly'},
+ {'c': 0x577A3A, 'n': 'Mtn Dew Bottle'},
+ {'c': 0x748269, 'n': 'Seaweed Roll'},
+ {'c': 0x83F52C, 'n': 'Neon Green'},
+ {'c': 0xC0D9AF, 'n': 'Lichen'},
+ {'c': 0xA6D785, 'n': 'Guacamole'},
+ {'c': 0x687E5A, 'n': 'Pond Scum'},
+ {'c': 0x3F602B, 'n': 'Douglas Fir'},
+ {'c': 0x3F6826, 'n': 'Royal Palm'},
+ {'c': 0x646F5E, 'n': 'Seaweed'},
+ {'c': 0x476A34, 'n': 'Noble Fir'},
+ {'c': 0x5DFC0A, 'n': 'Green Led'},
+ {'c': 0x435D36, 'n': 'Spinach'},
+ {'c': 0x84BE6A, 'n': 'Frog'},
+ {'c': 0x5B9C64, 'n': 'Emerald'},
+ {'c': 0x3A6629, 'n': 'Circuit Board'},
+ {'c': 0x308014, 'n': 'Sapgreen'},
+ {'c': 0x31B94D, 'n': 'Pool Table'},
+ {'c': 0x55AE3A, 'n': 'Leaf'},
+ {'c': 0x4DBD33, 'n': 'Grass'},
+ {'c': 0x596C56, 'n': 'Snake'},
+ {'c': 0x86C67C, 'n': '100 Euro'},
+ {'c': 0x7BCC70, 'n': 'Night Vision'},
+ {'c': 0xA020F0, 'n': 'Purple'},
+ {'c': 0x9B30FF, 'n': 'Purple'},
+ {'c': 0x912CEE, 'n': 'Purple'},
+ {'c': 0x7D26CD, 'n': 'Purple'},
+ {'c': 0xAA00FF, 'n': 'Purple'},
+ {'c': 0x800080, 'n': 'Purple'},
+ {'c': 0xA74CAB, 'n': 'Turnip'},
+ {'c': 0x8F5E99, 'n': 'Violet'},
+ {'c': 0x816687, 'n': 'Eggplant'},
+ {'c': 0xCC00FF, 'n': 'Grape'},
+ {'c': 0x820BBB, 'n': 'Wild Violet'},
+ {'c': 0x660198, 'n': 'Concord Grape'},
+ {'c': 0x71637D, 'n': 'Garden Plum'},
+ {'c': 0xB272A6, 'n': 'Purple Fish'},
+ {'c': 0x5C246E, 'n': 'Ultramarine Violet'},
+ {'c': 0x5E2D79, 'n': 'Purple Rose'},
+ {'c': 0x683A5E, 'n': 'Sea Urchin'},
+ {'c': 0x91219E, 'n': 'Cobalt Violet Deep'},
+ {'c': 0x8B668B, 'n': 'Plum'},
+ {'c': 0x9932CD, 'n': 'Dark Orchid'},
+ {'c': 0xBF5FFF, 'n': 'Violet Flower'},
+ {'c': 0xBDA0CB, 'n': 'Purple Candy'},
+ {'c': 0x551A8B, 'n': 'Deep Purple'},
+ {'c': 0xB5509C, 'n': 'Thistle'},
+ {'c': 0x871F78, 'n': 'Dark Purple'},
+ {'c': 0x9C6B98, 'n': 'Purple Ink'},
+ {'c': 0xDB70DB, 'n': 'Orchid'},
+ {'c': 0x990099, 'n': 'True Purple'},
+ {'c': 0x8B008B, 'n': 'Darkmagenta'},
+ {'c': 0xB62084, 'n': "Harold's Crayon"},
+ {'c': 0x694489, 'n': 'Purple Rain'},
+ {'c': 0xFFD700, 'n': 'Gold'}];
\ No newline at end of file
diff --git a/js/HuesSettings.js b/js/HuesSettings.js
new file mode 100644
index 0000000..9ae12a1
--- /dev/null
+++ b/js/HuesSettings.js
@@ -0,0 +1,387 @@
+function HuesSettings(defaults) {
+ this.core = null;
+ // TODO: HTML5 local storage or something
+}
+
+HuesSettings.prototype.connectCore = function(core) {
+ this.core = core;
+};
+
+//class HuesSettings
+package
+{
+ import flash.display.*;
+ import flash.net.*;
+
+ public class HuesSettings extends Object
+ {
+ public function HuesSettings()
+ {
+ this.bools = ["imageSmoothing", "blurEnabled", "smartAlign", "autosongShuffle", "blackoutUI"];
+ this.numbs = ["autosongDelay"];
+ super();
+ trace("Settings created");
+ this.callbacks = [];
+ this.currentSettings = {};
+ this.setDefaults();
+ this.load();
+ return;
+ }
+
+ public function set autosongDelay(arg1:int):void
+ {
+ trace("AutoSong delay:", arg1);
+ this.currentSettings["autosongDelay"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set autosongShuffle(arg1:Boolean):void
+ {
+ trace("Image scaling:", arg1);
+ this.currentSettings["autosongShuffle"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set imagescaling(arg1:String):void
+ {
+ trace("Image scaling:", arg1);
+ this.currentSettings["imagescaling"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set colors(arg1:String):void
+ {
+ trace("Colors:", arg1);
+ this.currentSettings["colors"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ internal function setDefaults():void
+ {
+ this.currentSettings = {"flashQuality":flash.display.StageQuality.HIGH, "imageSmoothing":true, "blurEnabled":true, "blurAmount":"medium", "blurDecay":"high", "channels":"stereo", "smartAlign":true, "buildups":"once", "blendMode":"hard", "ui":"modern", "autosong":"off", "autosongDelay":5, "autosongShuffle":true, "imagescaling":"on", "colors":"normal", "blackoutUI":false};
+ return;
+ }
+
+ public function defaults():void
+ {
+ this.setDefaults();
+ this.saveSettings();
+ return;
+ }
+
+ public function getSettingsFromParameters(arg1:Object):void
+ {
+ var loc1:*=undefined;
+ var loc2:*=null;
+ if (arg1)
+ {
+ var loc3:*=0;
+ var loc4:*=arg1;
+ for (loc2 in loc4)
+ {
+ loc1 = arg1[loc2];
+ if (this.bools.indexOf(loc2) == -1)
+ {
+ if (this.numbs.indexOf(loc2) != -1)
+ {
+ if (loc1.match(new RegExp("\\d+")))
+ {
+ loc1 = int(loc1);
+ if (loc2 == "autosongDelay")
+ {
+ loc1 = Math.max(1, loc1);
+ }
+ }
+ else
+ {
+ loc1 = null;
+ }
+ }
+ }
+ else if (loc1 != "true")
+ {
+ if (loc1 != "false")
+ {
+ loc1 = null;
+ }
+ else
+ {
+ loc1 = false;
+ }
+ }
+ else
+ {
+ loc1 = true;
+ }
+ if (loc1 == null)
+ {
+ continue;
+ }
+ this.currentSettings[loc2] = loc1;
+ }
+ this.saveSettings();
+ this.callCallbacks();
+ }
+ return;
+ }
+
+ public function saveSettings():void
+ {
+ var so:flash.net.SharedObject;
+ var k:String;
+
+ var loc1:*;
+ k = null;
+ trace("Saving settings!");
+ so = flash.net.SharedObject.getLocal(this.objectName);
+ var loc2:*=0;
+ var loc3:*=this.currentSettings;
+ for (k in loc3)
+ {
+ so.data[k] = this.currentSettings[k];
+ }
+ so.data.saved = true;
+ try
+ {
+ so.flush();
+ }
+ catch (e:Error)
+ {
+ trace("Saving settings failed, oh well");
+ }
+ return;
+ }
+
+ public function load():void
+ {
+ var loc1:*=flash.net.SharedObject.getLocal(this.objectName);
+ if ("saved" in loc1.data)
+ {
+ trace("Old settings");
+ this.currentSettings = loc1.data;
+ }
+ else
+ {
+ trace("Defaults");
+ this.defaults();
+ }
+ this.callCallbacks();
+ return;
+ }
+
+ public function set blackoutUI(arg1:Boolean):void
+ {
+ trace("Blackout UI:", arg1);
+ this.currentSettings["blackoutUI"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function addCallback(arg1:Function):void
+ {
+ if (this.callbacks.indexOf(arg1) == -1)
+ {
+ this.callbacks.push(arg1);
+ arg1();
+ }
+ return;
+ }
+
+ public function callCallbacks():void
+ {
+ var loc1:*=undefined;
+ this.saveSettings();
+ var loc2:*=0;
+ var loc3:*=this.callbacks;
+ for each (loc1 in loc3)
+ {
+ loc1();
+ }
+ return;
+ }
+
+ public function get flashQuality():String
+ {
+ return this.currentSettings["flashQuality"];
+ }
+
+ public function get imageSmoothing():Boolean
+ {
+ return this.currentSettings["imageSmoothing"];
+ }
+
+ public function get blurEnabled():Boolean
+ {
+ return this.currentSettings["blurEnabled"];
+ }
+
+ public function get blurAmount():String
+ {
+ return this.currentSettings["blurAmount"];
+ }
+
+ public function get blurDecay():String
+ {
+ return this.currentSettings["blurDecay"];
+ }
+
+ public function get channels():String
+ {
+ return this.currentSettings["channels"];
+ }
+
+ public function get smartAlign():Boolean
+ {
+ return this.currentSettings["smartAlign"];
+ }
+
+ public function get buildups():String
+ {
+ return this.currentSettings["buildups"];
+ }
+
+ public function get blendMode():String
+ {
+ return this.currentSettings["blendMode"];
+ }
+
+ public function get ui():String
+ {
+ return this.currentSettings["ui"];
+ }
+
+ public function get autosong():String
+ {
+ return this.currentSettings["autosong"];
+ }
+
+ public function get autosongDelay():int
+ {
+ return this.currentSettings["autosongDelay"];
+ }
+
+ public function get autosongShuffle():Boolean
+ {
+ return this.currentSettings["autosongShuffle"];
+ }
+
+ public function get imagescaling():String
+ {
+ return this.currentSettings["imagescaling"];
+ }
+
+ public function get colors():String
+ {
+ return this.currentSettings["colors"];
+ }
+
+ public function get blackoutUI():Boolean
+ {
+ return this.currentSettings["blackoutUI"];
+ }
+
+ public function set flashQuality(arg1:String):void
+ {
+ trace("Flash quality:", arg1);
+ this.currentSettings["flashQuality"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set imageSmoothing(arg1:Boolean):void
+ {
+ trace("Image smoothing:", arg1);
+ this.currentSettings["imageSmoothing"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set blurEnabled(arg1:Boolean):void
+ {
+ trace("Blur:", arg1);
+ this.currentSettings["blurEnabled"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set blurAmount(arg1:String):void
+ {
+ trace("Blur amount:", arg1);
+ this.currentSettings["blurAmount"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set blurDecay(arg1:String):void
+ {
+ trace("Blur decay:", arg1);
+ this.currentSettings["blurDecay"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set channels(arg1:String):void
+ {
+ trace("Channels:", arg1);
+ this.currentSettings["channels"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set smartAlign(arg1:Boolean):void
+ {
+ trace("Smart align:", arg1);
+ this.currentSettings["smartAlign"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set buildups(arg1:String):void
+ {
+ trace("Buildups:", arg1);
+ this.currentSettings["buildups"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set blendMode(arg1:String):void
+ {
+ trace("Blend mode:", arg1);
+ this.currentSettings["blendMode"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set ui(arg1:String):void
+ {
+ trace("UI:", arg1);
+ this.currentSettings["ui"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ public function set autosong(arg1:String):void
+ {
+ trace("AutoSong:", arg1);
+ this.currentSettings["autosong"] = arg1;
+ this.callCallbacks();
+ return;
+ }
+
+ internal var currentSettings:Object;
+
+ internal var bools:Array;
+
+ internal var numbs:Array;
+
+ internal var objectName:*="HuesSettings51";
+
+ internal var callbacks:Array;
+ }
+}
+
+
diff --git a/js/HuesUI.js b/js/HuesUI.js
new file mode 100644
index 0000000..fec4ef6
--- /dev/null
+++ b/js/HuesUI.js
@@ -0,0 +1,423 @@
+/*
+ Base UI Class for Hues display. Parent is an element
+ to put all your own elements under, but make a div
+ underneath so it can be entirely hidden.
+*/
+function HuesUI(parent) {
+ if(!parent) {
+ parent = document.getElementById("huesUI");
+ }
+ this.root = document.createElement("div");
+ this.root.className = this.constructor.name;
+ parent.appendChild(this.root);
+ this.root.style.display = "none";
+
+ this.core = null;
+
+ this.imageName = null;
+ this.imageLink = null;
+
+ this.songName = null;
+ this.songLink = null;
+
+ this.hueName = null;
+
+ this.volInput = null;
+ this.volLabel = null;
+
+ this.initUI();
+}
+
+HuesUI.prototype.initUI = function() {
+ var doc = this.root.ownerDocument
+
+ var imageName = doc.createElement("div");
+ this.imageName = imageName;
+
+ this.imageLink = doc.createElement("a");
+ this.imageLink.target = "_blank";
+ this.imageName.appendChild(this.imageLink);
+
+ var songName = doc.createElement("div");
+ this.songName = songName;
+
+ this.songLink = doc.createElement("a");
+ this.songLink.target = "_blank";
+ this.songName.appendChild(this.songLink);
+
+ var hueName = doc.createElement("div");
+ this.hueName = hueName;
+
+ //this.setupVolume(leftBox)
+}
+
+HuesUI.prototype.connectCore = function(core) {
+ this.core = core;
+ this.root.style.display = "block";
+}
+
+HuesUI.prototype.disconnect = function() {
+ this.core = null;
+ this.root.style.display = "none";
+}
+
+// May do nothing, may scale elements if needed etc etc
+HuesUI.prototype.resize = function() {}
+HuesUI.prototype.modeUpdated = function() {}
+HuesUI.prototype.beat = function() {}
+HuesUI.prototype.updateTime = function() {}
+HuesUI.prototype.blurUpdated = function(x, y) {}
+
+HuesUI.prototype.setSongText = function() {
+ var song = this.core.currentSong;
+
+ if(!song)
+ return;
+
+ this.songLink.textContent = song.title.toUpperCase();
+ this.songLink.href = song.source;
+}
+
+HuesUI.prototype.setImageText = function() {
+ var image = this.core.currentImage;
+
+ if(!image)
+ return;
+
+ this.imageLink.textContent = image.fullname.toUpperCase();
+ this.imageLink.href = image.source;
+}
+
+HuesUI.prototype.setColourText = function(colour) {
+ var colour = this.core.colours[this.core.colourIndex];
+
+ this.hueName.textContent = colour.n.toUpperCase();
+}
+
+HuesUI.prototype.updateLists = function() {
+ var songs = this.core.resourceManager.enabledSongs;
+ var images = this.core.resourceManager.enabledImages;
+ // TODO display this
+}
+
+HuesUI.prototype.updateTexts = function() {
+ // Timer, beat counter
+}
+
+/*
+ Individual UIs ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+*/
+
+function RetroUI() {
+ this.container = null;
+ this.mode = null;
+ this.beatBar = null;
+ this.beatCount = null;
+ this.timer = null;
+ this.colourIndex = null;
+ this.version = null;
+ HuesUI.call(this);
+}
+RetroUI.prototype = Object.create(HuesUI.prototype);
+RetroUI.prototype.constructor = RetroUI;
+
+RetroUI.prototype.initUI = function() {
+ HuesUI.prototype.initUI.call(this);
+
+ var doc = this.root.ownerDocument;
+
+ var container = doc.createElement("div");
+ container.className = "hues-r-container";
+ this.root.appendChild(container);
+ this.container = container;
+
+ this.mode = doc.createElement("div");
+ container.appendChild(this.mode);
+ container.appendChild(this.imageName);
+
+ this.timer = doc.createElement("div");
+ this.timer.textContent = "T=$0x0000";
+ container.appendChild(this.timer);
+
+ this.beatCount = doc.createElement("div");
+ this.beatCount.textContent = "B=$0x00";
+ container.appendChild(this.beatCount);
+
+ this.xBlur = doc.createElement("div");
+ this.xBlur.textContent = "X=$0x00";
+ container.appendChild(this.xBlur);
+
+ this.yBlur = doc.createElement("div");
+ this.yBlur.textContent = "Y=$0x00";
+ container.appendChild(this.yBlur);
+
+ this.colourIndex = doc.createElement("div");
+ this.colourIndex.textContent = "C=$0x00";
+ container.appendChild(this.colourIndex);
+
+ this.version = doc.createElement("div");
+ container.appendChild(this.version);
+
+ container.appendChild(this.hueName);
+ container.appendChild(this.songName);
+
+ this.beatBar = doc.createElement("div");
+ container.appendChild(this.beatBar);
+}
+
+RetroUI.prototype.connectCore = function(core) {
+ HuesUI.prototype.connectCore.call(this, core);
+
+ this.version.textContent = "V=$" + core.version;
+ this.modeUpdated();
+}
+
+RetroUI.prototype.modeUpdated = function() {
+ this.mode.textContent = "M=" + this.core.getCurrentMode();
+}
+
+RetroUI.prototype.blurUpdated = function(x, y) {
+ x = Math.floor(x * 0xFF);
+ y = Math.floor(y * 0xFF);;
+ this.xBlur.textContent = "X=" + this.intToHex2(x);
+ this.yBlur.textContent = "Y=" + this.intToHex2(y);
+}
+
+RetroUI.prototype.setImageText = function() {
+ var image = this.core.currentImage;
+
+ if(!image)
+ return;
+
+ this.imageLink.textContent = "I=" + image.name.toUpperCase();
+ this.imageLink.href = image.source;
+}
+
+RetroUI.prototype.setColourText = function(colour) {
+ HuesUI.prototype.setColourText.call(this, colour);
+
+ this.colourIndex.textContent = "C=" + this.intToHex2(this.core.colourIndex);
+}
+
+RetroUI.prototype.beat = function() {
+ var beats = this.core.getBeatString();
+ var rest = beats.slice(1);
+
+ this.beatBar.textContent = ">>" + rest;
+
+ this.beatCount.textContent = "B=" + this.intToHex2(this.core.getSafeBeatIndex());
+}
+
+RetroUI.prototype.updateTime = function(time) {
+ time = Math.floor(time * 1000);
+ this.timer.textContent = "T=" + this.intToHex4(time);
+}
+
+RetroUI.prototype.intToHex2 = function(num) {
+ return '$0x' + ("00"+num.toString(16)).slice(-2);
+}
+RetroUI.prototype.intToHex4 = function(num) {
+ return '$0x' + ("0000"+num.toString(16)).slice(-4);
+}
+
+function ModernUI() {
+ this.beatBar = null;
+ this.beatLeft = null;
+ this.beatRight = null;
+ this.beatCenter = null;
+ this.rightBox = null;
+ this.leftBox = null;
+ this.controls = null;
+
+ HuesUI.call(this);
+}
+ModernUI.prototype = Object.create(HuesUI.prototype);
+ModernUI.prototype.constructor = ModernUI;
+
+ModernUI.prototype.initUI = function() {
+ HuesUI.prototype.initUI.call(this);
+
+ var doc = this.root.ownerDocument;
+
+ this.imageName.className = "hues-m-imagename";
+ this.songName.className = "hues-m-songtitle";
+
+ var controls = doc.createElement("div");
+ controls.className = "hues-m-controls";
+ this.root.appendChild(controls);
+ this.controls = controls;
+
+ controls.appendChild(this.imageName);
+ controls.appendChild(this.songName);
+
+ var leftBox = doc.createElement("div");
+ leftBox.className = "hues-m-leftbox";
+ controls.appendChild(leftBox);
+ this.leftBox = leftBox;
+
+ this.hueName.className = "hues-m-huename";
+ leftBox.appendChild(this.hueName);
+
+ var rightBox = doc.createElement("div");
+ rightBox.className = "hues-m-rightbox";
+ controls.appendChild(rightBox);
+ this.rightBox = rightBox;
+
+ var beatBar = doc.createElement("div");
+ beatBar.className = "hues-m-beatbar";
+ this.root.appendChild(beatBar);
+ this.beatBar = beatBar;
+
+ var beatLeft = doc.createElement("div");
+ beatLeft.className = "hues-m-beatleft";
+ beatBar.appendChild(beatLeft);
+ this.beatLeft = beatLeft;
+
+ var beatRight = doc.createElement("div");
+ beatRight.className = "hues-m-beatright";
+ beatBar.appendChild(beatRight);
+ this.beatRight = beatRight;
+
+ var beatCenter = doc.createElement("div");
+ beatCenter.className = "hues-m-beatcenter";
+ this.root.appendChild(beatCenter);
+ this.beatCenter = beatCenter;
+}
+
+ModernUI.prototype.beat = function() {
+ var beats = this.core.getBeatString();
+
+ var current = beats[0];
+ var rest = beats.slice(1);
+
+ this.beatLeft.textContent = rest;
+ this.beatRight.textContent = rest;
+
+
+ if (current != ".") {
+ while (this.beatCenter.firstElementChild) {
+ this.beatCenter.removeChild(this.beatCenter.firstElementChild);
+ }
+ var span = this.beatCenter.ownerDocument.createElement("span");
+ span.textContent = current;
+ this.beatCenter.appendChild(span);
+ }
+}
+
+ModernUI.prototype.setSongText = function() {
+ HuesUI.prototype.setSongText.call(this);
+
+ if(!this.core.currentSong)
+ return;
+
+ var name = this.songName;
+
+ name.className = "hues-m-songtitle"
+ if (name.offsetWidth > name.clientWidth) {
+ name.className = "hues-m-songtitle small"
+ }
+ if (name.offsetWidth > name.clientWidth) {
+ name.className = "hues-m-songtitle x-small"
+ }
+}
+
+ModernUI.prototype.setImageText = function() {
+ HuesUI.prototype.setImageText.call(this);
+
+ if(!this.core.currentImage)
+ return;
+
+ var name = this.imageName
+
+ name.className = "hues-m-imagename"
+ if (name.offsetWidth > name.clientWidth) {
+ name.className = "hues-m-imagename small"
+ }
+ if (name.offsetWidth > name.clientWidth) {
+ name.className = "hues-m-imagename x-small"
+ }
+}
+
+function WeedUI() {
+ RetroUI.call(this);
+
+ this.xVariance = 10;
+ this.yVariance = 20;
+}
+WeedUI.prototype = Object.create(RetroUI.prototype);
+WeedUI.prototype.constructor = WeedUI;
+
+WeedUI.prototype.initUI = function() {
+ RetroUI.prototype.initUI.call(this);
+
+ this.container.removeChild(this.beatBar)
+ var doc = this.root.ownerDocument;
+
+ var beatBar = doc.createElement("div");
+ beatBar.className = "hues-w-beatbar";
+ this.root.appendChild(beatBar);
+ this.beatBar = beatBar;
+
+ var beatLeft = doc.createElement("div");
+ beatLeft.className = "hues-w-beatleft";
+ beatBar.appendChild(beatLeft);
+ this.beatLeft = beatLeft;
+
+ var beatRight = doc.createElement("div");
+ beatRight.className = "hues-w-beatright";
+ beatBar.appendChild(beatRight);
+ this.beatRight = beatRight;
+}
+
+WeedUI.prototype.beat = function() {
+ var beats = this.core.getBeatString();
+ var rest = beats.slice(1);
+
+ this.beatLeft.textContent = rest;
+ this.beatRight.textContent = rest;
+
+ this.beatCount.textContent = "B=" + this.intToHex2(this.core.getSafeBeatIndex());
+
+ if(["x", "o", "X", "O"].indexOf(beats[0]) != -1) {
+ var doc = this.root.ownerDocument;
+ var beatCenter = doc.createElement("div");
+ beatCenter.className = "hues-w-beataccent";
+ var rot = this.round10(15 - Math.random() * 30);
+ var x = this.round10(- this.xVariance / 2 + Math.random() * this.xVariance);
+ var y = this.round10(30 - this.yVariance / 2 + Math.random() * this.yVariance);
+ var transform = "rotate(" + rot + "deg) translate(" + x + "px," + y + "px)";
+ beatCenter.style.MozTransform = transform;
+ beatCenter.style.webkitTransform = transform;
+ beatCenter.style.transform = transform;
+ beatCenter.textContent = beats[0].toUpperCase();
+ this.root.appendChild(beatCenter);
+ window.setTimeout(this.getRemoveBeat(beatCenter), 1500);
+ }
+}
+
+WeedUI.prototype.round10 = function(num) {
+ return Math.round(num * 10) / 10;
+}
+
+WeedUI.prototype.getRemoveBeat = function(element) {
+ var that = this;
+ return function() {
+ that.root.removeChild(element);
+ };
+}
+
+function XmasUI() {
+ ModernUI.call(this);
+
+ this.controls.removeChild(this.leftBox);
+ this.controls.removeChild(this.rightBox);
+
+ this.leftBox = this.rightBox = this.hueName = null;
+
+ this.controls.className = "hues-x-controls";
+ this.beatBar.className = "hues-x-beatbar";
+}
+XmasUI.prototype = Object.create(ModernUI.prototype);
+XmasUI.prototype.constructor = XmasUI;
+
+XmasUI.prototype.setColourText = function(colour) {};
\ No newline at end of file
diff --git a/js/ResourceManager.js b/js/ResourceManager.js
new file mode 100644
index 0000000..a198b97
--- /dev/null
+++ b/js/ResourceManager.js
@@ -0,0 +1,132 @@
+function Resources() {
+ this.resourcePacks = [];
+
+ this.allSongs = [];
+ this.allImages = [];
+ this.enabledSongs = [];
+ this.enabledImages = [];
+
+ this.toLoad = 0;
+ this.progressState = [];
+ this.rToLoad = []
+ this.loadFinishCallback = null;
+ this.progressCallback = null;
+}
+
+// Array of URLs to load, and a callback for when we're done
+// Preserves order of URLs being loaded
+Resources.prototype.addAll = function(urls, callback, progressCallback) {
+ var that = this;
+ this.toLoad += urls.length;
+ if(progressCallback) {
+ this.progressCallback = progressCallback;
+ this.progressState = Array.apply(null, Array(urls.length)).map(Number.prototype.valueOf,0);
+ }
+ if(callback) {
+ this.loadFinishCallback = callback;
+ }
+ for(var i = 0; i < urls.length; i++) {
+ var r = new Respack();
+ this.rToLoad.push(r);
+ r.loadFromURL(urls[i], function() {
+ that.toLoad--;
+ if(that.toLoad <= 0) {
+ // could use a while() and shift(), but it'd be slower
+ for(var i = 0; i < that.rToLoad.length; i++) {
+ that.addPack(that.rToLoad[i]);
+ }
+ that.rToLoad = [];
+ if(that.loadFinishCallback) {
+ that.loadFinishCallback();
+ that.loadFinishCallback = null;
+ }
+ that.progressCallback = null;
+ }
+ }, this.createProgCallback(i));
+ }
+}
+
+Resources.prototype.createProgCallback = function(i) {
+ var that = this;
+ return function(progress) {
+ that.progressState[i] = progress;
+ that.updateProgress();
+ }
+}
+
+Resources.prototype.updateProgress = function() {
+ var total = 0;
+ for(var i = 0; i < this.progressState.length; i++) {
+ total += this.progressState[i];
+ }
+ total /= this.progressState.length;
+ this.progressCallback(total);
+}
+
+Resources.prototype.addPack = function(pack) {
+ console.log("Added", pack.name, "to respacks");
+ this.resourcePacks.push(pack);
+ this.addResourcesToArrays(pack);
+ this.rebuildEnabled();
+}
+
+Resources.prototype.addResourcesToArrays = function(pack) {
+ this.allImages = this.allImages.concat(pack.images);
+ this.allSongs = this.allSongs.concat(pack.songs);
+}
+
+Resources.prototype.rebuildArrays = function() {
+ this.allSongs = [];
+ this.allImages = [];
+ this.allAnimations = [];
+
+ for(var i = 0; i < this.resourcePacks.length; i++) {
+ this.addResourcesToArrays(this.resourcePacks[i]);
+ }
+}
+
+Resources.prototype.rebuildEnabled = function() {
+ enabledSongs = [];
+ enabledImages = [];
+ _enabledAnimations = [];
+
+ for(var i = 0; i < this.resourcePacks.length; i++) {
+ var pack = this.resourcePacks[i];
+ if (pack["enabled"] != true) {
+ continue;
+ }
+ for(var j = 0; j < pack.songs.length; j++) {
+ var song = pack.songs[j];
+ if (song.enabled && this.enabledSongs.indexOf(song) == -1) {
+ this.enabledSongs.push(song);
+ }
+ }
+ for(var j = 0; j < pack.images.length; j++) {
+ var image = pack.images[j];
+ if (image.enabled && this.enabledImages.indexOf(image) == -1) {
+ this.enabledImages.push(image);
+ }
+ }
+ }
+}
+
+Resources.prototype.removePack = function(pack) {
+ var index = this.resourcePacks.indexOf(pack);
+ if (index != -1) {
+ this.resourcePacks.splice(index, 1);
+ this.rebuildArrays();
+ }
+}
+
+Resources.prototype.removeAllPacks = function() {
+ this.resourcePacks = [];
+ this.rebuildArrays();
+}
+
+Resources.prototype.getSongNames = function() {
+ var names = []
+ for(var i = 0; i < this.allSongs.length; i++) {
+ names.push(this.allSongs[i]);
+ }
+ return names;
+}
\ No newline at end of file
diff --git a/js/ResourcePack.js b/js/ResourcePack.js
new file mode 100644
index 0000000..3873c24
--- /dev/null
+++ b/js/ResourcePack.js
@@ -0,0 +1,494 @@
+var debugConsole = false;
+function debug() {
+ if(debugConsole) {
+ console.log.apply(window.console, arguments);
+ }
+}
+
+function Respack(url) {
+ this.songs = [];
+ this.songQueue = [];
+ this.images = [];
+ this.imageQueue = [];
+
+ this.name = "
";
+ this.author = "";
+ this.description = "";
+ this.link = null;
+
+ this.size = -1;
+ this.enabled = true;
+
+ this._songFile = null;
+ this._songFileParsed = false;
+ this._imageFile = null;
+ this._infoFile = null;
+
+ this.totalFiles = -1;
+
+ this.file = null;
+ this._completionCallback = null;
+ // For zip parsing progress events
+ this.progressCallback = null;
+ this.filesToLoad = 0;
+ this.filesLoaded = 0;
+ this.loadedFromURL = false;
+
+ if(url)
+ this.loadFromURL(url);
+}
+
+Respack.prototype.audioExtensions = new RegExp("\\.(mp3)$", "i");
+Respack.prototype.imageExtensions = new RegExp("\\.(png|gif|jpg|jpeg)$", "i");
+Respack.prototype.animRegex = new RegExp("(.*?)_\\d+$");
+
+Respack.prototype.updateProgress = function() {
+ if(this.progressCallback) {
+ var percent = this.filesLoaded / this.filesToLoad;
+ if(this.loadedFromURL) {
+ percent = (percent / 2) + 0.5;
+ }
+ this.progressCallback(percent);
+ }
+}
+
+Respack.prototype.loadFromURL = function(url, callback, progress) {
+ var that = this;
+
+ this.progressCallback = progress;
+ this.loadedFromURL = true;
+
+ var req = new XMLHttpRequest();
+ req.open('GET', url, true);
+ req.responseType = 'blob';
+ req.onload = function() {
+ that.loadBlob(req.response, callback);
+ };
+ req.onerror = function() {
+ console.log("Could not load respack at URL", url);
+ }
+ req.onprogress = function(event) {
+ if (event.lengthComputable) {
+ var percent = event.loaded / event.total;
+ if(progress) {
+ progress(percent / 2); // because of processing too
+ }
+ } else {
+ // Unable to compute progress information since the total size is unknown
+ }
+ }
+ req.on
+ req.send();
+}
+
+Respack.prototype.loadBlob = function(blob, callback) {
+ this._completionCallback = callback;
+ var that = this;
+ this.size = blob.size;
+ this.file = new zip.fs.FS();
+ this.file.importBlob(blob,
+ function() { // success
+ that.parseWholeZip();
+ },
+ function(error) { // failure
+ console.log("Error loading respack : ", error.toString());
+ that.file = null;
+ }
+ );
+}
+
+Respack.prototype.parseWholeZip = function() {
+ // TODO might break on bad file
+ console.log("Loading new respack: " + this.file.root.children[0].name);
+
+ var entries = this.file.entries;
+
+ this.totalFiles = 0;
+ // Progress events
+ this.filesToLoad = 0;
+ this.filesLoaded = 0;
+
+ for(var i = 0; i < entries.length; i++) {
+ if(!entries[i].directory && entries[i].name) {
+ this.totalFiles++;
+ this.parseFile(entries[i]);
+ }
+ }
+
+ debug("ZIP loader: trying to finish");
+ this.tryFinish();
+}
+
+Respack.prototype.parseFile = function(file) {
+ var name = file.name
+ if (name.match(this.audioExtensions)) {
+ this.parseSong(file);
+ this.filesToLoad++;
+ } else if (name.match(this.imageExtensions)) {
+ this.parseImage(file);
+ this.filesToLoad++;
+ }
+ else {
+ switch(name.toLowerCase()) {
+ case "songs.xml":
+ this._songFile = file;
+ break;
+ case "images.xml":
+ this._imageFile = file;
+ break;
+ case "info.xml":
+ this._infoFile = file;
+ break;
+ default:
+ }
+ }
+}
+
+Respack.prototype.parseSong = function(file) {
+ this.songQueue.push(file);
+}
+
+Respack.prototype.parseImage = function(file) {
+ this.imageQueue.push(file);
+}
+
+Respack.prototype.parseXML = function() {
+ var that = this;
+
+ if (this.songs.length > 0) {
+ if (this._songFile) {
+ this._songFile.getText(function(text) {
+ //XML parser will complain about a bare '&'
+ text = text.replace(/&/g, '&');
+ that.parseSongFile(text);
+ // Go to next in series
+ that._songFile = null;
+ that._songFileParsed = true;
+ that.parseXML();
+ });
+ return;
+ } else if(!this._songFileParsed) {
+ console.log("!!!", "Got songs but no songs.xml!");
+ this._songFileParsed = true;
+ }
+ }
+ if (this.images.length > 0 && this._imageFile) {
+ this._imageFile.getText(function(text) {
+ text = text.replace(/&/g, '&');
+ that.parseImageFile(text);
+ that._imageFile = null;
+ that.parseXML();
+ });
+ return;
+ }
+ if (this._infoFile) {
+ this._infoFile.getText(function(text) {
+ text = text.replace(/&/g, '&');
+ that.parseInfoFile(text);
+ that._infoFile = null;
+ that.parseXML();
+ });
+ return;
+ }
+
+ // Finally done!
+ this.file = null;
+ console.log("Loaded", this.name, "successfully with", this.songs.length,
+ "songs and", this.images.length, "images.");
+ if(this._completionCallback) {
+ this._completionCallback();
+ }
+}
+
+// Save some chars
+Element.prototype.getTag = function(tag, def) {
+ var t = this.getElementsByTagName(tag)[0];
+ return t ? t.textContent : (def ? def : null);
+}
+
+Respack.prototype.parseSongFile = function(text) {
+ debug(" - Parsing songFile");
+
+ var oParser = new DOMParser();
+ var oDOM = oParser.parseFromString(text, "text/xml");
+ if(oDOM.documentElement.nodeName !== "songs"){
+ console.log("songs.xml error, corrupt file?")
+ return;
+ }
+
+ var newSongs = [];
+ // Not supported in mobile safari
+ // var songsXML = oDOM.documentElement.children;
+ var el = oDOM.documentElement.firstElementChild;
+ for(; el; el = el.nextElementSibling) {
+ var song = this.getSong(el.attributes[0].value);
+ if(song) {
+ song.title = el.getTag("title");
+ if(!song.title) {
+ song.title = "";
+ debug(" WARNING!", song.name, "has no title!");
+ }
+
+ song.rhythm = el.getTag("rhythm");
+ if(!song.rhythm) {
+ song.rhythm = "..no..rhythm..";
+ debug(" WARNING!!", song.name, "has no rhythm!!");
+ }
+
+ song.startSilence = el.getTag("startSilence");
+ song.endSilence = el.getTag("endSilence");
+ song.buildupLength = el.getTag("buildupLength");
+ if(song.buildupLength) {
+ debug("Using custom BU length:", song.buildupLength);
+ }
+ song.buildup = el.getTag("buildup");
+ if(song.buildup) {
+ debug(" Finding a buildup '" + song.buildup + "' for ", song.name);
+ var build = this.getSong(song.buildup);
+ if(build) {
+ song.buildup = build.sound;
+ song.buildupPlayed = false;
+ // get rid of the junk
+ this.songs.splice(this.songs.indexOf(build), 1);
+ } else {
+ debug(" WARNING!", "Didn't find a buildup '" + buildup + "'!");
+ }
+ }
+
+ song.buildupRhythm = el.getTag("buildupRhythm");
+ song.source = el.getTag("source");
+
+ newSongs.push(song);
+ debug(" [I] " + song.name, ": '" + song.title + "' added to songs");
+ } else {
+ debug(" WARNING!", "songs.xml: element", i + 1,
+ "- no song '" + el.attributes[0].value + "' found");
+ }
+ }
+ for(var i = 0; i < this.songs.length; i++) {
+ if(newSongs.indexOf(this.songs[i]) == -1) {
+ debug(" WARNING!", "We have a file for", song.name, "but no information for it");
+ }
+ }
+ this.songs = newSongs;
+}
+
+Respack.prototype.parseInfoFile = function(text) {
+ debug(" - Parsing infoFile");
+
+ var oParser = new DOMParser();
+ var oDOM = oParser.parseFromString(text, "text/xml");
+ var info = oDOM.documentElement;
+ if(info.nodeName !== "info"){
+ console.log("info.xml error, corrupt file?")
+ return;
+ }
+
+ // self reference strings to avoid changing strings twice in future
+ this.name = info.getTag("name", this.name);
+ this.author = info.getTag("author", this.author);
+ this.description = info.getTag("description", this.description);
+ this.link = info.getTag("link", this.link);
+}
+
+Respack.prototype.parseImageFile = function(text) {
+ debug(" - Parsing imagefile");
+
+ var oParser = new DOMParser();
+ var oDOM = oParser.parseFromString(text, "text/xml");
+ if(oDOM.documentElement.nodeName !== "images"){
+ console.log("images.xml error, corrupt file?")
+ return;
+ }
+
+ var newImages = [];
+ // not in mobile safari
+ // var imagesXML = oDOM.documentElement.children;
+ var el = oDOM.documentElement.firstElementChild;
+ for(; el; el = el.nextElementSibling) {
+ var image = this.getImage(el.attributes[0].value);
+ if(image) {
+ image.fullname = el.getTag("fullname");
+ if(!image.fullname) {
+ debug(" WARNING!", image.name, "has no full name!");
+ }
+ image.source = el.getTag("source");
+ // self reference strings to avoid changing strings twice in future
+ image.align = el.getTag("align", image.align);
+ var frameDur = el.getTag("frameDuration");
+ if(frameDur) {
+ image.frameDurations = []
+ var strSplit = frameDur.split(",");
+ for(var j = 0; j < strSplit.length; j++) {
+ image.frameDurations.push(parseInt(strSplit[j]));
+ }
+ while (image.frameDurations.length < image.bitmaps.length) {
+ image.frameDurations.push(image.frameDurations[image.frameDurations.length - 1]);
+ }
+ debug("Frame durations:", image.frameDurations);
+ }
+ debug(" [I] " + image.name, ":", image.fullname, "added to images");
+ if (image.bitmap || image.bitmaps) {
+ newImages.push(image);
+ }
+ else {
+ debug(" WARNING!!", "Image", image.name, "has no bitmap nor animation frames!");
+ }
+ } else {
+ debug(" WARNING!", "images.xml: element",
+ i + 1, "- no image '" + el.attributes[0].value + "' found");
+ }
+ }
+ for(var i = 0; i < this.images.length; i++) {
+ var image = this.images[i];
+ // Add all images with no info
+ if(newImages.indexOf(image) == -1) {
+ newImages.push(image);
+ }
+ }
+ this.images = newImages;
+}
+
+Respack.prototype.containsSong = function(name) {
+ return this.getSong(name) !== null;
+}
+
+Respack.prototype.containsImage = function(name) {
+ return this.getImage(name) !== null;
+}
+
+Respack.prototype.getSong = function(name) {
+ for(var i = 0; i < this.songs.length; i++) {
+ if (name == this.songs[i].name) {
+ return this.songs[i];
+ }
+ }
+ return null;
+}
+
+Respack.prototype.getImage = function(name) {
+ for(var i = 0; i < this.images.length; i++) {
+ if (name == this.images[i].name) {
+ return this.images[i];
+ }
+ }
+ return null;
+}
+
+Respack.prototype.parseSongQueue = function() {
+ var that = this;
+ var songFile = this.songQueue.shift();
+ var name = songFile.name.replace(this.audioExtensions, "");
+
+ debug("parsing song: " + name);
+ if (this.containsSong(name)) {
+ var oldSong = this.getSong(name);
+ debug("WARNING: Song", name, "already exists! Conflict with", name, "and", oldSong.name);
+ } else {
+ var newSong = {"name":name,
+ "title":null,
+ "rhythm":null,
+ "source":null,
+ //"crc":this.quickCRC(file), TODO
+ "sound":null,
+ "enabled":true,
+ "filename":songFile.name};
+ songFile.getBlob("audio/mpeg3", function(sound) {
+ // Because blobs are crap
+ var fr = new FileReader();
+ fr.onload = function() {
+ newSong.sound = this.result;
+ that.filesLoaded++;
+ that.updateProgress();
+ that.tryFinish();
+ };
+ fr.readAsArrayBuffer(sound);
+ });
+ this.songs.push(newSong);
+ }
+}
+
+Respack.prototype.parseImageQueue = function() {
+ var imgFile = this.imageQueue.shift();
+ var name = imgFile.name.replace(this.imageExtensions, "");
+
+ if (match = name.match(new RegExp("^(.*)_(\\d+)$"))) {
+ var anim = this.getImage(match[1])
+ if(!anim) { // make a fresh one
+ anim = {"name":match[1],
+ "fullname":match[1],
+ "align":"center",
+ //"crc":this.quickCRC(imgFile),
+ "bitmaps":[],
+ "frameDurations":[33],
+ "source":null,
+ "enabled":true,
+ "animated":true};
+ this.images.push(anim);
+ }
+ this.imageLoadStart(imgFile, anim);
+ } else if (!this.containsImage(name)) {
+ var img = {"name":name,
+ "fullname":name,
+ "bitmap":null,
+ "align":"center",
+ //"crc":this.quickCRC(imgFile),
+ "source":null,
+ "enabled":true,
+ "filename":imgFile.name,
+ "animated":false};
+ this.images.push(img);
+ this.imageLoadStart(imgFile, img);
+ } else {
+ existing = this.getImage(name);
+ debug("WARNING: Image", name, "already exists! Conflict with", imgFile.name, "and", existing.name);
+ }
+}
+
+Respack.prototype.imageLoadStart = function(imgFile, imageObj) {
+ var that = this;
+ var extension = imgFile.name.split('.').pop().toLowerCase();
+ var mime;
+ switch(extension) {
+ case "png":
+ mime = "image/png";
+ break;
+ case "gif":
+ mime = "image/gif";
+ break;
+ case "jpg":
+ case "jpeg":
+ mime = "image/jpeg";
+ break;
+ default:
+ mime = "application/octet-stream";
+ }
+ imgFile.getData64URI(mime, function(image) {
+ that.imageLoadComplete(image, imageObj);
+ });
+}
+
+Respack.prototype.imageLoadComplete = function(imageBmp, imageObj) {
+ newImg = new Image();
+ newImg.src = imageBmp;
+ if (imageObj.animated) {
+ imageObj.bitmaps.push(newImg);
+ } else {
+ imageObj.bitmap = newImg;
+ debug("parsing image:", imageObj.name);
+ }
+ this.filesLoaded++;
+ this.updateProgress();
+ this.tryFinish();
+}
+
+Respack.prototype.tryFinish = function() {
+ if (this.imageQueue.length > 0) {
+ this.parseImageQueue();
+ } else if(this.songQueue.length > 0) {
+ this.parseSongQueue();
+ } else {
+ debug("Finished parsing images/songs, parsing xml files...");
+ this.parseXML();
+ }
+}
\ No newline at end of file
diff --git a/js/SoundManager.js b/js/SoundManager.js
new file mode 100644
index 0000000..1763457
--- /dev/null
+++ b/js/SoundManager.js
@@ -0,0 +1,258 @@
+var LAME_DELAY_START = 2258;
+var LAME_DELAY_END = 1000;
+
+function SoundManager() {
+ this.playing = false;
+ this.song = null;
+
+ /* Lower level audio and timing info */
+ this.bufSource = null;
+ this.buffer = null;
+ this.context = null; // Audio context, Web Audio API
+ this.startTime = 0; // File start time - 0 is loop start, not build start
+ this.loopStart = 0; // When the build ends, if any
+ this.loopLength = 0; // For calculating beat lengths
+
+ // Volume
+ this.gainNode = null;
+ this.mute = false;
+ this.lastVol = 1;
+
+ // For concatenating our files
+ this.leftToLoad = 0;
+ this.tmpBuffer = null;
+ this.tmpBuild = null;
+ this.onLoadCallback = null;
+
+ // In case of API non-support
+ this.canUse = true;
+
+ // Check Web Audio API Support
+ try {
+ // More info at http://caniuse.com/#feat=audio-api
+ window.AudioContext = window.AudioContext || window.webkitAudioContext;
+ this.context = new window.AudioContext();
+ this.gainNode = this.context.createGain();
+ this.gainNode.connect(this.context.destination);
+ } catch(e) {
+ this.canUse = false;
+ }
+
+ var that = this;
+ window.addEventListener('touchstart', function() {
+
+ // create empty buffer
+ var buffer = that.context.createBuffer(1, 1, 22050);
+ var source = that.context.createBufferSource();
+ source.buffer = buffer;
+
+ // connect to output (your speakers)
+ source.connect( that.context.destination);
+
+ // play the file
+ source.noteOn(0);
+
+ }, false);
+}
+
+SoundManager.prototype.playSong = function(song, callback) {
+ var that = this;
+ if(this.song == song) {
+ return;
+ }
+ this.stop();
+ this.song = song;
+
+ this.loadBuffer(song, function() {
+ // To prevent race condition if you press "next" twice fast
+ if(song == that.song) {
+ that.startTime = that.context.currentTime + that.loopStart;
+ that.bufSource = that.context.createBufferSource();
+ that.bufSource.buffer = that.buffer;
+ that.bufSource.loop = true;
+ that.bufSource.loopStart = that.loopStart;
+ that.bufSource.loopEnd = that.buffer.duration;
+ that.bufSource.connect(that.gainNode);
+
+ // Mobile Safari requires offset, even if 0
+ that.bufSource.start(0);
+ // offset to after the build
+ //that.startTime = that.context.currentTime + that.loopStart;
+ that.playing = true;
+ if(callback)
+ callback();
+ }
+ });
+}
+
+SoundManager.prototype.stop = function() {
+ if (this.playing) {
+ this.bufSource.stop();
+ this.bufSource.disconnect(); // TODO needed?
+ this.bufSource = null;
+ this.playing = false;
+ this.startTime = 0;
+ this.loopStart = 0;
+ this.loopLength = 0;
+ }
+}
+
+// In seconds, relative to the loop start
+SoundManager.prototype.currentTime = function() {
+ if(!this.playing) {
+ return 0;
+ }
+ return this.context.currentTime - this.startTime;
+}
+
+SoundManager.prototype.displayableTime = function() {
+ if(!this.playing) {
+ return 0;
+ }
+ var time = this.currentTime();
+ if(time < 0) {
+ return 0;
+ } else {
+ return time % this.loopLength;
+ }
+}
+
+SoundManager.prototype.loadBuffer = function(song, callback) {
+ if(callback) {
+ this.onLoadCallback = callback;
+ }
+ if(song.buildup) {
+ this.loadAudioFile(song, true);
+ }
+ this.loadAudioFile(song, false);
+}
+
+SoundManager.prototype.loadAudioFile = function(song, isBuild) {
+ this.context.decodeAudioData(
+ isBuild ? song.buildup : song.sound,
+ this.getAudioCallback(song, isBuild),
+ function() {
+ console.log('Error decoding audio "' + song.name + '".');
+ }
+ );
+}
+
+/* decodeAudioData nukes our original MP3 array, but we want to keep it around
+ for memory saving purposes, so we must duplicate it locally here */
+SoundManager.prototype.getAudioCallback = function(song, isBuild) {
+ var that = this;
+ var current = isBuild ? song.buildup : song.sound;
+ var copy = current.slice(0);
+ return function(buffer) {
+ // before the race condition check or we might lose data
+ if(isBuild) {
+ song.buildup = copy;
+ } else {
+ song.sound = copy;
+ }
+ // race condition prevention
+ if(that.song != song) {
+ return;
+ }
+ if(isBuild) {
+ that.tmpBuild = that.trimMP3(buffer);
+ } else {
+ that.tmpBuffer = that.trimMP3(buffer);
+ }
+ that.onSongLoad(song);
+ };
+}
+
+SoundManager.prototype.onSongLoad = function(song) {
+ // if this fails, we need to wait for the other part to load
+ if(this.tmpBuffer && (!song.buildup || this.tmpBuild)) {
+ if(song.buildup) {
+ this.buffer = this.concatenateAudioBuffers(this.tmpBuild, this.tmpBuffer);
+ this.loopStart = this.tmpBuild.duration;
+ } else {
+ this.buffer = this.tmpBuffer;
+ this.loopStart = 0;
+ }
+ this.loopLength = this.buffer.duration - this.loopStart;
+ // free dat memory
+ this.tmpBuild = this.tmpBuffer = null;
+ if(this.onLoadCallback) {
+ this.onLoadCallback();
+ this.onLoadCallback = null;
+ }
+ }
+}
+
+// because MP3 is bad, we nuke silence
+SoundManager.prototype.trimMP3 = function(buffer) {
+ var isFirefox = navigator.userAgent.toLowerCase().indexOf('firefox') > -1;
+ if(!isFirefox) {
+ // Webkit is better than Gecko, clearly
+ return buffer;
+ }
+ var ret = this.context.createBuffer(buffer.numberOfChannels,
+ buffer.length - LAME_DELAY_START - LAME_DELAY_END, buffer.sampleRate);
+ for(var i=0; i max_length) {
+ bits = max_length;
+ overflow++;
+ }
+ tree[n * 2 + 1] = bits;
+ // We overwrite tree[n*2+1] which is no longer needed
+
+ if (n > that.max_code)
+ continue; // not a leaf node
+
+ s.bl_count[bits]++;
+ xbits = 0;
+ if (n >= base)
+ xbits = extra[n - base];
+ f = tree[n * 2];
+ s.opt_len += f * (bits + xbits);
+ if (stree)
+ s.static_len += f * (stree[n * 2 + 1] + xbits);
+ }
+ if (overflow === 0)
+ return;
+
+ // This happens for example on obj2 and pic of the Calgary corpus
+ // Find the first bit length which could increase:
+ do {
+ bits = max_length - 1;
+ while (s.bl_count[bits] === 0)
+ bits--;
+ s.bl_count[bits]--; // move one leaf down the tree
+ s.bl_count[bits + 1] += 2; // move one overflow item as its brother
+ s.bl_count[max_length]--;
+ // The brother of the overflow item also moves one step up,
+ // but this does not affect bl_count[max_length]
+ overflow -= 2;
+ } while (overflow > 0);
+
+ for (bits = max_length; bits !== 0; bits--) {
+ n = s.bl_count[bits];
+ while (n !== 0) {
+ m = s.heap[--h];
+ if (m > that.max_code)
+ continue;
+ if (tree[m * 2 + 1] != bits) {
+ s.opt_len += (bits - tree[m * 2 + 1]) * tree[m * 2];
+ tree[m * 2 + 1] = bits;
+ }
+ n--;
+ }
+ }
+ }
+
+ // Reverse the first len bits of a code, using straightforward code (a
+ // faster
+ // method would use a table)
+ // IN assertion: 1 <= len <= 15
+ function bi_reverse(code, // the value to invert
+ len // its bit length
+ ) {
+ var res = 0;
+ do {
+ res |= code & 1;
+ code >>>= 1;
+ res <<= 1;
+ } while (--len > 0);
+ return res >>> 1;
+ }
+
+ // Generate the codes for a given tree and bit counts (which need not be
+ // optimal).
+ // IN assertion: the array bl_count contains the bit length statistics for
+ // the given tree and the field len is set for all tree elements.
+ // OUT assertion: the field code is set for all tree elements of non
+ // zero code length.
+ function gen_codes(tree, // the tree to decorate
+ max_code, // largest code with non zero frequency
+ bl_count // number of codes at each bit length
+ ) {
+ var next_code = []; // next code value for each
+ // bit length
+ var code = 0; // running code value
+ var bits; // bit index
+ var n; // code index
+ var len;
+
+ // The distribution counts are first used to generate the code values
+ // without bit reversal.
+ for (bits = 1; bits <= MAX_BITS; bits++) {
+ next_code[bits] = code = ((code + bl_count[bits - 1]) << 1);
+ }
+
+ // Check that the bit counts in bl_count are consistent. The last code
+ // must be all ones.
+ // Assert (code + bl_count[MAX_BITS]-1 == (1<= 1; n--)
+ s.pqdownheap(tree, n);
+
+ // Construct the Huffman tree by repeatedly combining the least two
+ // frequent nodes.
+
+ node = elems; // next internal node of the tree
+ do {
+ // n = node of least frequency
+ n = s.heap[1];
+ s.heap[1] = s.heap[s.heap_len--];
+ s.pqdownheap(tree, 1);
+ m = s.heap[1]; // m = node of next least frequency
+
+ s.heap[--s.heap_max] = n; // keep the nodes sorted by frequency
+ s.heap[--s.heap_max] = m;
+
+ // Create a new node father of n and m
+ tree[node * 2] = (tree[n * 2] + tree[m * 2]);
+ s.depth[node] = Math.max(s.depth[n], s.depth[m]) + 1;
+ tree[n * 2 + 1] = tree[m * 2 + 1] = node;
+
+ // and insert the new node in the heap
+ s.heap[1] = node++;
+ s.pqdownheap(tree, 1);
+ } while (s.heap_len >= 2);
+
+ s.heap[--s.heap_max] = s.heap[1];
+
+ // At this point, the fields freq and dad are set. We can now
+ // generate the bit lengths.
+
+ gen_bitlen(s);
+
+ // The field len is now set, we can generate the bit codes
+ gen_codes(tree, that.max_code, s.bl_count);
+ };
+
+ }
+
+ Tree._length_code = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 12, 12, 13, 13, 13, 13, 14, 14, 14, 14, 15, 15, 15, 15, 16, 16, 16, 16,
+ 16, 16, 16, 16, 17, 17, 17, 17, 17, 17, 17, 17, 18, 18, 18, 18, 18, 18, 18, 18, 19, 19, 19, 19, 19, 19, 19, 19, 20, 20, 20, 20, 20, 20, 20, 20, 20,
+ 20, 20, 20, 20, 20, 20, 20, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22,
+ 22, 22, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
+ 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25,
+ 25, 25, 25, 25, 25, 25, 25, 25, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
+ 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 28 ];
+
+ Tree.base_length = [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 10, 12, 14, 16, 20, 24, 28, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 0 ];
+
+ Tree.base_dist = [ 0, 1, 2, 3, 4, 6, 8, 12, 16, 24, 32, 48, 64, 96, 128, 192, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096, 6144, 8192, 12288, 16384,
+ 24576 ];
+
+ // Mapping from a distance to a distance code. dist is the distance - 1 and
+ // must not have side effects. _dist_code[256] and _dist_code[257] are never
+ // used.
+ Tree.d_code = function(dist) {
+ return ((dist) < 256 ? _dist_code[dist] : _dist_code[256 + ((dist) >>> 7)]);
+ };
+
+ // extra bits for each length code
+ Tree.extra_lbits = [ 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0 ];
+
+ // extra bits for each distance code
+ Tree.extra_dbits = [ 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 ];
+
+ // extra bits for each bit length code
+ Tree.extra_blbits = [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 3, 7 ];
+
+ Tree.bl_order = [ 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 ];
+
+ // StaticTree
+
+ function StaticTree(static_tree, extra_bits, extra_base, elems, max_length) {
+ var that = this;
+ that.static_tree = static_tree;
+ that.extra_bits = extra_bits;
+ that.extra_base = extra_base;
+ that.elems = elems;
+ that.max_length = max_length;
+ }
+
+ StaticTree.static_ltree = [ 12, 8, 140, 8, 76, 8, 204, 8, 44, 8, 172, 8, 108, 8, 236, 8, 28, 8, 156, 8, 92, 8, 220, 8, 60, 8, 188, 8, 124, 8, 252, 8, 2, 8,
+ 130, 8, 66, 8, 194, 8, 34, 8, 162, 8, 98, 8, 226, 8, 18, 8, 146, 8, 82, 8, 210, 8, 50, 8, 178, 8, 114, 8, 242, 8, 10, 8, 138, 8, 74, 8, 202, 8, 42,
+ 8, 170, 8, 106, 8, 234, 8, 26, 8, 154, 8, 90, 8, 218, 8, 58, 8, 186, 8, 122, 8, 250, 8, 6, 8, 134, 8, 70, 8, 198, 8, 38, 8, 166, 8, 102, 8, 230, 8,
+ 22, 8, 150, 8, 86, 8, 214, 8, 54, 8, 182, 8, 118, 8, 246, 8, 14, 8, 142, 8, 78, 8, 206, 8, 46, 8, 174, 8, 110, 8, 238, 8, 30, 8, 158, 8, 94, 8,
+ 222, 8, 62, 8, 190, 8, 126, 8, 254, 8, 1, 8, 129, 8, 65, 8, 193, 8, 33, 8, 161, 8, 97, 8, 225, 8, 17, 8, 145, 8, 81, 8, 209, 8, 49, 8, 177, 8, 113,
+ 8, 241, 8, 9, 8, 137, 8, 73, 8, 201, 8, 41, 8, 169, 8, 105, 8, 233, 8, 25, 8, 153, 8, 89, 8, 217, 8, 57, 8, 185, 8, 121, 8, 249, 8, 5, 8, 133, 8,
+ 69, 8, 197, 8, 37, 8, 165, 8, 101, 8, 229, 8, 21, 8, 149, 8, 85, 8, 213, 8, 53, 8, 181, 8, 117, 8, 245, 8, 13, 8, 141, 8, 77, 8, 205, 8, 45, 8,
+ 173, 8, 109, 8, 237, 8, 29, 8, 157, 8, 93, 8, 221, 8, 61, 8, 189, 8, 125, 8, 253, 8, 19, 9, 275, 9, 147, 9, 403, 9, 83, 9, 339, 9, 211, 9, 467, 9,
+ 51, 9, 307, 9, 179, 9, 435, 9, 115, 9, 371, 9, 243, 9, 499, 9, 11, 9, 267, 9, 139, 9, 395, 9, 75, 9, 331, 9, 203, 9, 459, 9, 43, 9, 299, 9, 171, 9,
+ 427, 9, 107, 9, 363, 9, 235, 9, 491, 9, 27, 9, 283, 9, 155, 9, 411, 9, 91, 9, 347, 9, 219, 9, 475, 9, 59, 9, 315, 9, 187, 9, 443, 9, 123, 9, 379,
+ 9, 251, 9, 507, 9, 7, 9, 263, 9, 135, 9, 391, 9, 71, 9, 327, 9, 199, 9, 455, 9, 39, 9, 295, 9, 167, 9, 423, 9, 103, 9, 359, 9, 231, 9, 487, 9, 23,
+ 9, 279, 9, 151, 9, 407, 9, 87, 9, 343, 9, 215, 9, 471, 9, 55, 9, 311, 9, 183, 9, 439, 9, 119, 9, 375, 9, 247, 9, 503, 9, 15, 9, 271, 9, 143, 9,
+ 399, 9, 79, 9, 335, 9, 207, 9, 463, 9, 47, 9, 303, 9, 175, 9, 431, 9, 111, 9, 367, 9, 239, 9, 495, 9, 31, 9, 287, 9, 159, 9, 415, 9, 95, 9, 351, 9,
+ 223, 9, 479, 9, 63, 9, 319, 9, 191, 9, 447, 9, 127, 9, 383, 9, 255, 9, 511, 9, 0, 7, 64, 7, 32, 7, 96, 7, 16, 7, 80, 7, 48, 7, 112, 7, 8, 7, 72, 7,
+ 40, 7, 104, 7, 24, 7, 88, 7, 56, 7, 120, 7, 4, 7, 68, 7, 36, 7, 100, 7, 20, 7, 84, 7, 52, 7, 116, 7, 3, 8, 131, 8, 67, 8, 195, 8, 35, 8, 163, 8,
+ 99, 8, 227, 8 ];
+
+ StaticTree.static_dtree = [ 0, 5, 16, 5, 8, 5, 24, 5, 4, 5, 20, 5, 12, 5, 28, 5, 2, 5, 18, 5, 10, 5, 26, 5, 6, 5, 22, 5, 14, 5, 30, 5, 1, 5, 17, 5, 9, 5,
+ 25, 5, 5, 5, 21, 5, 13, 5, 29, 5, 3, 5, 19, 5, 11, 5, 27, 5, 7, 5, 23, 5 ];
+
+ StaticTree.static_l_desc = new StaticTree(StaticTree.static_ltree, Tree.extra_lbits, LITERALS + 1, L_CODES, MAX_BITS);
+
+ StaticTree.static_d_desc = new StaticTree(StaticTree.static_dtree, Tree.extra_dbits, 0, D_CODES, MAX_BITS);
+
+ StaticTree.static_bl_desc = new StaticTree(null, Tree.extra_blbits, 0, BL_CODES, MAX_BL_BITS);
+
+ // Deflate
+
+ var MAX_MEM_LEVEL = 9;
+ var DEF_MEM_LEVEL = 8;
+
+ function Config(good_length, max_lazy, nice_length, max_chain, func) {
+ var that = this;
+ that.good_length = good_length;
+ that.max_lazy = max_lazy;
+ that.nice_length = nice_length;
+ that.max_chain = max_chain;
+ that.func = func;
+ }
+
+ var STORED = 0;
+ var FAST = 1;
+ var SLOW = 2;
+ var config_table = [ new Config(0, 0, 0, 0, STORED), new Config(4, 4, 8, 4, FAST), new Config(4, 5, 16, 8, FAST), new Config(4, 6, 32, 32, FAST),
+ new Config(4, 4, 16, 16, SLOW), new Config(8, 16, 32, 32, SLOW), new Config(8, 16, 128, 128, SLOW), new Config(8, 32, 128, 256, SLOW),
+ new Config(32, 128, 258, 1024, SLOW), new Config(32, 258, 258, 4096, SLOW) ];
+
+ var z_errmsg = [ "need dictionary", // Z_NEED_DICT
+ // 2
+ "stream end", // Z_STREAM_END 1
+ "", // Z_OK 0
+ "", // Z_ERRNO (-1)
+ "stream error", // Z_STREAM_ERROR (-2)
+ "data error", // Z_DATA_ERROR (-3)
+ "", // Z_MEM_ERROR (-4)
+ "buffer error", // Z_BUF_ERROR (-5)
+ "",// Z_VERSION_ERROR (-6)
+ "" ];
+
+ // block not completed, need more input or more output
+ var NeedMore = 0;
+
+ // block flush performed
+ var BlockDone = 1;
+
+ // finish started, need only more output at next deflate
+ var FinishStarted = 2;
+
+ // finish done, accept no more input or output
+ var FinishDone = 3;
+
+ // preset dictionary flag in zlib header
+ var PRESET_DICT = 0x20;
+
+ var INIT_STATE = 42;
+ var BUSY_STATE = 113;
+ var FINISH_STATE = 666;
+
+ // The deflate compression method
+ var Z_DEFLATED = 8;
+
+ var STORED_BLOCK = 0;
+ var STATIC_TREES = 1;
+ var DYN_TREES = 2;
+
+ var MIN_MATCH = 3;
+ var MAX_MATCH = 258;
+ var MIN_LOOKAHEAD = (MAX_MATCH + MIN_MATCH + 1);
+
+ function smaller(tree, n, m, depth) {
+ var tn2 = tree[n * 2];
+ var tm2 = tree[m * 2];
+ return (tn2 < tm2 || (tn2 == tm2 && depth[n] <= depth[m]));
+ }
+
+ function Deflate() {
+
+ var that = this;
+ var strm; // pointer back to this zlib stream
+ var status; // as the name implies
+ // pending_buf; // output still pending
+ var pending_buf_size; // size of pending_buf
+ // pending_out; // next pending byte to output to the stream
+ // pending; // nb of bytes in the pending buffer
+ var method; // STORED (for zip only) or DEFLATED
+ var last_flush; // value of flush param for previous deflate call
+
+ var w_size; // LZ77 window size (32K by default)
+ var w_bits; // log2(w_size) (8..16)
+ var w_mask; // w_size - 1
+
+ var window;
+ // Sliding window. Input bytes are read into the second half of the window,
+ // and move to the first half later to keep a dictionary of at least wSize
+ // bytes. With this organization, matches are limited to a distance of
+ // wSize-MAX_MATCH bytes, but this ensures that IO is always
+ // performed with a length multiple of the block size. Also, it limits
+ // the window size to 64K, which is quite useful on MSDOS.
+ // To do: use the user input buffer as sliding window.
+
+ var window_size;
+ // Actual size of window: 2*wSize, except when the user input buffer
+ // is directly used as sliding window.
+
+ var prev;
+ // Link to older string with same hash index. To limit the size of this
+ // array to 64K, this link is maintained only for the last 32K strings.
+ // An index in this array is thus a window index modulo 32K.
+
+ var head; // Heads of the hash chains or NIL.
+
+ var ins_h; // hash index of string to be inserted
+ var hash_size; // number of elements in hash table
+ var hash_bits; // log2(hash_size)
+ var hash_mask; // hash_size-1
+
+ // Number of bits by which ins_h must be shifted at each input
+ // step. It must be such that after MIN_MATCH steps, the oldest
+ // byte no longer takes part in the hash key, that is:
+ // hash_shift * MIN_MATCH >= hash_bits
+ var hash_shift;
+
+ // Window position at the beginning of the current output block. Gets
+ // negative when the window is moved backwards.
+
+ var block_start;
+
+ var match_length; // length of best match
+ var prev_match; // previous match
+ var match_available; // set if previous match exists
+ var strstart; // start of string to insert
+ var match_start; // start of matching string
+ var lookahead; // number of valid bytes ahead in window
+
+ // Length of the best match at previous step. Matches not greater than this
+ // are discarded. This is used in the lazy match evaluation.
+ var prev_length;
+
+ // To speed up deflation, hash chains are never searched beyond this
+ // length. A higher limit improves compression ratio but degrades the speed.
+ var max_chain_length;
+
+ // Attempt to find a better match only when the current match is strictly
+ // smaller than this value. This mechanism is used only for compression
+ // levels >= 4.
+ var max_lazy_match;
+
+ // Insert new strings in the hash table only if the match length is not
+ // greater than this length. This saves time but degrades compression.
+ // max_insert_length is used only for compression levels <= 3.
+
+ var level; // compression level (1..9)
+ var strategy; // favor or force Huffman coding
+
+ // Use a faster search when the previous match is longer than this
+ var good_match;
+
+ // Stop searching when current match exceeds this
+ var nice_match;
+
+ var dyn_ltree; // literal and length tree
+ var dyn_dtree; // distance tree
+ var bl_tree; // Huffman tree for bit lengths
+
+ var l_desc = new Tree(); // desc for literal tree
+ var d_desc = new Tree(); // desc for distance tree
+ var bl_desc = new Tree(); // desc for bit length tree
+
+ // that.heap_len; // number of elements in the heap
+ // that.heap_max; // element of largest frequency
+ // The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used.
+ // The same heap array is used to build all trees.
+
+ // Depth of each subtree used as tie breaker for trees of equal frequency
+ that.depth = [];
+
+ var l_buf; // index for literals or lengths */
+
+ // Size of match buffer for literals/lengths. There are 4 reasons for
+ // limiting lit_bufsize to 64K:
+ // - frequencies can be kept in 16 bit counters
+ // - if compression is not successful for the first block, all input
+ // data is still in the window so we can still emit a stored block even
+ // when input comes from standard input. (This can also be done for
+ // all blocks if lit_bufsize is not greater than 32K.)
+ // - if compression is not successful for a file smaller than 64K, we can
+ // even emit a stored file instead of a stored block (saving 5 bytes).
+ // This is applicable only for zip (not gzip or zlib).
+ // - creating new Huffman trees less frequently may not provide fast
+ // adaptation to changes in the input data statistics. (Take for
+ // example a binary file with poorly compressible code followed by
+ // a highly compressible string table.) Smaller buffer sizes give
+ // fast adaptation but have of course the overhead of transmitting
+ // trees more frequently.
+ // - I can't count above 4
+ var lit_bufsize;
+
+ var last_lit; // running index in l_buf
+
+ // Buffer for distances. To simplify the code, d_buf and l_buf have
+ // the same number of elements. To use different lengths, an extra flag
+ // array would be necessary.
+
+ var d_buf; // index of pendig_buf
+
+ // that.opt_len; // bit length of current block with optimal trees
+ // that.static_len; // bit length of current block with static trees
+ var matches; // number of string matches in current block
+ var last_eob_len; // bit length of EOB code for last block
+
+ // Output buffer. bits are inserted starting at the bottom (least
+ // significant bits).
+ var bi_buf;
+
+ // Number of valid bits in bi_buf. All bits above the last valid bit
+ // are always zero.
+ var bi_valid;
+
+ // number of codes at each bit length for an optimal tree
+ that.bl_count = [];
+
+ // heap used to build the Huffman trees
+ that.heap = [];
+
+ dyn_ltree = [];
+ dyn_dtree = [];
+ bl_tree = [];
+
+ function lm_init() {
+ var i;
+ window_size = 2 * w_size;
+
+ head[hash_size - 1] = 0;
+ for (i = 0; i < hash_size - 1; i++) {
+ head[i] = 0;
+ }
+
+ // Set the default configuration parameters:
+ max_lazy_match = config_table[level].max_lazy;
+ good_match = config_table[level].good_length;
+ nice_match = config_table[level].nice_length;
+ max_chain_length = config_table[level].max_chain;
+
+ strstart = 0;
+ block_start = 0;
+ lookahead = 0;
+ match_length = prev_length = MIN_MATCH - 1;
+ match_available = 0;
+ ins_h = 0;
+ }
+
+ function init_block() {
+ var i;
+ // Initialize the trees.
+ for (i = 0; i < L_CODES; i++)
+ dyn_ltree[i * 2] = 0;
+ for (i = 0; i < D_CODES; i++)
+ dyn_dtree[i * 2] = 0;
+ for (i = 0; i < BL_CODES; i++)
+ bl_tree[i * 2] = 0;
+
+ dyn_ltree[END_BLOCK * 2] = 1;
+ that.opt_len = that.static_len = 0;
+ last_lit = matches = 0;
+ }
+
+ // Initialize the tree data structures for a new zlib stream.
+ function tr_init() {
+
+ l_desc.dyn_tree = dyn_ltree;
+ l_desc.stat_desc = StaticTree.static_l_desc;
+
+ d_desc.dyn_tree = dyn_dtree;
+ d_desc.stat_desc = StaticTree.static_d_desc;
+
+ bl_desc.dyn_tree = bl_tree;
+ bl_desc.stat_desc = StaticTree.static_bl_desc;
+
+ bi_buf = 0;
+ bi_valid = 0;
+ last_eob_len = 8; // enough lookahead for inflate
+
+ // Initialize the first block of the first file:
+ init_block();
+ }
+
+ // Restore the heap property by moving down the tree starting at node k,
+ // exchanging a node with the smallest of its two sons if necessary,
+ // stopping
+ // when the heap property is re-established (each father smaller than its
+ // two sons).
+ that.pqdownheap = function(tree, // the tree to restore
+ k // node to move down
+ ) {
+ var heap = that.heap;
+ var v = heap[k];
+ var j = k << 1; // left son of k
+ while (j <= that.heap_len) {
+ // Set j to the smallest of the two sons:
+ if (j < that.heap_len && smaller(tree, heap[j + 1], heap[j], that.depth)) {
+ j++;
+ }
+ // Exit if v is smaller than both sons
+ if (smaller(tree, v, heap[j], that.depth))
+ break;
+
+ // Exchange v with the smallest son
+ heap[k] = heap[j];
+ k = j;
+ // And continue down the tree, setting j to the left son of k
+ j <<= 1;
+ }
+ heap[k] = v;
+ };
+
+ // Scan a literal or distance tree to determine the frequencies of the codes
+ // in the bit length tree.
+ function scan_tree(tree,// the tree to be scanned
+ max_code // and its largest code of non zero frequency
+ ) {
+ var n; // iterates over all tree elements
+ var prevlen = -1; // last emitted length
+ var curlen; // length of current code
+ var nextlen = tree[0 * 2 + 1]; // length of next code
+ var count = 0; // repeat count of the current code
+ var max_count = 7; // max repeat count
+ var min_count = 4; // min repeat count
+
+ if (nextlen === 0) {
+ max_count = 138;
+ min_count = 3;
+ }
+ tree[(max_code + 1) * 2 + 1] = 0xffff; // guard
+
+ for (n = 0; n <= max_code; n++) {
+ curlen = nextlen;
+ nextlen = tree[(n + 1) * 2 + 1];
+ if (++count < max_count && curlen == nextlen) {
+ continue;
+ } else if (count < min_count) {
+ bl_tree[curlen * 2] += count;
+ } else if (curlen !== 0) {
+ if (curlen != prevlen)
+ bl_tree[curlen * 2]++;
+ bl_tree[REP_3_6 * 2]++;
+ } else if (count <= 10) {
+ bl_tree[REPZ_3_10 * 2]++;
+ } else {
+ bl_tree[REPZ_11_138 * 2]++;
+ }
+ count = 0;
+ prevlen = curlen;
+ if (nextlen === 0) {
+ max_count = 138;
+ min_count = 3;
+ } else if (curlen == nextlen) {
+ max_count = 6;
+ min_count = 3;
+ } else {
+ max_count = 7;
+ min_count = 4;
+ }
+ }
+ }
+
+ // Construct the Huffman tree for the bit lengths and return the index in
+ // bl_order of the last bit length code to send.
+ function build_bl_tree() {
+ var max_blindex; // index of last bit length code of non zero freq
+
+ // Determine the bit length frequencies for literal and distance trees
+ scan_tree(dyn_ltree, l_desc.max_code);
+ scan_tree(dyn_dtree, d_desc.max_code);
+
+ // Build the bit length tree:
+ bl_desc.build_tree(that);
+ // opt_len now includes the length of the tree representations, except
+ // the lengths of the bit lengths codes and the 5+5+4 bits for the
+ // counts.
+
+ // Determine the number of bit length codes to send. The pkzip format
+ // requires that at least 4 bit length codes be sent. (appnote.txt says
+ // 3 but the actual value used is 4.)
+ for (max_blindex = BL_CODES - 1; max_blindex >= 3; max_blindex--) {
+ if (bl_tree[Tree.bl_order[max_blindex] * 2 + 1] !== 0)
+ break;
+ }
+ // Update opt_len to include the bit length tree and counts
+ that.opt_len += 3 * (max_blindex + 1) + 5 + 5 + 4;
+
+ return max_blindex;
+ }
+
+ // Output a byte on the stream.
+ // IN assertion: there is enough room in pending_buf.
+ function put_byte(p) {
+ that.pending_buf[that.pending++] = p;
+ }
+
+ function put_short(w) {
+ put_byte(w & 0xff);
+ put_byte((w >>> 8) & 0xff);
+ }
+
+ function putShortMSB(b) {
+ put_byte((b >> 8) & 0xff);
+ put_byte((b & 0xff) & 0xff);
+ }
+
+ function send_bits(value, length) {
+ var val, len = length;
+ if (bi_valid > Buf_size - len) {
+ val = value;
+ // bi_buf |= (val << bi_valid);
+ bi_buf |= ((val << bi_valid) & 0xffff);
+ put_short(bi_buf);
+ bi_buf = val >>> (Buf_size - bi_valid);
+ bi_valid += len - Buf_size;
+ } else {
+ // bi_buf |= (value) << bi_valid;
+ bi_buf |= (((value) << bi_valid) & 0xffff);
+ bi_valid += len;
+ }
+ }
+
+ function send_code(c, tree) {
+ var c2 = c * 2;
+ send_bits(tree[c2] & 0xffff, tree[c2 + 1] & 0xffff);
+ }
+
+ // Send a literal or distance tree in compressed form, using the codes in
+ // bl_tree.
+ function send_tree(tree,// the tree to be sent
+ max_code // and its largest code of non zero frequency
+ ) {
+ var n; // iterates over all tree elements
+ var prevlen = -1; // last emitted length
+ var curlen; // length of current code
+ var nextlen = tree[0 * 2 + 1]; // length of next code
+ var count = 0; // repeat count of the current code
+ var max_count = 7; // max repeat count
+ var min_count = 4; // min repeat count
+
+ if (nextlen === 0) {
+ max_count = 138;
+ min_count = 3;
+ }
+
+ for (n = 0; n <= max_code; n++) {
+ curlen = nextlen;
+ nextlen = tree[(n + 1) * 2 + 1];
+ if (++count < max_count && curlen == nextlen) {
+ continue;
+ } else if (count < min_count) {
+ do {
+ send_code(curlen, bl_tree);
+ } while (--count !== 0);
+ } else if (curlen !== 0) {
+ if (curlen != prevlen) {
+ send_code(curlen, bl_tree);
+ count--;
+ }
+ send_code(REP_3_6, bl_tree);
+ send_bits(count - 3, 2);
+ } else if (count <= 10) {
+ send_code(REPZ_3_10, bl_tree);
+ send_bits(count - 3, 3);
+ } else {
+ send_code(REPZ_11_138, bl_tree);
+ send_bits(count - 11, 7);
+ }
+ count = 0;
+ prevlen = curlen;
+ if (nextlen === 0) {
+ max_count = 138;
+ min_count = 3;
+ } else if (curlen == nextlen) {
+ max_count = 6;
+ min_count = 3;
+ } else {
+ max_count = 7;
+ min_count = 4;
+ }
+ }
+ }
+
+ // Send the header for a block using dynamic Huffman trees: the counts, the
+ // lengths of the bit length codes, the literal tree and the distance tree.
+ // IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4.
+ function send_all_trees(lcodes, dcodes, blcodes) {
+ var rank; // index in bl_order
+
+ send_bits(lcodes - 257, 5); // not +255 as stated in appnote.txt
+ send_bits(dcodes - 1, 5);
+ send_bits(blcodes - 4, 4); // not -3 as stated in appnote.txt
+ for (rank = 0; rank < blcodes; rank++) {
+ send_bits(bl_tree[Tree.bl_order[rank] * 2 + 1], 3);
+ }
+ send_tree(dyn_ltree, lcodes - 1); // literal tree
+ send_tree(dyn_dtree, dcodes - 1); // distance tree
+ }
+
+ // Flush the bit buffer, keeping at most 7 bits in it.
+ function bi_flush() {
+ if (bi_valid == 16) {
+ put_short(bi_buf);
+ bi_buf = 0;
+ bi_valid = 0;
+ } else if (bi_valid >= 8) {
+ put_byte(bi_buf & 0xff);
+ bi_buf >>>= 8;
+ bi_valid -= 8;
+ }
+ }
+
+ // Send one empty static block to give enough lookahead for inflate.
+ // This takes 10 bits, of which 7 may remain in the bit buffer.
+ // The current inflate code requires 9 bits of lookahead. If the
+ // last two codes for the previous block (real code plus EOB) were coded
+ // on 5 bits or less, inflate may have only 5+3 bits of lookahead to decode
+ // the last real code. In this case we send two empty static blocks instead
+ // of one. (There are no problems if the previous block is stored or fixed.)
+ // To simplify the code, we assume the worst case of last real code encoded
+ // on one bit only.
+ function _tr_align() {
+ send_bits(STATIC_TREES << 1, 3);
+ send_code(END_BLOCK, StaticTree.static_ltree);
+
+ bi_flush();
+
+ // Of the 10 bits for the empty block, we have already sent
+ // (10 - bi_valid) bits. The lookahead for the last real code (before
+ // the EOB of the previous block) was thus at least one plus the length
+ // of the EOB plus what we have just sent of the empty static block.
+ if (1 + last_eob_len + 10 - bi_valid < 9) {
+ send_bits(STATIC_TREES << 1, 3);
+ send_code(END_BLOCK, StaticTree.static_ltree);
+ bi_flush();
+ }
+ last_eob_len = 7;
+ }
+
+ // Save the match info and tally the frequency counts. Return true if
+ // the current block must be flushed.
+ function _tr_tally(dist, // distance of matched string
+ lc // match length-MIN_MATCH or unmatched char (if dist==0)
+ ) {
+ var out_length, in_length, dcode;
+ that.pending_buf[d_buf + last_lit * 2] = (dist >>> 8) & 0xff;
+ that.pending_buf[d_buf + last_lit * 2 + 1] = dist & 0xff;
+
+ that.pending_buf[l_buf + last_lit] = lc & 0xff;
+ last_lit++;
+
+ if (dist === 0) {
+ // lc is the unmatched char
+ dyn_ltree[lc * 2]++;
+ } else {
+ matches++;
+ // Here, lc is the match length - MIN_MATCH
+ dist--; // dist = match distance - 1
+ dyn_ltree[(Tree._length_code[lc] + LITERALS + 1) * 2]++;
+ dyn_dtree[Tree.d_code(dist) * 2]++;
+ }
+
+ if ((last_lit & 0x1fff) === 0 && level > 2) {
+ // Compute an upper bound for the compressed length
+ out_length = last_lit * 8;
+ in_length = strstart - block_start;
+ for (dcode = 0; dcode < D_CODES; dcode++) {
+ out_length += dyn_dtree[dcode * 2] * (5 + Tree.extra_dbits[dcode]);
+ }
+ out_length >>>= 3;
+ if ((matches < Math.floor(last_lit / 2)) && out_length < Math.floor(in_length / 2))
+ return true;
+ }
+
+ return (last_lit == lit_bufsize - 1);
+ // We avoid equality with lit_bufsize because of wraparound at 64K
+ // on 16 bit machines and because stored blocks are restricted to
+ // 64K-1 bytes.
+ }
+
+ // Send the block data compressed using the given Huffman trees
+ function compress_block(ltree, dtree) {
+ var dist; // distance of matched string
+ var lc; // match length or unmatched char (if dist === 0)
+ var lx = 0; // running index in l_buf
+ var code; // the code to send
+ var extra; // number of extra bits to send
+
+ if (last_lit !== 0) {
+ do {
+ dist = ((that.pending_buf[d_buf + lx * 2] << 8) & 0xff00) | (that.pending_buf[d_buf + lx * 2 + 1] & 0xff);
+ lc = (that.pending_buf[l_buf + lx]) & 0xff;
+ lx++;
+
+ if (dist === 0) {
+ send_code(lc, ltree); // send a literal byte
+ } else {
+ // Here, lc is the match length - MIN_MATCH
+ code = Tree._length_code[lc];
+
+ send_code(code + LITERALS + 1, ltree); // send the length
+ // code
+ extra = Tree.extra_lbits[code];
+ if (extra !== 0) {
+ lc -= Tree.base_length[code];
+ send_bits(lc, extra); // send the extra length bits
+ }
+ dist--; // dist is now the match distance - 1
+ code = Tree.d_code(dist);
+
+ send_code(code, dtree); // send the distance code
+ extra = Tree.extra_dbits[code];
+ if (extra !== 0) {
+ dist -= Tree.base_dist[code];
+ send_bits(dist, extra); // send the extra distance bits
+ }
+ } // literal or match pair ?
+
+ // Check that the overlay between pending_buf and d_buf+l_buf is
+ // ok:
+ } while (lx < last_lit);
+ }
+
+ send_code(END_BLOCK, ltree);
+ last_eob_len = ltree[END_BLOCK * 2 + 1];
+ }
+
+ // Flush the bit buffer and align the output on a byte boundary
+ function bi_windup() {
+ if (bi_valid > 8) {
+ put_short(bi_buf);
+ } else if (bi_valid > 0) {
+ put_byte(bi_buf & 0xff);
+ }
+ bi_buf = 0;
+ bi_valid = 0;
+ }
+
+ // Copy a stored block, storing first the length and its
+ // one's complement if requested.
+ function copy_block(buf, // the input data
+ len, // its length
+ header // true if block header must be written
+ ) {
+ bi_windup(); // align on byte boundary
+ last_eob_len = 8; // enough lookahead for inflate
+
+ if (header) {
+ put_short(len);
+ put_short(~len);
+ }
+
+ that.pending_buf.set(window.subarray(buf, buf + len), that.pending);
+ that.pending += len;
+ }
+
+ // Send a stored block
+ function _tr_stored_block(buf, // input block
+ stored_len, // length of input block
+ eof // true if this is the last block for a file
+ ) {
+ send_bits((STORED_BLOCK << 1) + (eof ? 1 : 0), 3); // send block type
+ copy_block(buf, stored_len, true); // with header
+ }
+
+ // Determine the best encoding for the current block: dynamic trees, static
+ // trees or store, and output the encoded block to the zip file.
+ function _tr_flush_block(buf, // input block, or NULL if too old
+ stored_len, // length of input block
+ eof // true if this is the last block for a file
+ ) {
+ var opt_lenb, static_lenb;// opt_len and static_len in bytes
+ var max_blindex = 0; // index of last bit length code of non zero freq
+
+ // Build the Huffman trees unless a stored block is forced
+ if (level > 0) {
+ // Construct the literal and distance trees
+ l_desc.build_tree(that);
+
+ d_desc.build_tree(that);
+
+ // At this point, opt_len and static_len are the total bit lengths
+ // of
+ // the compressed block data, excluding the tree representations.
+
+ // Build the bit length tree for the above two trees, and get the
+ // index
+ // in bl_order of the last bit length code to send.
+ max_blindex = build_bl_tree();
+
+ // Determine the best encoding. Compute first the block length in
+ // bytes
+ opt_lenb = (that.opt_len + 3 + 7) >>> 3;
+ static_lenb = (that.static_len + 3 + 7) >>> 3;
+
+ if (static_lenb <= opt_lenb)
+ opt_lenb = static_lenb;
+ } else {
+ opt_lenb = static_lenb = stored_len + 5; // force a stored block
+ }
+
+ if ((stored_len + 4 <= opt_lenb) && buf != -1) {
+ // 4: two words for the lengths
+ // The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE.
+ // Otherwise we can't have processed more than WSIZE input bytes
+ // since
+ // the last block flush, because compression would have been
+ // successful. If LIT_BUFSIZE <= WSIZE, it is never too late to
+ // transform a block into a stored block.
+ _tr_stored_block(buf, stored_len, eof);
+ } else if (static_lenb == opt_lenb) {
+ send_bits((STATIC_TREES << 1) + (eof ? 1 : 0), 3);
+ compress_block(StaticTree.static_ltree, StaticTree.static_dtree);
+ } else {
+ send_bits((DYN_TREES << 1) + (eof ? 1 : 0), 3);
+ send_all_trees(l_desc.max_code + 1, d_desc.max_code + 1, max_blindex + 1);
+ compress_block(dyn_ltree, dyn_dtree);
+ }
+
+ // The above check is made mod 2^32, for files larger than 512 MB
+ // and uLong implemented on 32 bits.
+
+ init_block();
+
+ if (eof) {
+ bi_windup();
+ }
+ }
+
+ function flush_block_only(eof) {
+ _tr_flush_block(block_start >= 0 ? block_start : -1, strstart - block_start, eof);
+ block_start = strstart;
+ strm.flush_pending();
+ }
+
+ // Fill the window when the lookahead becomes insufficient.
+ // Updates strstart and lookahead.
+ //
+ // IN assertion: lookahead < MIN_LOOKAHEAD
+ // OUT assertions: strstart <= window_size-MIN_LOOKAHEAD
+ // At least one byte has been read, or avail_in === 0; reads are
+ // performed for at least two bytes (required for the zip translate_eol
+ // option -- not supported here).
+ function fill_window() {
+ var n, m;
+ var p;
+ var more; // Amount of free space at the end of the window.
+
+ do {
+ more = (window_size - lookahead - strstart);
+
+ // Deal with !@#$% 64K limit:
+ if (more === 0 && strstart === 0 && lookahead === 0) {
+ more = w_size;
+ } else if (more == -1) {
+ // Very unlikely, but possible on 16 bit machine if strstart ==
+ // 0
+ // and lookahead == 1 (input done one byte at time)
+ more--;
+
+ // If the window is almost full and there is insufficient
+ // lookahead,
+ // move the upper half to the lower one to make room in the
+ // upper half.
+ } else if (strstart >= w_size + w_size - MIN_LOOKAHEAD) {
+ window.set(window.subarray(w_size, w_size + w_size), 0);
+
+ match_start -= w_size;
+ strstart -= w_size; // we now have strstart >= MAX_DIST
+ block_start -= w_size;
+
+ // Slide the hash table (could be avoided with 32 bit values
+ // at the expense of memory usage). We slide even when level ==
+ // 0
+ // to keep the hash table consistent if we switch back to level
+ // > 0
+ // later. (Using level 0 permanently is not an optimal usage of
+ // zlib, so we don't care about this pathological case.)
+
+ n = hash_size;
+ p = n;
+ do {
+ m = (head[--p] & 0xffff);
+ head[p] = (m >= w_size ? m - w_size : 0);
+ } while (--n !== 0);
+
+ n = w_size;
+ p = n;
+ do {
+ m = (prev[--p] & 0xffff);
+ prev[p] = (m >= w_size ? m - w_size : 0);
+ // If n is not on any hash chain, prev[n] is garbage but
+ // its value will never be used.
+ } while (--n !== 0);
+ more += w_size;
+ }
+
+ if (strm.avail_in === 0)
+ return;
+
+ // If there was no sliding:
+ // strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 &&
+ // more == window_size - lookahead - strstart
+ // => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1)
+ // => more >= window_size - 2*WSIZE + 2
+ // In the BIG_MEM or MMAP case (not yet supported),
+ // window_size == input_size + MIN_LOOKAHEAD &&
+ // strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD.
+ // Otherwise, window_size == 2*WSIZE so more >= 2.
+ // If there was sliding, more >= WSIZE. So in all cases, more >= 2.
+
+ n = strm.read_buf(window, strstart + lookahead, more);
+ lookahead += n;
+
+ // Initialize the hash value now that we have some input:
+ if (lookahead >= MIN_MATCH) {
+ ins_h = window[strstart] & 0xff;
+ ins_h = (((ins_h) << hash_shift) ^ (window[strstart + 1] & 0xff)) & hash_mask;
+ }
+ // If the whole input has less than MIN_MATCH bytes, ins_h is
+ // garbage,
+ // but this is not important since only literal bytes will be
+ // emitted.
+ } while (lookahead < MIN_LOOKAHEAD && strm.avail_in !== 0);
+ }
+
+ // Copy without compression as much as possible from the input stream,
+ // return
+ // the current block state.
+ // This function does not insert new strings in the dictionary since
+ // uncompressible data is probably not useful. This function is used
+ // only for the level=0 compression option.
+ // NOTE: this function should be optimized to avoid extra copying from
+ // window to pending_buf.
+ function deflate_stored(flush) {
+ // Stored blocks are limited to 0xffff bytes, pending_buf is limited
+ // to pending_buf_size, and each stored block has a 5 byte header:
+
+ var max_block_size = 0xffff;
+ var max_start;
+
+ if (max_block_size > pending_buf_size - 5) {
+ max_block_size = pending_buf_size - 5;
+ }
+
+ // Copy as much as possible from input to output:
+ while (true) {
+ // Fill the window as much as possible:
+ if (lookahead <= 1) {
+ fill_window();
+ if (lookahead === 0 && flush == Z_NO_FLUSH)
+ return NeedMore;
+ if (lookahead === 0)
+ break; // flush the current block
+ }
+
+ strstart += lookahead;
+ lookahead = 0;
+
+ // Emit a stored block if pending_buf will be full:
+ max_start = block_start + max_block_size;
+ if (strstart === 0 || strstart >= max_start) {
+ // strstart === 0 is possible when wraparound on 16-bit machine
+ lookahead = (strstart - max_start);
+ strstart = max_start;
+
+ flush_block_only(false);
+ if (strm.avail_out === 0)
+ return NeedMore;
+
+ }
+
+ // Flush if we may have to slide, otherwise block_start may become
+ // negative and the data will be gone:
+ if (strstart - block_start >= w_size - MIN_LOOKAHEAD) {
+ flush_block_only(false);
+ if (strm.avail_out === 0)
+ return NeedMore;
+ }
+ }
+
+ flush_block_only(flush == Z_FINISH);
+ if (strm.avail_out === 0)
+ return (flush == Z_FINISH) ? FinishStarted : NeedMore;
+
+ return flush == Z_FINISH ? FinishDone : BlockDone;
+ }
+
+ function longest_match(cur_match) {
+ var chain_length = max_chain_length; // max hash chain length
+ var scan = strstart; // current string
+ var match; // matched string
+ var len; // length of current match
+ var best_len = prev_length; // best match length so far
+ var limit = strstart > (w_size - MIN_LOOKAHEAD) ? strstart - (w_size - MIN_LOOKAHEAD) : 0;
+ var _nice_match = nice_match;
+
+ // Stop when cur_match becomes <= limit. To simplify the code,
+ // we prevent matches with the string of window index 0.
+
+ var wmask = w_mask;
+
+ var strend = strstart + MAX_MATCH;
+ var scan_end1 = window[scan + best_len - 1];
+ var scan_end = window[scan + best_len];
+
+ // The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of
+ // 16.
+ // It is easy to get rid of this optimization if necessary.
+
+ // Do not waste too much time if we already have a good match:
+ if (prev_length >= good_match) {
+ chain_length >>= 2;
+ }
+
+ // Do not look for matches beyond the end of the input. This is
+ // necessary
+ // to make deflate deterministic.
+ if (_nice_match > lookahead)
+ _nice_match = lookahead;
+
+ do {
+ match = cur_match;
+
+ // Skip to next match if the match length cannot increase
+ // or if the match length is less than 2:
+ if (window[match + best_len] != scan_end || window[match + best_len - 1] != scan_end1 || window[match] != window[scan]
+ || window[++match] != window[scan + 1])
+ continue;
+
+ // The check at best_len-1 can be removed because it will be made
+ // again later. (This heuristic is not always a win.)
+ // It is not necessary to compare scan[2] and match[2] since they
+ // are always equal when the other bytes match, given that
+ // the hash keys are equal and that HASH_BITS >= 8.
+ scan += 2;
+ match++;
+
+ // We check for insufficient lookahead only every 8th comparison;
+ // the 256th check will be made at strstart+258.
+ do {
+ } while (window[++scan] == window[++match] && window[++scan] == window[++match] && window[++scan] == window[++match]
+ && window[++scan] == window[++match] && window[++scan] == window[++match] && window[++scan] == window[++match]
+ && window[++scan] == window[++match] && window[++scan] == window[++match] && scan < strend);
+
+ len = MAX_MATCH - (strend - scan);
+ scan = strend - MAX_MATCH;
+
+ if (len > best_len) {
+ match_start = cur_match;
+ best_len = len;
+ if (len >= _nice_match)
+ break;
+ scan_end1 = window[scan + best_len - 1];
+ scan_end = window[scan + best_len];
+ }
+
+ } while ((cur_match = (prev[cur_match & wmask] & 0xffff)) > limit && --chain_length !== 0);
+
+ if (best_len <= lookahead)
+ return best_len;
+ return lookahead;
+ }
+
+ // Compress as much as possible from the input stream, return the current
+ // block state.
+ // This function does not perform lazy evaluation of matches and inserts
+ // new strings in the dictionary only for unmatched strings or for short
+ // matches. It is used only for the fast compression options.
+ function deflate_fast(flush) {
+ // short hash_head = 0; // head of the hash chain
+ var hash_head = 0; // head of the hash chain
+ var bflush; // set if current block must be flushed
+
+ while (true) {
+ // Make sure that we always have enough lookahead, except
+ // at the end of the input file. We need MAX_MATCH bytes
+ // for the next match, plus MIN_MATCH bytes to insert the
+ // string following the next match.
+ if (lookahead < MIN_LOOKAHEAD) {
+ fill_window();
+ if (lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
+ return NeedMore;
+ }
+ if (lookahead === 0)
+ break; // flush the current block
+ }
+
+ // Insert the string window[strstart .. strstart+2] in the
+ // dictionary, and set hash_head to the head of the hash chain:
+ if (lookahead >= MIN_MATCH) {
+ ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
+
+ // prev[strstart&w_mask]=hash_head=head[ins_h];
+ hash_head = (head[ins_h] & 0xffff);
+ prev[strstart & w_mask] = head[ins_h];
+ head[ins_h] = strstart;
+ }
+
+ // Find the longest match, discarding those <= prev_length.
+ // At this point we have always match_length < MIN_MATCH
+
+ if (hash_head !== 0 && ((strstart - hash_head) & 0xffff) <= w_size - MIN_LOOKAHEAD) {
+ // To simplify the code, we prevent matches with the string
+ // of window index 0 (in particular we have to avoid a match
+ // of the string with itself at the start of the input file).
+ if (strategy != Z_HUFFMAN_ONLY) {
+ match_length = longest_match(hash_head);
+ }
+ // longest_match() sets match_start
+ }
+ if (match_length >= MIN_MATCH) {
+ // check_match(strstart, match_start, match_length);
+
+ bflush = _tr_tally(strstart - match_start, match_length - MIN_MATCH);
+
+ lookahead -= match_length;
+
+ // Insert new strings in the hash table only if the match length
+ // is not too large. This saves time but degrades compression.
+ if (match_length <= max_lazy_match && lookahead >= MIN_MATCH) {
+ match_length--; // string at strstart already in hash table
+ do {
+ strstart++;
+
+ ins_h = ((ins_h << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
+ // prev[strstart&w_mask]=hash_head=head[ins_h];
+ hash_head = (head[ins_h] & 0xffff);
+ prev[strstart & w_mask] = head[ins_h];
+ head[ins_h] = strstart;
+
+ // strstart never exceeds WSIZE-MAX_MATCH, so there are
+ // always MIN_MATCH bytes ahead.
+ } while (--match_length !== 0);
+ strstart++;
+ } else {
+ strstart += match_length;
+ match_length = 0;
+ ins_h = window[strstart] & 0xff;
+
+ ins_h = (((ins_h) << hash_shift) ^ (window[strstart + 1] & 0xff)) & hash_mask;
+ // If lookahead < MIN_MATCH, ins_h is garbage, but it does
+ // not
+ // matter since it will be recomputed at next deflate call.
+ }
+ } else {
+ // No match, output a literal byte
+
+ bflush = _tr_tally(0, window[strstart] & 0xff);
+ lookahead--;
+ strstart++;
+ }
+ if (bflush) {
+
+ flush_block_only(false);
+ if (strm.avail_out === 0)
+ return NeedMore;
+ }
+ }
+
+ flush_block_only(flush == Z_FINISH);
+ if (strm.avail_out === 0) {
+ if (flush == Z_FINISH)
+ return FinishStarted;
+ else
+ return NeedMore;
+ }
+ return flush == Z_FINISH ? FinishDone : BlockDone;
+ }
+
+ // Same as above, but achieves better compression. We use a lazy
+ // evaluation for matches: a match is finally adopted only if there is
+ // no better match at the next window position.
+ function deflate_slow(flush) {
+ // short hash_head = 0; // head of hash chain
+ var hash_head = 0; // head of hash chain
+ var bflush; // set if current block must be flushed
+ var max_insert;
+
+ // Process the input block.
+ while (true) {
+ // Make sure that we always have enough lookahead, except
+ // at the end of the input file. We need MAX_MATCH bytes
+ // for the next match, plus MIN_MATCH bytes to insert the
+ // string following the next match.
+
+ if (lookahead < MIN_LOOKAHEAD) {
+ fill_window();
+ if (lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
+ return NeedMore;
+ }
+ if (lookahead === 0)
+ break; // flush the current block
+ }
+
+ // Insert the string window[strstart .. strstart+2] in the
+ // dictionary, and set hash_head to the head of the hash chain:
+
+ if (lookahead >= MIN_MATCH) {
+ ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
+ // prev[strstart&w_mask]=hash_head=head[ins_h];
+ hash_head = (head[ins_h] & 0xffff);
+ prev[strstart & w_mask] = head[ins_h];
+ head[ins_h] = strstart;
+ }
+
+ // Find the longest match, discarding those <= prev_length.
+ prev_length = match_length;
+ prev_match = match_start;
+ match_length = MIN_MATCH - 1;
+
+ if (hash_head !== 0 && prev_length < max_lazy_match && ((strstart - hash_head) & 0xffff) <= w_size - MIN_LOOKAHEAD) {
+ // To simplify the code, we prevent matches with the string
+ // of window index 0 (in particular we have to avoid a match
+ // of the string with itself at the start of the input file).
+
+ if (strategy != Z_HUFFMAN_ONLY) {
+ match_length = longest_match(hash_head);
+ }
+ // longest_match() sets match_start
+
+ if (match_length <= 5 && (strategy == Z_FILTERED || (match_length == MIN_MATCH && strstart - match_start > 4096))) {
+
+ // If prev_match is also MIN_MATCH, match_start is garbage
+ // but we will ignore the current match anyway.
+ match_length = MIN_MATCH - 1;
+ }
+ }
+
+ // If there was a match at the previous step and the current
+ // match is not better, output the previous match:
+ if (prev_length >= MIN_MATCH && match_length <= prev_length) {
+ max_insert = strstart + lookahead - MIN_MATCH;
+ // Do not insert strings in hash table beyond this.
+
+ // check_match(strstart-1, prev_match, prev_length);
+
+ bflush = _tr_tally(strstart - 1 - prev_match, prev_length - MIN_MATCH);
+
+ // Insert in hash table all strings up to the end of the match.
+ // strstart-1 and strstart are already inserted. If there is not
+ // enough lookahead, the last two strings are not inserted in
+ // the hash table.
+ lookahead -= prev_length - 1;
+ prev_length -= 2;
+ do {
+ if (++strstart <= max_insert) {
+ ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
+ // prev[strstart&w_mask]=hash_head=head[ins_h];
+ hash_head = (head[ins_h] & 0xffff);
+ prev[strstart & w_mask] = head[ins_h];
+ head[ins_h] = strstart;
+ }
+ } while (--prev_length !== 0);
+ match_available = 0;
+ match_length = MIN_MATCH - 1;
+ strstart++;
+
+ if (bflush) {
+ flush_block_only(false);
+ if (strm.avail_out === 0)
+ return NeedMore;
+ }
+ } else if (match_available !== 0) {
+
+ // If there was no match at the previous position, output a
+ // single literal. If there was a match but the current match
+ // is longer, truncate the previous match to a single literal.
+
+ bflush = _tr_tally(0, window[strstart - 1] & 0xff);
+
+ if (bflush) {
+ flush_block_only(false);
+ }
+ strstart++;
+ lookahead--;
+ if (strm.avail_out === 0)
+ return NeedMore;
+ } else {
+ // There is no previous match to compare with, wait for
+ // the next step to decide.
+
+ match_available = 1;
+ strstart++;
+ lookahead--;
+ }
+ }
+
+ if (match_available !== 0) {
+ bflush = _tr_tally(0, window[strstart - 1] & 0xff);
+ match_available = 0;
+ }
+ flush_block_only(flush == Z_FINISH);
+
+ if (strm.avail_out === 0) {
+ if (flush == Z_FINISH)
+ return FinishStarted;
+ else
+ return NeedMore;
+ }
+
+ return flush == Z_FINISH ? FinishDone : BlockDone;
+ }
+
+ function deflateReset(strm) {
+ strm.total_in = strm.total_out = 0;
+ strm.msg = null; //
+
+ that.pending = 0;
+ that.pending_out = 0;
+
+ status = BUSY_STATE;
+
+ last_flush = Z_NO_FLUSH;
+
+ tr_init();
+ lm_init();
+ return Z_OK;
+ }
+
+ that.deflateInit = function(strm, _level, bits, _method, memLevel, _strategy) {
+ if (!_method)
+ _method = Z_DEFLATED;
+ if (!memLevel)
+ memLevel = DEF_MEM_LEVEL;
+ if (!_strategy)
+ _strategy = Z_DEFAULT_STRATEGY;
+
+ // byte[] my_version=ZLIB_VERSION;
+
+ //
+ // if (!version || version[0] != my_version[0]
+ // || stream_size != sizeof(z_stream)) {
+ // return Z_VERSION_ERROR;
+ // }
+
+ strm.msg = null;
+
+ if (_level == Z_DEFAULT_COMPRESSION)
+ _level = 6;
+
+ if (memLevel < 1 || memLevel > MAX_MEM_LEVEL || _method != Z_DEFLATED || bits < 9 || bits > 15 || _level < 0 || _level > 9 || _strategy < 0
+ || _strategy > Z_HUFFMAN_ONLY) {
+ return Z_STREAM_ERROR;
+ }
+
+ strm.dstate = that;
+
+ w_bits = bits;
+ w_size = 1 << w_bits;
+ w_mask = w_size - 1;
+
+ hash_bits = memLevel + 7;
+ hash_size = 1 << hash_bits;
+ hash_mask = hash_size - 1;
+ hash_shift = Math.floor((hash_bits + MIN_MATCH - 1) / MIN_MATCH);
+
+ window = new Uint8Array(w_size * 2);
+ prev = [];
+ head = [];
+
+ lit_bufsize = 1 << (memLevel + 6); // 16K elements by default
+
+ // We overlay pending_buf and d_buf+l_buf. This works since the average
+ // output size for (length,distance) codes is <= 24 bits.
+ that.pending_buf = new Uint8Array(lit_bufsize * 4);
+ pending_buf_size = lit_bufsize * 4;
+
+ d_buf = Math.floor(lit_bufsize / 2);
+ l_buf = (1 + 2) * lit_bufsize;
+
+ level = _level;
+
+ strategy = _strategy;
+ method = _method & 0xff;
+
+ return deflateReset(strm);
+ };
+
+ that.deflateEnd = function() {
+ if (status != INIT_STATE && status != BUSY_STATE && status != FINISH_STATE) {
+ return Z_STREAM_ERROR;
+ }
+ // Deallocate in reverse order of allocations:
+ that.pending_buf = null;
+ head = null;
+ prev = null;
+ window = null;
+ // free
+ that.dstate = null;
+ return status == BUSY_STATE ? Z_DATA_ERROR : Z_OK;
+ };
+
+ that.deflateParams = function(strm, _level, _strategy) {
+ var err = Z_OK;
+
+ if (_level == Z_DEFAULT_COMPRESSION) {
+ _level = 6;
+ }
+ if (_level < 0 || _level > 9 || _strategy < 0 || _strategy > Z_HUFFMAN_ONLY) {
+ return Z_STREAM_ERROR;
+ }
+
+ if (config_table[level].func != config_table[_level].func && strm.total_in !== 0) {
+ // Flush the last buffer:
+ err = strm.deflate(Z_PARTIAL_FLUSH);
+ }
+
+ if (level != _level) {
+ level = _level;
+ max_lazy_match = config_table[level].max_lazy;
+ good_match = config_table[level].good_length;
+ nice_match = config_table[level].nice_length;
+ max_chain_length = config_table[level].max_chain;
+ }
+ strategy = _strategy;
+ return err;
+ };
+
+ that.deflateSetDictionary = function(strm, dictionary, dictLength) {
+ var length = dictLength;
+ var n, index = 0;
+
+ if (!dictionary || status != INIT_STATE)
+ return Z_STREAM_ERROR;
+
+ if (length < MIN_MATCH)
+ return Z_OK;
+ if (length > w_size - MIN_LOOKAHEAD) {
+ length = w_size - MIN_LOOKAHEAD;
+ index = dictLength - length; // use the tail of the dictionary
+ }
+ window.set(dictionary.subarray(index, index + length), 0);
+
+ strstart = length;
+ block_start = length;
+
+ // Insert all strings in the hash table (except for the last two bytes).
+ // s->lookahead stays null, so s->ins_h will be recomputed at the next
+ // call of fill_window.
+
+ ins_h = window[0] & 0xff;
+ ins_h = (((ins_h) << hash_shift) ^ (window[1] & 0xff)) & hash_mask;
+
+ for (n = 0; n <= length - MIN_MATCH; n++) {
+ ins_h = (((ins_h) << hash_shift) ^ (window[(n) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
+ prev[n & w_mask] = head[ins_h];
+ head[ins_h] = n;
+ }
+ return Z_OK;
+ };
+
+ that.deflate = function(_strm, flush) {
+ var i, header, level_flags, old_flush, bstate;
+
+ if (flush > Z_FINISH || flush < 0) {
+ return Z_STREAM_ERROR;
+ }
+
+ if (!_strm.next_out || (!_strm.next_in && _strm.avail_in !== 0) || (status == FINISH_STATE && flush != Z_FINISH)) {
+ _strm.msg = z_errmsg[Z_NEED_DICT - (Z_STREAM_ERROR)];
+ return Z_STREAM_ERROR;
+ }
+ if (_strm.avail_out === 0) {
+ _strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
+ return Z_BUF_ERROR;
+ }
+
+ strm = _strm; // just in case
+ old_flush = last_flush;
+ last_flush = flush;
+
+ // Write the zlib header
+ if (status == INIT_STATE) {
+ header = (Z_DEFLATED + ((w_bits - 8) << 4)) << 8;
+ level_flags = ((level - 1) & 0xff) >> 1;
+
+ if (level_flags > 3)
+ level_flags = 3;
+ header |= (level_flags << 6);
+ if (strstart !== 0)
+ header |= PRESET_DICT;
+ header += 31 - (header % 31);
+
+ status = BUSY_STATE;
+ putShortMSB(header);
+ }
+
+ // Flush as much pending output as possible
+ if (that.pending !== 0) {
+ strm.flush_pending();
+ if (strm.avail_out === 0) {
+ // console.log(" avail_out==0");
+ // Since avail_out is 0, deflate will be called again with
+ // more output space, but possibly with both pending and
+ // avail_in equal to zero. There won't be anything to do,
+ // but this is not an error situation so make sure we
+ // return OK instead of BUF_ERROR at next call of deflate:
+ last_flush = -1;
+ return Z_OK;
+ }
+
+ // Make sure there is something to do and avoid duplicate
+ // consecutive
+ // flushes. For repeated and useless calls with Z_FINISH, we keep
+ // returning Z_STREAM_END instead of Z_BUFF_ERROR.
+ } else if (strm.avail_in === 0 && flush <= old_flush && flush != Z_FINISH) {
+ strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
+ return Z_BUF_ERROR;
+ }
+
+ // User must not provide more input after the first FINISH:
+ if (status == FINISH_STATE && strm.avail_in !== 0) {
+ _strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
+ return Z_BUF_ERROR;
+ }
+
+ // Start a new block or continue the current one.
+ if (strm.avail_in !== 0 || lookahead !== 0 || (flush != Z_NO_FLUSH && status != FINISH_STATE)) {
+ bstate = -1;
+ switch (config_table[level].func) {
+ case STORED:
+ bstate = deflate_stored(flush);
+ break;
+ case FAST:
+ bstate = deflate_fast(flush);
+ break;
+ case SLOW:
+ bstate = deflate_slow(flush);
+ break;
+ default:
+ }
+
+ if (bstate == FinishStarted || bstate == FinishDone) {
+ status = FINISH_STATE;
+ }
+ if (bstate == NeedMore || bstate == FinishStarted) {
+ if (strm.avail_out === 0) {
+ last_flush = -1; // avoid BUF_ERROR next call, see above
+ }
+ return Z_OK;
+ // If flush != Z_NO_FLUSH && avail_out === 0, the next call
+ // of deflate should use the same flush parameter to make sure
+ // that the flush is complete. So we don't have to output an
+ // empty block here, this will be done at next call. This also
+ // ensures that for a very small output buffer, we emit at most
+ // one empty block.
+ }
+
+ if (bstate == BlockDone) {
+ if (flush == Z_PARTIAL_FLUSH) {
+ _tr_align();
+ } else { // FULL_FLUSH or SYNC_FLUSH
+ _tr_stored_block(0, 0, false);
+ // For a full flush, this empty block will be recognized
+ // as a special marker by inflate_sync().
+ if (flush == Z_FULL_FLUSH) {
+ // state.head[s.hash_size-1]=0;
+ for (i = 0; i < hash_size/*-1*/; i++)
+ // forget history
+ head[i] = 0;
+ }
+ }
+ strm.flush_pending();
+ if (strm.avail_out === 0) {
+ last_flush = -1; // avoid BUF_ERROR at next call, see above
+ return Z_OK;
+ }
+ }
+ }
+
+ if (flush != Z_FINISH)
+ return Z_OK;
+ return Z_STREAM_END;
+ };
+ }
+
+ // ZStream
+
+ function ZStream() {
+ var that = this;
+ that.next_in_index = 0;
+ that.next_out_index = 0;
+ // that.next_in; // next input byte
+ that.avail_in = 0; // number of bytes available at next_in
+ that.total_in = 0; // total nb of input bytes read so far
+ // that.next_out; // next output byte should be put there
+ that.avail_out = 0; // remaining free space at next_out
+ that.total_out = 0; // total nb of bytes output so far
+ // that.msg;
+ // that.dstate;
+ }
+
+ ZStream.prototype = {
+ deflateInit : function(level, bits) {
+ var that = this;
+ that.dstate = new Deflate();
+ if (!bits)
+ bits = MAX_BITS;
+ return that.dstate.deflateInit(that, level, bits);
+ },
+
+ deflate : function(flush) {
+ var that = this;
+ if (!that.dstate) {
+ return Z_STREAM_ERROR;
+ }
+ return that.dstate.deflate(that, flush);
+ },
+
+ deflateEnd : function() {
+ var that = this;
+ if (!that.dstate)
+ return Z_STREAM_ERROR;
+ var ret = that.dstate.deflateEnd();
+ that.dstate = null;
+ return ret;
+ },
+
+ deflateParams : function(level, strategy) {
+ var that = this;
+ if (!that.dstate)
+ return Z_STREAM_ERROR;
+ return that.dstate.deflateParams(that, level, strategy);
+ },
+
+ deflateSetDictionary : function(dictionary, dictLength) {
+ var that = this;
+ if (!that.dstate)
+ return Z_STREAM_ERROR;
+ return that.dstate.deflateSetDictionary(that, dictionary, dictLength);
+ },
+
+ // Read a new buffer from the current input stream, update the
+ // total number of bytes read. All deflate() input goes through
+ // this function so some applications may wish to modify it to avoid
+ // allocating a large strm->next_in buffer and copying from it.
+ // (See also flush_pending()).
+ read_buf : function(buf, start, size) {
+ var that = this;
+ var len = that.avail_in;
+ if (len > size)
+ len = size;
+ if (len === 0)
+ return 0;
+ that.avail_in -= len;
+ buf.set(that.next_in.subarray(that.next_in_index, that.next_in_index + len), start);
+ that.next_in_index += len;
+ that.total_in += len;
+ return len;
+ },
+
+ // Flush as much pending output as possible. All deflate() output goes
+ // through this function so some applications may wish to modify it
+ // to avoid allocating a large strm->next_out buffer and copying into it.
+ // (See also read_buf()).
+ flush_pending : function() {
+ var that = this;
+ var len = that.dstate.pending;
+
+ if (len > that.avail_out)
+ len = that.avail_out;
+ if (len === 0)
+ return;
+
+ // if (that.dstate.pending_buf.length <= that.dstate.pending_out || that.next_out.length <= that.next_out_index
+ // || that.dstate.pending_buf.length < (that.dstate.pending_out + len) || that.next_out.length < (that.next_out_index +
+ // len)) {
+ // console.log(that.dstate.pending_buf.length + ", " + that.dstate.pending_out + ", " + that.next_out.length + ", " +
+ // that.next_out_index + ", " + len);
+ // console.log("avail_out=" + that.avail_out);
+ // }
+
+ that.next_out.set(that.dstate.pending_buf.subarray(that.dstate.pending_out, that.dstate.pending_out + len), that.next_out_index);
+
+ that.next_out_index += len;
+ that.dstate.pending_out += len;
+ that.total_out += len;
+ that.avail_out -= len;
+ that.dstate.pending -= len;
+ if (that.dstate.pending === 0) {
+ that.dstate.pending_out = 0;
+ }
+ }
+ };
+
+ // Deflater
+
+ function Deflater(options) {
+ var that = this;
+ var z = new ZStream();
+ var bufsize = 512;
+ var flush = Z_NO_FLUSH;
+ var buf = new Uint8Array(bufsize);
+ var level = options ? options.level : Z_DEFAULT_COMPRESSION;
+ if (typeof level == "undefined")
+ level = Z_DEFAULT_COMPRESSION;
+ z.deflateInit(level);
+ z.next_out = buf;
+
+ that.append = function(data, onprogress) {
+ var err, buffers = [], lastIndex = 0, bufferIndex = 0, bufferSize = 0, array;
+ if (!data.length)
+ return;
+ z.next_in_index = 0;
+ z.next_in = data;
+ z.avail_in = data.length;
+ do {
+ z.next_out_index = 0;
+ z.avail_out = bufsize;
+ err = z.deflate(flush);
+ if (err != Z_OK)
+ throw new Error("deflating: " + z.msg);
+ if (z.next_out_index)
+ if (z.next_out_index == bufsize)
+ buffers.push(new Uint8Array(buf));
+ else
+ buffers.push(new Uint8Array(buf.subarray(0, z.next_out_index)));
+ bufferSize += z.next_out_index;
+ if (onprogress && z.next_in_index > 0 && z.next_in_index != lastIndex) {
+ onprogress(z.next_in_index);
+ lastIndex = z.next_in_index;
+ }
+ } while (z.avail_in > 0 || z.avail_out === 0);
+ array = new Uint8Array(bufferSize);
+ buffers.forEach(function(chunk) {
+ array.set(chunk, bufferIndex);
+ bufferIndex += chunk.length;
+ });
+ return array;
+ };
+ that.flush = function() {
+ var err, buffers = [], bufferIndex = 0, bufferSize = 0, array;
+ do {
+ z.next_out_index = 0;
+ z.avail_out = bufsize;
+ err = z.deflate(Z_FINISH);
+ if (err != Z_STREAM_END && err != Z_OK)
+ throw new Error("deflating: " + z.msg);
+ if (bufsize - z.avail_out > 0)
+ buffers.push(new Uint8Array(buf.subarray(0, z.next_out_index)));
+ bufferSize += z.next_out_index;
+ } while (z.avail_in > 0 || z.avail_out === 0);
+ z.deflateEnd();
+ array = new Uint8Array(bufferSize);
+ buffers.forEach(function(chunk) {
+ array.set(chunk, bufferIndex);
+ bufferIndex += chunk.length;
+ });
+ return array;
+ };
+ }
+
+ // 'zip' may not be defined in z-worker and some tests
+ var env = global.zip || global;
+ env.Deflater = env._jzlib_Deflater = Deflater;
+})(this);
diff --git a/lib/inflate.js b/lib/inflate.js
new file mode 100644
index 0000000..7174050
--- /dev/null
+++ b/lib/inflate.js
@@ -0,0 +1,2155 @@
+/*
+ Copyright (c) 2013 Gildas Lormeau. All rights reserved.
+
+ Redistribution and use in source and binary forms, with or without
+ modification, are permitted provided that the following conditions are met:
+
+ 1. Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+ 2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the distribution.
+
+ 3. The names of the authors may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+ THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+ INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+ INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+ INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+ OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+ LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+/*
+ * This program is based on JZlib 1.0.2 ymnk, JCraft,Inc.
+ * JZlib is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+(function(global) {
+ "use strict";
+
+ // Global
+ var MAX_BITS = 15;
+
+ var Z_OK = 0;
+ var Z_STREAM_END = 1;
+ var Z_NEED_DICT = 2;
+ var Z_STREAM_ERROR = -2;
+ var Z_DATA_ERROR = -3;
+ var Z_MEM_ERROR = -4;
+ var Z_BUF_ERROR = -5;
+
+ var inflate_mask = [ 0x00000000, 0x00000001, 0x00000003, 0x00000007, 0x0000000f, 0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, 0x000001ff, 0x000003ff,
+ 0x000007ff, 0x00000fff, 0x00001fff, 0x00003fff, 0x00007fff, 0x0000ffff ];
+
+ var MANY = 1440;
+
+ // JZlib version : "1.0.2"
+ var Z_NO_FLUSH = 0;
+ var Z_FINISH = 4;
+
+ // InfTree
+ var fixed_bl = 9;
+ var fixed_bd = 5;
+
+ var fixed_tl = [ 96, 7, 256, 0, 8, 80, 0, 8, 16, 84, 8, 115, 82, 7, 31, 0, 8, 112, 0, 8, 48, 0, 9, 192, 80, 7, 10, 0, 8, 96, 0, 8, 32, 0, 9, 160, 0, 8, 0,
+ 0, 8, 128, 0, 8, 64, 0, 9, 224, 80, 7, 6, 0, 8, 88, 0, 8, 24, 0, 9, 144, 83, 7, 59, 0, 8, 120, 0, 8, 56, 0, 9, 208, 81, 7, 17, 0, 8, 104, 0, 8, 40,
+ 0, 9, 176, 0, 8, 8, 0, 8, 136, 0, 8, 72, 0, 9, 240, 80, 7, 4, 0, 8, 84, 0, 8, 20, 85, 8, 227, 83, 7, 43, 0, 8, 116, 0, 8, 52, 0, 9, 200, 81, 7, 13,
+ 0, 8, 100, 0, 8, 36, 0, 9, 168, 0, 8, 4, 0, 8, 132, 0, 8, 68, 0, 9, 232, 80, 7, 8, 0, 8, 92, 0, 8, 28, 0, 9, 152, 84, 7, 83, 0, 8, 124, 0, 8, 60,
+ 0, 9, 216, 82, 7, 23, 0, 8, 108, 0, 8, 44, 0, 9, 184, 0, 8, 12, 0, 8, 140, 0, 8, 76, 0, 9, 248, 80, 7, 3, 0, 8, 82, 0, 8, 18, 85, 8, 163, 83, 7,
+ 35, 0, 8, 114, 0, 8, 50, 0, 9, 196, 81, 7, 11, 0, 8, 98, 0, 8, 34, 0, 9, 164, 0, 8, 2, 0, 8, 130, 0, 8, 66, 0, 9, 228, 80, 7, 7, 0, 8, 90, 0, 8,
+ 26, 0, 9, 148, 84, 7, 67, 0, 8, 122, 0, 8, 58, 0, 9, 212, 82, 7, 19, 0, 8, 106, 0, 8, 42, 0, 9, 180, 0, 8, 10, 0, 8, 138, 0, 8, 74, 0, 9, 244, 80,
+ 7, 5, 0, 8, 86, 0, 8, 22, 192, 8, 0, 83, 7, 51, 0, 8, 118, 0, 8, 54, 0, 9, 204, 81, 7, 15, 0, 8, 102, 0, 8, 38, 0, 9, 172, 0, 8, 6, 0, 8, 134, 0,
+ 8, 70, 0, 9, 236, 80, 7, 9, 0, 8, 94, 0, 8, 30, 0, 9, 156, 84, 7, 99, 0, 8, 126, 0, 8, 62, 0, 9, 220, 82, 7, 27, 0, 8, 110, 0, 8, 46, 0, 9, 188, 0,
+ 8, 14, 0, 8, 142, 0, 8, 78, 0, 9, 252, 96, 7, 256, 0, 8, 81, 0, 8, 17, 85, 8, 131, 82, 7, 31, 0, 8, 113, 0, 8, 49, 0, 9, 194, 80, 7, 10, 0, 8, 97,
+ 0, 8, 33, 0, 9, 162, 0, 8, 1, 0, 8, 129, 0, 8, 65, 0, 9, 226, 80, 7, 6, 0, 8, 89, 0, 8, 25, 0, 9, 146, 83, 7, 59, 0, 8, 121, 0, 8, 57, 0, 9, 210,
+ 81, 7, 17, 0, 8, 105, 0, 8, 41, 0, 9, 178, 0, 8, 9, 0, 8, 137, 0, 8, 73, 0, 9, 242, 80, 7, 4, 0, 8, 85, 0, 8, 21, 80, 8, 258, 83, 7, 43, 0, 8, 117,
+ 0, 8, 53, 0, 9, 202, 81, 7, 13, 0, 8, 101, 0, 8, 37, 0, 9, 170, 0, 8, 5, 0, 8, 133, 0, 8, 69, 0, 9, 234, 80, 7, 8, 0, 8, 93, 0, 8, 29, 0, 9, 154,
+ 84, 7, 83, 0, 8, 125, 0, 8, 61, 0, 9, 218, 82, 7, 23, 0, 8, 109, 0, 8, 45, 0, 9, 186, 0, 8, 13, 0, 8, 141, 0, 8, 77, 0, 9, 250, 80, 7, 3, 0, 8, 83,
+ 0, 8, 19, 85, 8, 195, 83, 7, 35, 0, 8, 115, 0, 8, 51, 0, 9, 198, 81, 7, 11, 0, 8, 99, 0, 8, 35, 0, 9, 166, 0, 8, 3, 0, 8, 131, 0, 8, 67, 0, 9, 230,
+ 80, 7, 7, 0, 8, 91, 0, 8, 27, 0, 9, 150, 84, 7, 67, 0, 8, 123, 0, 8, 59, 0, 9, 214, 82, 7, 19, 0, 8, 107, 0, 8, 43, 0, 9, 182, 0, 8, 11, 0, 8, 139,
+ 0, 8, 75, 0, 9, 246, 80, 7, 5, 0, 8, 87, 0, 8, 23, 192, 8, 0, 83, 7, 51, 0, 8, 119, 0, 8, 55, 0, 9, 206, 81, 7, 15, 0, 8, 103, 0, 8, 39, 0, 9, 174,
+ 0, 8, 7, 0, 8, 135, 0, 8, 71, 0, 9, 238, 80, 7, 9, 0, 8, 95, 0, 8, 31, 0, 9, 158, 84, 7, 99, 0, 8, 127, 0, 8, 63, 0, 9, 222, 82, 7, 27, 0, 8, 111,
+ 0, 8, 47, 0, 9, 190, 0, 8, 15, 0, 8, 143, 0, 8, 79, 0, 9, 254, 96, 7, 256, 0, 8, 80, 0, 8, 16, 84, 8, 115, 82, 7, 31, 0, 8, 112, 0, 8, 48, 0, 9,
+ 193, 80, 7, 10, 0, 8, 96, 0, 8, 32, 0, 9, 161, 0, 8, 0, 0, 8, 128, 0, 8, 64, 0, 9, 225, 80, 7, 6, 0, 8, 88, 0, 8, 24, 0, 9, 145, 83, 7, 59, 0, 8,
+ 120, 0, 8, 56, 0, 9, 209, 81, 7, 17, 0, 8, 104, 0, 8, 40, 0, 9, 177, 0, 8, 8, 0, 8, 136, 0, 8, 72, 0, 9, 241, 80, 7, 4, 0, 8, 84, 0, 8, 20, 85, 8,
+ 227, 83, 7, 43, 0, 8, 116, 0, 8, 52, 0, 9, 201, 81, 7, 13, 0, 8, 100, 0, 8, 36, 0, 9, 169, 0, 8, 4, 0, 8, 132, 0, 8, 68, 0, 9, 233, 80, 7, 8, 0, 8,
+ 92, 0, 8, 28, 0, 9, 153, 84, 7, 83, 0, 8, 124, 0, 8, 60, 0, 9, 217, 82, 7, 23, 0, 8, 108, 0, 8, 44, 0, 9, 185, 0, 8, 12, 0, 8, 140, 0, 8, 76, 0, 9,
+ 249, 80, 7, 3, 0, 8, 82, 0, 8, 18, 85, 8, 163, 83, 7, 35, 0, 8, 114, 0, 8, 50, 0, 9, 197, 81, 7, 11, 0, 8, 98, 0, 8, 34, 0, 9, 165, 0, 8, 2, 0, 8,
+ 130, 0, 8, 66, 0, 9, 229, 80, 7, 7, 0, 8, 90, 0, 8, 26, 0, 9, 149, 84, 7, 67, 0, 8, 122, 0, 8, 58, 0, 9, 213, 82, 7, 19, 0, 8, 106, 0, 8, 42, 0, 9,
+ 181, 0, 8, 10, 0, 8, 138, 0, 8, 74, 0, 9, 245, 80, 7, 5, 0, 8, 86, 0, 8, 22, 192, 8, 0, 83, 7, 51, 0, 8, 118, 0, 8, 54, 0, 9, 205, 81, 7, 15, 0, 8,
+ 102, 0, 8, 38, 0, 9, 173, 0, 8, 6, 0, 8, 134, 0, 8, 70, 0, 9, 237, 80, 7, 9, 0, 8, 94, 0, 8, 30, 0, 9, 157, 84, 7, 99, 0, 8, 126, 0, 8, 62, 0, 9,
+ 221, 82, 7, 27, 0, 8, 110, 0, 8, 46, 0, 9, 189, 0, 8, 14, 0, 8, 142, 0, 8, 78, 0, 9, 253, 96, 7, 256, 0, 8, 81, 0, 8, 17, 85, 8, 131, 82, 7, 31, 0,
+ 8, 113, 0, 8, 49, 0, 9, 195, 80, 7, 10, 0, 8, 97, 0, 8, 33, 0, 9, 163, 0, 8, 1, 0, 8, 129, 0, 8, 65, 0, 9, 227, 80, 7, 6, 0, 8, 89, 0, 8, 25, 0, 9,
+ 147, 83, 7, 59, 0, 8, 121, 0, 8, 57, 0, 9, 211, 81, 7, 17, 0, 8, 105, 0, 8, 41, 0, 9, 179, 0, 8, 9, 0, 8, 137, 0, 8, 73, 0, 9, 243, 80, 7, 4, 0, 8,
+ 85, 0, 8, 21, 80, 8, 258, 83, 7, 43, 0, 8, 117, 0, 8, 53, 0, 9, 203, 81, 7, 13, 0, 8, 101, 0, 8, 37, 0, 9, 171, 0, 8, 5, 0, 8, 133, 0, 8, 69, 0, 9,
+ 235, 80, 7, 8, 0, 8, 93, 0, 8, 29, 0, 9, 155, 84, 7, 83, 0, 8, 125, 0, 8, 61, 0, 9, 219, 82, 7, 23, 0, 8, 109, 0, 8, 45, 0, 9, 187, 0, 8, 13, 0, 8,
+ 141, 0, 8, 77, 0, 9, 251, 80, 7, 3, 0, 8, 83, 0, 8, 19, 85, 8, 195, 83, 7, 35, 0, 8, 115, 0, 8, 51, 0, 9, 199, 81, 7, 11, 0, 8, 99, 0, 8, 35, 0, 9,
+ 167, 0, 8, 3, 0, 8, 131, 0, 8, 67, 0, 9, 231, 80, 7, 7, 0, 8, 91, 0, 8, 27, 0, 9, 151, 84, 7, 67, 0, 8, 123, 0, 8, 59, 0, 9, 215, 82, 7, 19, 0, 8,
+ 107, 0, 8, 43, 0, 9, 183, 0, 8, 11, 0, 8, 139, 0, 8, 75, 0, 9, 247, 80, 7, 5, 0, 8, 87, 0, 8, 23, 192, 8, 0, 83, 7, 51, 0, 8, 119, 0, 8, 55, 0, 9,
+ 207, 81, 7, 15, 0, 8, 103, 0, 8, 39, 0, 9, 175, 0, 8, 7, 0, 8, 135, 0, 8, 71, 0, 9, 239, 80, 7, 9, 0, 8, 95, 0, 8, 31, 0, 9, 159, 84, 7, 99, 0, 8,
+ 127, 0, 8, 63, 0, 9, 223, 82, 7, 27, 0, 8, 111, 0, 8, 47, 0, 9, 191, 0, 8, 15, 0, 8, 143, 0, 8, 79, 0, 9, 255 ];
+ var fixed_td = [ 80, 5, 1, 87, 5, 257, 83, 5, 17, 91, 5, 4097, 81, 5, 5, 89, 5, 1025, 85, 5, 65, 93, 5, 16385, 80, 5, 3, 88, 5, 513, 84, 5, 33, 92, 5,
+ 8193, 82, 5, 9, 90, 5, 2049, 86, 5, 129, 192, 5, 24577, 80, 5, 2, 87, 5, 385, 83, 5, 25, 91, 5, 6145, 81, 5, 7, 89, 5, 1537, 85, 5, 97, 93, 5,
+ 24577, 80, 5, 4, 88, 5, 769, 84, 5, 49, 92, 5, 12289, 82, 5, 13, 90, 5, 3073, 86, 5, 193, 192, 5, 24577 ];
+
+ // Tables for deflate from PKZIP's appnote.txt.
+ var cplens = [ // Copy lengths for literal codes 257..285
+ 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0 ];
+
+ // see note #13 above about 258
+ var cplext = [ // Extra bits for literal codes 257..285
+ 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 112, 112 // 112==invalid
+ ];
+
+ var cpdist = [ // Copy offsets for distance codes 0..29
+ 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193, 257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577 ];
+
+ var cpdext = [ // Extra bits for distance codes
+ 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 ];
+
+ // If BMAX needs to be larger than 16, then h and x[] should be uLong.
+ var BMAX = 15; // maximum bit length of any code
+
+ function InfTree() {
+ var that = this;
+
+ var hn; // hufts used in space
+ var v; // work area for huft_build
+ var c; // bit length count table
+ var r; // table entry for structure assignment
+ var u; // table stack
+ var x; // bit offsets, then code stack
+
+ function huft_build(b, // code lengths in bits (all assumed <=
+ // BMAX)
+ bindex, n, // number of codes (assumed <= 288)
+ s, // number of simple-valued codes (0..s-1)
+ d, // list of base values for non-simple codes
+ e, // list of extra bits for non-simple codes
+ t, // result: starting table
+ m, // maximum lookup bits, returns actual
+ hp,// space for trees
+ hn,// hufts used in space
+ v // working area: values in order of bit length
+ ) {
+ // Given a list of code lengths and a maximum table size, make a set of
+ // tables to decode that set of codes. Return Z_OK on success,
+ // Z_BUF_ERROR
+ // if the given code set is incomplete (the tables are still built in
+ // this
+ // case), Z_DATA_ERROR if the input is invalid (an over-subscribed set
+ // of
+ // lengths), or Z_MEM_ERROR if not enough memory.
+
+ var a; // counter for codes of length k
+ var f; // i repeats in table every f entries
+ var g; // maximum code length
+ var h; // table level
+ var i; // counter, current code
+ var j; // counter
+ var k; // number of bits in current code
+ var l; // bits per table (returned in m)
+ var mask; // (1 << w) - 1, to avoid cc -O bug on HP
+ var p; // pointer into c[], b[], or v[]
+ var q; // points to current table
+ var w; // bits before this table == (l * h)
+ var xp; // pointer into x
+ var y; // number of dummy codes added
+ var z; // number of entries in current table
+
+ // Generate counts for each bit length
+
+ p = 0;
+ i = n;
+ do {
+ c[b[bindex + p]]++;
+ p++;
+ i--; // assume all entries <= BMAX
+ } while (i !== 0);
+
+ if (c[0] == n) { // null input--all zero length codes
+ t[0] = -1;
+ m[0] = 0;
+ return Z_OK;
+ }
+
+ // Find minimum and maximum length, bound *m by those
+ l = m[0];
+ for (j = 1; j <= BMAX; j++)
+ if (c[j] !== 0)
+ break;
+ k = j; // minimum code length
+ if (l < j) {
+ l = j;
+ }
+ for (i = BMAX; i !== 0; i--) {
+ if (c[i] !== 0)
+ break;
+ }
+ g = i; // maximum code length
+ if (l > i) {
+ l = i;
+ }
+ m[0] = l;
+
+ // Adjust last length count to fill out codes, if needed
+ for (y = 1 << j; j < i; j++, y <<= 1) {
+ if ((y -= c[j]) < 0) {
+ return Z_DATA_ERROR;
+ }
+ }
+ if ((y -= c[i]) < 0) {
+ return Z_DATA_ERROR;
+ }
+ c[i] += y;
+
+ // Generate starting offsets into the value table for each length
+ x[1] = j = 0;
+ p = 1;
+ xp = 2;
+ while (--i !== 0) { // note that i == g from above
+ x[xp] = (j += c[p]);
+ xp++;
+ p++;
+ }
+
+ // Make a table of values in order of bit lengths
+ i = 0;
+ p = 0;
+ do {
+ if ((j = b[bindex + p]) !== 0) {
+ v[x[j]++] = i;
+ }
+ p++;
+ } while (++i < n);
+ n = x[g]; // set n to length of v
+
+ // Generate the Huffman codes and for each, make the table entries
+ x[0] = i = 0; // first Huffman code is zero
+ p = 0; // grab values in bit order
+ h = -1; // no tables yet--level -1
+ w = -l; // bits decoded == (l * h)
+ u[0] = 0; // just to keep compilers happy
+ q = 0; // ditto
+ z = 0; // ditto
+
+ // go through the bit lengths (k already is bits in shortest code)
+ for (; k <= g; k++) {
+ a = c[k];
+ while (a-- !== 0) {
+ // here i is the Huffman code of length k bits for value *p
+ // make tables up to required level
+ while (k > w + l) {
+ h++;
+ w += l; // previous table always l bits
+ // compute minimum size table less than or equal to l bits
+ z = g - w;
+ z = (z > l) ? l : z; // table size upper limit
+ if ((f = 1 << (j = k - w)) > a + 1) { // try a k-w bit table
+ // too few codes for
+ // k-w bit table
+ f -= a + 1; // deduct codes from patterns left
+ xp = k;
+ if (j < z) {
+ while (++j < z) { // try smaller tables up to z bits
+ if ((f <<= 1) <= c[++xp])
+ break; // enough codes to use up j bits
+ f -= c[xp]; // else deduct codes from patterns
+ }
+ }
+ }
+ z = 1 << j; // table entries for j-bit table
+
+ // allocate new table
+ if (hn[0] + z > MANY) { // (note: doesn't matter for fixed)
+ return Z_DATA_ERROR; // overflow of MANY
+ }
+ u[h] = q = /* hp+ */hn[0]; // DEBUG
+ hn[0] += z;
+
+ // connect to last table, if there is one
+ if (h !== 0) {
+ x[h] = i; // save pattern for backing up
+ r[0] = /* (byte) */j; // bits in this table
+ r[1] = /* (byte) */l; // bits to dump before this table
+ j = i >>> (w - l);
+ r[2] = /* (int) */(q - u[h - 1] - j); // offset to this table
+ hp.set(r, (u[h - 1] + j) * 3);
+ // to
+ // last
+ // table
+ } else {
+ t[0] = q; // first table is returned result
+ }
+ }
+
+ // set up table entry in r
+ r[1] = /* (byte) */(k - w);
+ if (p >= n) {
+ r[0] = 128 + 64; // out of values--invalid code
+ } else if (v[p] < s) {
+ r[0] = /* (byte) */(v[p] < 256 ? 0 : 32 + 64); // 256 is
+ // end-of-block
+ r[2] = v[p++]; // simple code is just the value
+ } else {
+ r[0] = /* (byte) */(e[v[p] - s] + 16 + 64); // non-simple--look
+ // up in lists
+ r[2] = d[v[p++] - s];
+ }
+
+ // fill code-like entries with r
+ f = 1 << (k - w);
+ for (j = i >>> w; j < z; j += f) {
+ hp.set(r, (q + j) * 3);
+ }
+
+ // backwards increment the k-bit code i
+ for (j = 1 << (k - 1); (i & j) !== 0; j >>>= 1) {
+ i ^= j;
+ }
+ i ^= j;
+
+ // backup over finished tables
+ mask = (1 << w) - 1; // needed on HP, cc -O bug
+ while ((i & mask) != x[h]) {
+ h--; // don't need to update q
+ w -= l;
+ mask = (1 << w) - 1;
+ }
+ }
+ }
+ // Return Z_BUF_ERROR if we were given an incomplete table
+ return y !== 0 && g != 1 ? Z_BUF_ERROR : Z_OK;
+ }
+
+ function initWorkArea(vsize) {
+ var i;
+ if (!hn) {
+ hn = []; // []; //new Array(1);
+ v = []; // new Array(vsize);
+ c = new Int32Array(BMAX + 1); // new Array(BMAX + 1);
+ r = []; // new Array(3);
+ u = new Int32Array(BMAX); // new Array(BMAX);
+ x = new Int32Array(BMAX + 1); // new Array(BMAX + 1);
+ }
+ if (v.length < vsize) {
+ v = []; // new Array(vsize);
+ }
+ for (i = 0; i < vsize; i++) {
+ v[i] = 0;
+ }
+ for (i = 0; i < BMAX + 1; i++) {
+ c[i] = 0;
+ }
+ for (i = 0; i < 3; i++) {
+ r[i] = 0;
+ }
+ // for(int i=0; i 257)) {
+ if (result == Z_DATA_ERROR) {
+ z.msg = "oversubscribed distance tree";
+ } else if (result == Z_BUF_ERROR) {
+ z.msg = "incomplete distance tree";
+ result = Z_DATA_ERROR;
+ } else if (result != Z_MEM_ERROR) {
+ z.msg = "empty distance tree with lengths";
+ result = Z_DATA_ERROR;
+ }
+ return result;
+ }
+
+ return Z_OK;
+ };
+
+ }
+
+ InfTree.inflate_trees_fixed = function(bl, // literal desired/actual bit depth
+ bd, // distance desired/actual bit depth
+ tl,// literal/length tree result
+ td// distance tree result
+ ) {
+ bl[0] = fixed_bl;
+ bd[0] = fixed_bd;
+ tl[0] = fixed_tl;
+ td[0] = fixed_td;
+ return Z_OK;
+ };
+
+ // InfCodes
+
+ // waiting for "i:"=input,
+ // "o:"=output,
+ // "x:"=nothing
+ var START = 0; // x: set up for LEN
+ var LEN = 1; // i: get length/literal/eob next
+ var LENEXT = 2; // i: getting length extra (have base)
+ var DIST = 3; // i: get distance next
+ var DISTEXT = 4;// i: getting distance extra
+ var COPY = 5; // o: copying bytes in window, waiting
+ // for space
+ var LIT = 6; // o: got literal, waiting for output
+ // space
+ var WASH = 7; // o: got eob, possibly still output
+ // waiting
+ var END = 8; // x: got eob and all data flushed
+ var BADCODE = 9;// x: got error
+
+ function InfCodes() {
+ var that = this;
+
+ var mode; // current inflate_codes mode
+
+ // mode dependent information
+ var len = 0;
+
+ var tree; // pointer into tree
+ var tree_index = 0;
+ var need = 0; // bits needed
+
+ var lit = 0;
+
+ // if EXT or COPY, where and how much
+ var get = 0; // bits to get for extra
+ var dist = 0; // distance back to copy from
+
+ var lbits = 0; // ltree bits decoded per branch
+ var dbits = 0; // dtree bits decoder per branch
+ var ltree; // literal/length/eob tree
+ var ltree_index = 0; // literal/length/eob tree
+ var dtree; // distance tree
+ var dtree_index = 0; // distance tree
+
+ // Called with number of bytes left to write in window at least 258
+ // (the maximum string length) and number of input bytes available
+ // at least ten. The ten bytes are six bytes for the longest length/
+ // distance pair plus four bytes for overloading the bit buffer.
+
+ function inflate_fast(bl, bd, tl, tl_index, td, td_index, s, z) {
+ var t; // temporary pointer
+ var tp; // temporary pointer
+ var tp_index; // temporary pointer
+ var e; // extra bits or operation
+ var b; // bit buffer
+ var k; // bits in bit buffer
+ var p; // input data pointer
+ var n; // bytes available there
+ var q; // output window write pointer
+ var m; // bytes to end of window or read pointer
+ var ml; // mask for literal/length tree
+ var md; // mask for distance tree
+ var c; // bytes to copy
+ var d; // distance back to copy from
+ var r; // copy source pointer
+
+ var tp_index_t_3; // (tp_index+t)*3
+
+ // load input, output, bit values
+ p = z.next_in_index;
+ n = z.avail_in;
+ b = s.bitb;
+ k = s.bitk;
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ // initialize masks
+ ml = inflate_mask[bl];
+ md = inflate_mask[bd];
+
+ // do until not enough input or output space for fast loop
+ do { // assume called with m >= 258 && n >= 10
+ // get literal/length code
+ while (k < (20)) { // max bits for literal/length code
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ t = b & ml;
+ tp = tl;
+ tp_index = tl_index;
+ tp_index_t_3 = (tp_index + t) * 3;
+ if ((e = tp[tp_index_t_3]) === 0) {
+ b >>= (tp[tp_index_t_3 + 1]);
+ k -= (tp[tp_index_t_3 + 1]);
+
+ s.window[q++] = /* (byte) */tp[tp_index_t_3 + 2];
+ m--;
+ continue;
+ }
+ do {
+
+ b >>= (tp[tp_index_t_3 + 1]);
+ k -= (tp[tp_index_t_3 + 1]);
+
+ if ((e & 16) !== 0) {
+ e &= 15;
+ c = tp[tp_index_t_3 + 2] + (/* (int) */b & inflate_mask[e]);
+
+ b >>= e;
+ k -= e;
+
+ // decode distance base of block to copy
+ while (k < (15)) { // max bits for distance code
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ t = b & md;
+ tp = td;
+ tp_index = td_index;
+ tp_index_t_3 = (tp_index + t) * 3;
+ e = tp[tp_index_t_3];
+
+ do {
+
+ b >>= (tp[tp_index_t_3 + 1]);
+ k -= (tp[tp_index_t_3 + 1]);
+
+ if ((e & 16) !== 0) {
+ // get extra bits to add to distance base
+ e &= 15;
+ while (k < (e)) { // get extra bits (up to 13)
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ d = tp[tp_index_t_3 + 2] + (b & inflate_mask[e]);
+
+ b >>= (e);
+ k -= (e);
+
+ // do the copy
+ m -= c;
+ if (q >= d) { // offset before dest
+ // just copy
+ r = q - d;
+ if (q - r > 0 && 2 > (q - r)) {
+ s.window[q++] = s.window[r++]; // minimum
+ // count is
+ // three,
+ s.window[q++] = s.window[r++]; // so unroll
+ // loop a
+ // little
+ c -= 2;
+ } else {
+ s.window.set(s.window.subarray(r, r + 2), q);
+ q += 2;
+ r += 2;
+ c -= 2;
+ }
+ } else { // else offset after destination
+ r = q - d;
+ do {
+ r += s.end; // force pointer in window
+ } while (r < 0); // covers invalid distances
+ e = s.end - r;
+ if (c > e) { // if source crosses,
+ c -= e; // wrapped copy
+ if (q - r > 0 && e > (q - r)) {
+ do {
+ s.window[q++] = s.window[r++];
+ } while (--e !== 0);
+ } else {
+ s.window.set(s.window.subarray(r, r + e), q);
+ q += e;
+ r += e;
+ e = 0;
+ }
+ r = 0; // copy rest from start of window
+ }
+
+ }
+
+ // copy all or what's left
+ if (q - r > 0 && c > (q - r)) {
+ do {
+ s.window[q++] = s.window[r++];
+ } while (--c !== 0);
+ } else {
+ s.window.set(s.window.subarray(r, r + c), q);
+ q += c;
+ r += c;
+ c = 0;
+ }
+ break;
+ } else if ((e & 64) === 0) {
+ t += tp[tp_index_t_3 + 2];
+ t += (b & inflate_mask[e]);
+ tp_index_t_3 = (tp_index + t) * 3;
+ e = tp[tp_index_t_3];
+ } else {
+ z.msg = "invalid distance code";
+
+ c = z.avail_in - n;
+ c = (k >> 3) < c ? k >> 3 : c;
+ n += c;
+ p -= c;
+ k -= c << 3;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+
+ return Z_DATA_ERROR;
+ }
+ } while (true);
+ break;
+ }
+
+ if ((e & 64) === 0) {
+ t += tp[tp_index_t_3 + 2];
+ t += (b & inflate_mask[e]);
+ tp_index_t_3 = (tp_index + t) * 3;
+ if ((e = tp[tp_index_t_3]) === 0) {
+
+ b >>= (tp[tp_index_t_3 + 1]);
+ k -= (tp[tp_index_t_3 + 1]);
+
+ s.window[q++] = /* (byte) */tp[tp_index_t_3 + 2];
+ m--;
+ break;
+ }
+ } else if ((e & 32) !== 0) {
+
+ c = z.avail_in - n;
+ c = (k >> 3) < c ? k >> 3 : c;
+ n += c;
+ p -= c;
+ k -= c << 3;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+
+ return Z_STREAM_END;
+ } else {
+ z.msg = "invalid literal/length code";
+
+ c = z.avail_in - n;
+ c = (k >> 3) < c ? k >> 3 : c;
+ n += c;
+ p -= c;
+ k -= c << 3;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+
+ return Z_DATA_ERROR;
+ }
+ } while (true);
+ } while (m >= 258 && n >= 10);
+
+ // not enough input or output--restore pointers and return
+ c = z.avail_in - n;
+ c = (k >> 3) < c ? k >> 3 : c;
+ n += c;
+ p -= c;
+ k -= c << 3;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+
+ return Z_OK;
+ }
+
+ that.init = function(bl, bd, tl, tl_index, td, td_index) {
+ mode = START;
+ lbits = /* (byte) */bl;
+ dbits = /* (byte) */bd;
+ ltree = tl;
+ ltree_index = tl_index;
+ dtree = td;
+ dtree_index = td_index;
+ tree = null;
+ };
+
+ that.proc = function(s, z, r) {
+ var j; // temporary storage
+ var tindex; // temporary pointer
+ var e; // extra bits or operation
+ var b = 0; // bit buffer
+ var k = 0; // bits in bit buffer
+ var p = 0; // input data pointer
+ var n; // bytes available there
+ var q; // output window write pointer
+ var m; // bytes to end of window or read pointer
+ var f; // pointer to copy strings from
+
+ // copy input/output information to locals (UPDATE macro restores)
+ p = z.next_in_index;
+ n = z.avail_in;
+ b = s.bitb;
+ k = s.bitk;
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ // process input and output based on current state
+ while (true) {
+ switch (mode) {
+ // waiting for "i:"=input, "o:"=output, "x:"=nothing
+ case START: // x: set up for LEN
+ if (m >= 258 && n >= 10) {
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ r = inflate_fast(lbits, dbits, ltree, ltree_index, dtree, dtree_index, s, z);
+
+ p = z.next_in_index;
+ n = z.avail_in;
+ b = s.bitb;
+ k = s.bitk;
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ if (r != Z_OK) {
+ mode = r == Z_STREAM_END ? WASH : BADCODE;
+ break;
+ }
+ }
+ need = lbits;
+ tree = ltree;
+ tree_index = ltree_index;
+
+ mode = LEN;
+ /* falls through */
+ case LEN: // i: get length/literal/eob next
+ j = need;
+
+ while (k < (j)) {
+ if (n !== 0)
+ r = Z_OK;
+ else {
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ tindex = (tree_index + (b & inflate_mask[j])) * 3;
+
+ b >>>= (tree[tindex + 1]);
+ k -= (tree[tindex + 1]);
+
+ e = tree[tindex];
+
+ if (e === 0) { // literal
+ lit = tree[tindex + 2];
+ mode = LIT;
+ break;
+ }
+ if ((e & 16) !== 0) { // length
+ get = e & 15;
+ len = tree[tindex + 2];
+ mode = LENEXT;
+ break;
+ }
+ if ((e & 64) === 0) { // next table
+ need = e;
+ tree_index = tindex / 3 + tree[tindex + 2];
+ break;
+ }
+ if ((e & 32) !== 0) { // end of block
+ mode = WASH;
+ break;
+ }
+ mode = BADCODE; // invalid code
+ z.msg = "invalid literal/length code";
+ r = Z_DATA_ERROR;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+
+ case LENEXT: // i: getting length extra (have base)
+ j = get;
+
+ while (k < (j)) {
+ if (n !== 0)
+ r = Z_OK;
+ else {
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ len += (b & inflate_mask[j]);
+
+ b >>= j;
+ k -= j;
+
+ need = dbits;
+ tree = dtree;
+ tree_index = dtree_index;
+ mode = DIST;
+ /* falls through */
+ case DIST: // i: get distance next
+ j = need;
+
+ while (k < (j)) {
+ if (n !== 0)
+ r = Z_OK;
+ else {
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ tindex = (tree_index + (b & inflate_mask[j])) * 3;
+
+ b >>= tree[tindex + 1];
+ k -= tree[tindex + 1];
+
+ e = (tree[tindex]);
+ if ((e & 16) !== 0) { // distance
+ get = e & 15;
+ dist = tree[tindex + 2];
+ mode = DISTEXT;
+ break;
+ }
+ if ((e & 64) === 0) { // next table
+ need = e;
+ tree_index = tindex / 3 + tree[tindex + 2];
+ break;
+ }
+ mode = BADCODE; // invalid code
+ z.msg = "invalid distance code";
+ r = Z_DATA_ERROR;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+
+ case DISTEXT: // i: getting distance extra
+ j = get;
+
+ while (k < (j)) {
+ if (n !== 0)
+ r = Z_OK;
+ else {
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ dist += (b & inflate_mask[j]);
+
+ b >>= j;
+ k -= j;
+
+ mode = COPY;
+ /* falls through */
+ case COPY: // o: copying bytes in window, waiting for space
+ f = q - dist;
+ while (f < 0) { // modulo window size-"while" instead
+ f += s.end; // of "if" handles invalid distances
+ }
+ while (len !== 0) {
+
+ if (m === 0) {
+ if (q == s.end && s.read !== 0) {
+ q = 0;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+ }
+ if (m === 0) {
+ s.write = q;
+ r = s.inflate_flush(z, r);
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ if (q == s.end && s.read !== 0) {
+ q = 0;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+ }
+
+ if (m === 0) {
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ }
+ }
+
+ s.window[q++] = s.window[f++];
+ m--;
+
+ if (f == s.end)
+ f = 0;
+ len--;
+ }
+ mode = START;
+ break;
+ case LIT: // o: got literal, waiting for output space
+ if (m === 0) {
+ if (q == s.end && s.read !== 0) {
+ q = 0;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+ }
+ if (m === 0) {
+ s.write = q;
+ r = s.inflate_flush(z, r);
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ if (q == s.end && s.read !== 0) {
+ q = 0;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+ }
+ if (m === 0) {
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ }
+ }
+ r = Z_OK;
+
+ s.window[q++] = /* (byte) */lit;
+ m--;
+
+ mode = START;
+ break;
+ case WASH: // o: got eob, possibly more output
+ if (k > 7) { // return unused byte, if any
+ k -= 8;
+ n++;
+ p--; // can always return one
+ }
+
+ s.write = q;
+ r = s.inflate_flush(z, r);
+ q = s.write;
+ m = q < s.read ? s.read - q - 1 : s.end - q;
+
+ if (s.read != s.write) {
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ mode = END;
+ /* falls through */
+ case END:
+ r = Z_STREAM_END;
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+
+ case BADCODE: // x: got error
+
+ r = Z_DATA_ERROR;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+
+ default:
+ r = Z_STREAM_ERROR;
+
+ s.bitb = b;
+ s.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ s.write = q;
+ return s.inflate_flush(z, r);
+ }
+ }
+ };
+
+ that.free = function() {
+ // ZFREE(z, c);
+ };
+
+ }
+
+ // InfBlocks
+
+ // Table for deflate from PKZIP's appnote.txt.
+ var border = [ // Order of the bit length code lengths
+ 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15 ];
+
+ var TYPE = 0; // get type bits (3, including end bit)
+ var LENS = 1; // get lengths for stored
+ var STORED = 2;// processing stored block
+ var TABLE = 3; // get table lengths
+ var BTREE = 4; // get bit lengths tree for a dynamic
+ // block
+ var DTREE = 5; // get length, distance trees for a
+ // dynamic block
+ var CODES = 6; // processing fixed or dynamic block
+ var DRY = 7; // output remaining window bytes
+ var DONELOCKS = 8; // finished last block, done
+ var BADBLOCKS = 9; // ot a data error--stuck here
+
+ function InfBlocks(z, w) {
+ var that = this;
+
+ var mode = TYPE; // current inflate_block mode
+
+ var left = 0; // if STORED, bytes left to copy
+
+ var table = 0; // table lengths (14 bits)
+ var index = 0; // index into blens (or border)
+ var blens; // bit lengths of codes
+ var bb = [ 0 ]; // bit length tree depth
+ var tb = [ 0 ]; // bit length decoding tree
+
+ var codes = new InfCodes(); // if CODES, current state
+
+ var last = 0; // true if this block is the last block
+
+ var hufts = new Int32Array(MANY * 3); // single malloc for tree space
+ var check = 0; // check on output
+ var inftree = new InfTree();
+
+ that.bitk = 0; // bits in bit buffer
+ that.bitb = 0; // bit buffer
+ that.window = new Uint8Array(w); // sliding window
+ that.end = w; // one byte after sliding window
+ that.read = 0; // window read pointer
+ that.write = 0; // window write pointer
+
+ that.reset = function(z, c) {
+ if (c)
+ c[0] = check;
+ // if (mode == BTREE || mode == DTREE) {
+ // }
+ if (mode == CODES) {
+ codes.free(z);
+ }
+ mode = TYPE;
+ that.bitk = 0;
+ that.bitb = 0;
+ that.read = that.write = 0;
+ };
+
+ that.reset(z, null);
+
+ // copy as much as possible from the sliding window to the output area
+ that.inflate_flush = function(z, r) {
+ var n;
+ var p;
+ var q;
+
+ // local copies of source and destination pointers
+ p = z.next_out_index;
+ q = that.read;
+
+ // compute number of bytes to copy as far as end of window
+ n = /* (int) */((q <= that.write ? that.write : that.end) - q);
+ if (n > z.avail_out)
+ n = z.avail_out;
+ if (n !== 0 && r == Z_BUF_ERROR)
+ r = Z_OK;
+
+ // update counters
+ z.avail_out -= n;
+ z.total_out += n;
+
+ // copy as far as end of window
+ z.next_out.set(that.window.subarray(q, q + n), p);
+ p += n;
+ q += n;
+
+ // see if more to copy at beginning of window
+ if (q == that.end) {
+ // wrap pointers
+ q = 0;
+ if (that.write == that.end)
+ that.write = 0;
+
+ // compute bytes to copy
+ n = that.write - q;
+ if (n > z.avail_out)
+ n = z.avail_out;
+ if (n !== 0 && r == Z_BUF_ERROR)
+ r = Z_OK;
+
+ // update counters
+ z.avail_out -= n;
+ z.total_out += n;
+
+ // copy
+ z.next_out.set(that.window.subarray(q, q + n), p);
+ p += n;
+ q += n;
+ }
+
+ // update pointers
+ z.next_out_index = p;
+ that.read = q;
+
+ // done
+ return r;
+ };
+
+ that.proc = function(z, r) {
+ var t; // temporary storage
+ var b; // bit buffer
+ var k; // bits in bit buffer
+ var p; // input data pointer
+ var n; // bytes available there
+ var q; // output window write pointer
+ var m; // bytes to end of window or read pointer
+
+ var i;
+
+ // copy input/output information to locals (UPDATE macro restores)
+ // {
+ p = z.next_in_index;
+ n = z.avail_in;
+ b = that.bitb;
+ k = that.bitk;
+ // }
+ // {
+ q = that.write;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+ // }
+
+ // process input based on current state
+ // DEBUG dtree
+ while (true) {
+ switch (mode) {
+ case TYPE:
+
+ while (k < (3)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+ t = /* (int) */(b & 7);
+ last = t & 1;
+
+ switch (t >>> 1) {
+ case 0: // stored
+ // {
+ b >>>= (3);
+ k -= (3);
+ // }
+ t = k & 7; // go to byte boundary
+
+ // {
+ b >>>= (t);
+ k -= (t);
+ // }
+ mode = LENS; // get length of stored block
+ break;
+ case 1: // fixed
+ // {
+ var bl = []; // new Array(1);
+ var bd = []; // new Array(1);
+ var tl = [ [] ]; // new Array(1);
+ var td = [ [] ]; // new Array(1);
+
+ InfTree.inflate_trees_fixed(bl, bd, tl, td);
+ codes.init(bl[0], bd[0], tl[0], 0, td[0], 0);
+ // }
+
+ // {
+ b >>>= (3);
+ k -= (3);
+ // }
+
+ mode = CODES;
+ break;
+ case 2: // dynamic
+
+ // {
+ b >>>= (3);
+ k -= (3);
+ // }
+
+ mode = TABLE;
+ break;
+ case 3: // illegal
+
+ // {
+ b >>>= (3);
+ k -= (3);
+ // }
+ mode = BADBLOCKS;
+ z.msg = "invalid block type";
+ r = Z_DATA_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ break;
+ case LENS:
+
+ while (k < (32)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ if ((((~b) >>> 16) & 0xffff) != (b & 0xffff)) {
+ mode = BADBLOCKS;
+ z.msg = "invalid stored block lengths";
+ r = Z_DATA_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ left = (b & 0xffff);
+ b = k = 0; // dump bits
+ mode = left !== 0 ? STORED : (last !== 0 ? DRY : TYPE);
+ break;
+ case STORED:
+ if (n === 0) {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+
+ if (m === 0) {
+ if (q == that.end && that.read !== 0) {
+ q = 0;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+ }
+ if (m === 0) {
+ that.write = q;
+ r = that.inflate_flush(z, r);
+ q = that.write;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+ if (q == that.end && that.read !== 0) {
+ q = 0;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+ }
+ if (m === 0) {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ }
+ }
+ r = Z_OK;
+
+ t = left;
+ if (t > n)
+ t = n;
+ if (t > m)
+ t = m;
+ that.window.set(z.read_buf(p, t), q);
+ p += t;
+ n -= t;
+ q += t;
+ m -= t;
+ if ((left -= t) !== 0)
+ break;
+ mode = last !== 0 ? DRY : TYPE;
+ break;
+ case TABLE:
+
+ while (k < (14)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ table = t = (b & 0x3fff);
+ if ((t & 0x1f) > 29 || ((t >> 5) & 0x1f) > 29) {
+ mode = BADBLOCKS;
+ z.msg = "too many length or distance symbols";
+ r = Z_DATA_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ t = 258 + (t & 0x1f) + ((t >> 5) & 0x1f);
+ if (!blens || blens.length < t) {
+ blens = []; // new Array(t);
+ } else {
+ for (i = 0; i < t; i++) {
+ blens[i] = 0;
+ }
+ }
+
+ // {
+ b >>>= (14);
+ k -= (14);
+ // }
+
+ index = 0;
+ mode = BTREE;
+ /* falls through */
+ case BTREE:
+ while (index < 4 + (table >>> 10)) {
+ while (k < (3)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ blens[border[index++]] = b & 7;
+
+ // {
+ b >>>= (3);
+ k -= (3);
+ // }
+ }
+
+ while (index < 19) {
+ blens[border[index++]] = 0;
+ }
+
+ bb[0] = 7;
+ t = inftree.inflate_trees_bits(blens, bb, tb, hufts, z);
+ if (t != Z_OK) {
+ r = t;
+ if (r == Z_DATA_ERROR) {
+ blens = null;
+ mode = BADBLOCKS;
+ }
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+
+ index = 0;
+ mode = DTREE;
+ /* falls through */
+ case DTREE:
+ while (true) {
+ t = table;
+ if (index >= 258 + (t & 0x1f) + ((t >> 5) & 0x1f)) {
+ break;
+ }
+
+ var j, c;
+
+ t = bb[0];
+
+ while (k < (t)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ // if (tb[0] == -1) {
+ // System.err.println("null...");
+ // }
+
+ t = hufts[(tb[0] + (b & inflate_mask[t])) * 3 + 1];
+ c = hufts[(tb[0] + (b & inflate_mask[t])) * 3 + 2];
+
+ if (c < 16) {
+ b >>>= (t);
+ k -= (t);
+ blens[index++] = c;
+ } else { // c == 16..18
+ i = c == 18 ? 7 : c - 14;
+ j = c == 18 ? 11 : 3;
+
+ while (k < (t + i)) {
+ if (n !== 0) {
+ r = Z_OK;
+ } else {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ n--;
+ b |= (z.read_byte(p++) & 0xff) << k;
+ k += 8;
+ }
+
+ b >>>= (t);
+ k -= (t);
+
+ j += (b & inflate_mask[i]);
+
+ b >>>= (i);
+ k -= (i);
+
+ i = index;
+ t = table;
+ if (i + j > 258 + (t & 0x1f) + ((t >> 5) & 0x1f) || (c == 16 && i < 1)) {
+ blens = null;
+ mode = BADBLOCKS;
+ z.msg = "invalid bit length repeat";
+ r = Z_DATA_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+
+ c = c == 16 ? blens[i - 1] : 0;
+ do {
+ blens[i++] = c;
+ } while (--j !== 0);
+ index = i;
+ }
+ }
+
+ tb[0] = -1;
+ // {
+ var bl_ = []; // new Array(1);
+ var bd_ = []; // new Array(1);
+ var tl_ = []; // new Array(1);
+ var td_ = []; // new Array(1);
+ bl_[0] = 9; // must be <= 9 for lookahead assumptions
+ bd_[0] = 6; // must be <= 9 for lookahead assumptions
+
+ t = table;
+ t = inftree.inflate_trees_dynamic(257 + (t & 0x1f), 1 + ((t >> 5) & 0x1f), blens, bl_, bd_, tl_, td_, hufts, z);
+
+ if (t != Z_OK) {
+ if (t == Z_DATA_ERROR) {
+ blens = null;
+ mode = BADBLOCKS;
+ }
+ r = t;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ codes.init(bl_[0], bd_[0], hufts, tl_[0], hufts, td_[0]);
+ // }
+ mode = CODES;
+ /* falls through */
+ case CODES:
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+
+ if ((r = codes.proc(that, z, r)) != Z_STREAM_END) {
+ return that.inflate_flush(z, r);
+ }
+ r = Z_OK;
+ codes.free(z);
+
+ p = z.next_in_index;
+ n = z.avail_in;
+ b = that.bitb;
+ k = that.bitk;
+ q = that.write;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+
+ if (last === 0) {
+ mode = TYPE;
+ break;
+ }
+ mode = DRY;
+ /* falls through */
+ case DRY:
+ that.write = q;
+ r = that.inflate_flush(z, r);
+ q = that.write;
+ m = /* (int) */(q < that.read ? that.read - q - 1 : that.end - q);
+ if (that.read != that.write) {
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ mode = DONELOCKS;
+ /* falls through */
+ case DONELOCKS:
+ r = Z_STREAM_END;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ case BADBLOCKS:
+ r = Z_DATA_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+
+ default:
+ r = Z_STREAM_ERROR;
+
+ that.bitb = b;
+ that.bitk = k;
+ z.avail_in = n;
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ that.write = q;
+ return that.inflate_flush(z, r);
+ }
+ }
+ };
+
+ that.free = function(z) {
+ that.reset(z, null);
+ that.window = null;
+ hufts = null;
+ // ZFREE(z, s);
+ };
+
+ that.set_dictionary = function(d, start, n) {
+ that.window.set(d.subarray(start, start + n), 0);
+ that.read = that.write = n;
+ };
+
+ // Returns true if inflate is currently at the end of a block generated
+ // by Z_SYNC_FLUSH or Z_FULL_FLUSH.
+ that.sync_point = function() {
+ return mode == LENS ? 1 : 0;
+ };
+
+ }
+
+ // Inflate
+
+ // preset dictionary flag in zlib header
+ var PRESET_DICT = 0x20;
+
+ var Z_DEFLATED = 8;
+
+ var METHOD = 0; // waiting for method byte
+ var FLAG = 1; // waiting for flag byte
+ var DICT4 = 2; // four dictionary check bytes to go
+ var DICT3 = 3; // three dictionary check bytes to go
+ var DICT2 = 4; // two dictionary check bytes to go
+ var DICT1 = 5; // one dictionary check byte to go
+ var DICT0 = 6; // waiting for inflateSetDictionary
+ var BLOCKS = 7; // decompressing blocks
+ var DONE = 12; // finished check, done
+ var BAD = 13; // got an error--stay here
+
+ var mark = [ 0, 0, 0xff, 0xff ];
+
+ function Inflate() {
+ var that = this;
+
+ that.mode = 0; // current inflate mode
+
+ // mode dependent information
+ that.method = 0; // if FLAGS, method byte
+
+ // if CHECK, check values to compare
+ that.was = [ 0 ]; // new Array(1); // computed check value
+ that.need = 0; // stream check value
+
+ // if BAD, inflateSync's marker bytes count
+ that.marker = 0;
+
+ // mode independent information
+ that.wbits = 0; // log2(window size) (8..15, defaults to 15)
+
+ // this.blocks; // current inflate_blocks state
+
+ function inflateReset(z) {
+ if (!z || !z.istate)
+ return Z_STREAM_ERROR;
+
+ z.total_in = z.total_out = 0;
+ z.msg = null;
+ z.istate.mode = BLOCKS;
+ z.istate.blocks.reset(z, null);
+ return Z_OK;
+ }
+
+ that.inflateEnd = function(z) {
+ if (that.blocks)
+ that.blocks.free(z);
+ that.blocks = null;
+ // ZFREE(z, z->state);
+ return Z_OK;
+ };
+
+ that.inflateInit = function(z, w) {
+ z.msg = null;
+ that.blocks = null;
+
+ // set window size
+ if (w < 8 || w > 15) {
+ that.inflateEnd(z);
+ return Z_STREAM_ERROR;
+ }
+ that.wbits = w;
+
+ z.istate.blocks = new InfBlocks(z, 1 << w);
+
+ // reset state
+ inflateReset(z);
+ return Z_OK;
+ };
+
+ that.inflate = function(z, f) {
+ var r;
+ var b;
+
+ if (!z || !z.istate || !z.next_in)
+ return Z_STREAM_ERROR;
+ f = f == Z_FINISH ? Z_BUF_ERROR : Z_OK;
+ r = Z_BUF_ERROR;
+ while (true) {
+ // System.out.println("mode: "+z.istate.mode);
+ switch (z.istate.mode) {
+ case METHOD:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ if (((z.istate.method = z.read_byte(z.next_in_index++)) & 0xf) != Z_DEFLATED) {
+ z.istate.mode = BAD;
+ z.msg = "unknown compression method";
+ z.istate.marker = 5; // can't try inflateSync
+ break;
+ }
+ if ((z.istate.method >> 4) + 8 > z.istate.wbits) {
+ z.istate.mode = BAD;
+ z.msg = "invalid window size";
+ z.istate.marker = 5; // can't try inflateSync
+ break;
+ }
+ z.istate.mode = FLAG;
+ /* falls through */
+ case FLAG:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ b = (z.read_byte(z.next_in_index++)) & 0xff;
+
+ if ((((z.istate.method << 8) + b) % 31) !== 0) {
+ z.istate.mode = BAD;
+ z.msg = "incorrect header check";
+ z.istate.marker = 5; // can't try inflateSync
+ break;
+ }
+
+ if ((b & PRESET_DICT) === 0) {
+ z.istate.mode = BLOCKS;
+ break;
+ }
+ z.istate.mode = DICT4;
+ /* falls through */
+ case DICT4:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ z.istate.need = ((z.read_byte(z.next_in_index++) & 0xff) << 24) & 0xff000000;
+ z.istate.mode = DICT3;
+ /* falls through */
+ case DICT3:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ z.istate.need += ((z.read_byte(z.next_in_index++) & 0xff) << 16) & 0xff0000;
+ z.istate.mode = DICT2;
+ /* falls through */
+ case DICT2:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ z.istate.need += ((z.read_byte(z.next_in_index++) & 0xff) << 8) & 0xff00;
+ z.istate.mode = DICT1;
+ /* falls through */
+ case DICT1:
+
+ if (z.avail_in === 0)
+ return r;
+ r = f;
+
+ z.avail_in--;
+ z.total_in++;
+ z.istate.need += (z.read_byte(z.next_in_index++) & 0xff);
+ z.istate.mode = DICT0;
+ return Z_NEED_DICT;
+ case DICT0:
+ z.istate.mode = BAD;
+ z.msg = "need dictionary";
+ z.istate.marker = 0; // can try inflateSync
+ return Z_STREAM_ERROR;
+ case BLOCKS:
+
+ r = z.istate.blocks.proc(z, r);
+ if (r == Z_DATA_ERROR) {
+ z.istate.mode = BAD;
+ z.istate.marker = 0; // can try inflateSync
+ break;
+ }
+ if (r == Z_OK) {
+ r = f;
+ }
+ if (r != Z_STREAM_END) {
+ return r;
+ }
+ r = f;
+ z.istate.blocks.reset(z, z.istate.was);
+ z.istate.mode = DONE;
+ /* falls through */
+ case DONE:
+ return Z_STREAM_END;
+ case BAD:
+ return Z_DATA_ERROR;
+ default:
+ return Z_STREAM_ERROR;
+ }
+ }
+ };
+
+ that.inflateSetDictionary = function(z, dictionary, dictLength) {
+ var index = 0;
+ var length = dictLength;
+ if (!z || !z.istate || z.istate.mode != DICT0)
+ return Z_STREAM_ERROR;
+
+ if (length >= (1 << z.istate.wbits)) {
+ length = (1 << z.istate.wbits) - 1;
+ index = dictLength - length;
+ }
+ z.istate.blocks.set_dictionary(dictionary, index, length);
+ z.istate.mode = BLOCKS;
+ return Z_OK;
+ };
+
+ that.inflateSync = function(z) {
+ var n; // number of bytes to look at
+ var p; // pointer to bytes
+ var m; // number of marker bytes found in a row
+ var r, w; // temporaries to save total_in and total_out
+
+ // set up
+ if (!z || !z.istate)
+ return Z_STREAM_ERROR;
+ if (z.istate.mode != BAD) {
+ z.istate.mode = BAD;
+ z.istate.marker = 0;
+ }
+ if ((n = z.avail_in) === 0)
+ return Z_BUF_ERROR;
+ p = z.next_in_index;
+ m = z.istate.marker;
+
+ // search
+ while (n !== 0 && m < 4) {
+ if (z.read_byte(p) == mark[m]) {
+ m++;
+ } else if (z.read_byte(p) !== 0) {
+ m = 0;
+ } else {
+ m = 4 - m;
+ }
+ p++;
+ n--;
+ }
+
+ // restore
+ z.total_in += p - z.next_in_index;
+ z.next_in_index = p;
+ z.avail_in = n;
+ z.istate.marker = m;
+
+ // return no joy or set up to restart on a new block
+ if (m != 4) {
+ return Z_DATA_ERROR;
+ }
+ r = z.total_in;
+ w = z.total_out;
+ inflateReset(z);
+ z.total_in = r;
+ z.total_out = w;
+ z.istate.mode = BLOCKS;
+ return Z_OK;
+ };
+
+ // Returns true if inflate is currently at the end of a block generated
+ // by Z_SYNC_FLUSH or Z_FULL_FLUSH. This function is used by one PPP
+ // implementation to provide an additional safety check. PPP uses
+ // Z_SYNC_FLUSH
+ // but removes the length bytes of the resulting empty stored block. When
+ // decompressing, PPP checks that at the end of input packet, inflate is
+ // waiting for these length bytes.
+ that.inflateSyncPoint = function(z) {
+ if (!z || !z.istate || !z.istate.blocks)
+ return Z_STREAM_ERROR;
+ return z.istate.blocks.sync_point();
+ };
+ }
+
+ // ZStream
+
+ function ZStream() {
+ }
+
+ ZStream.prototype = {
+ inflateInit : function(bits) {
+ var that = this;
+ that.istate = new Inflate();
+ if (!bits)
+ bits = MAX_BITS;
+ return that.istate.inflateInit(that, bits);
+ },
+
+ inflate : function(f) {
+ var that = this;
+ if (!that.istate)
+ return Z_STREAM_ERROR;
+ return that.istate.inflate(that, f);
+ },
+
+ inflateEnd : function() {
+ var that = this;
+ if (!that.istate)
+ return Z_STREAM_ERROR;
+ var ret = that.istate.inflateEnd(that);
+ that.istate = null;
+ return ret;
+ },
+
+ inflateSync : function() {
+ var that = this;
+ if (!that.istate)
+ return Z_STREAM_ERROR;
+ return that.istate.inflateSync(that);
+ },
+ inflateSetDictionary : function(dictionary, dictLength) {
+ var that = this;
+ if (!that.istate)
+ return Z_STREAM_ERROR;
+ return that.istate.inflateSetDictionary(that, dictionary, dictLength);
+ },
+ read_byte : function(start) {
+ var that = this;
+ return that.next_in.subarray(start, start + 1)[0];
+ },
+ read_buf : function(start, size) {
+ var that = this;
+ return that.next_in.subarray(start, start + size);
+ }
+ };
+
+ // Inflater
+
+ function Inflater() {
+ var that = this;
+ var z = new ZStream();
+ var bufsize = 512;
+ var flush = Z_NO_FLUSH;
+ var buf = new Uint8Array(bufsize);
+ var nomoreinput = false;
+
+ z.inflateInit();
+ z.next_out = buf;
+
+ that.append = function(data, onprogress) {
+ var err, buffers = [], lastIndex = 0, bufferIndex = 0, bufferSize = 0, array;
+ if (data.length === 0)
+ return;
+ z.next_in_index = 0;
+ z.next_in = data;
+ z.avail_in = data.length;
+ do {
+ z.next_out_index = 0;
+ z.avail_out = bufsize;
+ if ((z.avail_in === 0) && (!nomoreinput)) { // if buffer is empty and more input is available, refill it
+ z.next_in_index = 0;
+ nomoreinput = true;
+ }
+ err = z.inflate(flush);
+ if (nomoreinput && (err === Z_BUF_ERROR)) {
+ if (z.avail_in !== 0)
+ throw new Error("inflating: bad input");
+ } else if (err !== Z_OK && err !== Z_STREAM_END)
+ throw new Error("inflating: " + z.msg);
+ if ((nomoreinput || err === Z_STREAM_END) && (z.avail_in === data.length))
+ throw new Error("inflating: bad input");
+ if (z.next_out_index)
+ if (z.next_out_index === bufsize)
+ buffers.push(new Uint8Array(buf));
+ else
+ buffers.push(new Uint8Array(buf.subarray(0, z.next_out_index)));
+ bufferSize += z.next_out_index;
+ if (onprogress && z.next_in_index > 0 && z.next_in_index != lastIndex) {
+ onprogress(z.next_in_index);
+ lastIndex = z.next_in_index;
+ }
+ } while (z.avail_in > 0 || z.avail_out === 0);
+ array = new Uint8Array(bufferSize);
+ buffers.forEach(function(chunk) {
+ array.set(chunk, bufferIndex);
+ bufferIndex += chunk.length;
+ });
+ return array;
+ };
+ that.flush = function() {
+ z.inflateEnd();
+ };
+ }
+
+ // 'zip' may not be defined in z-worker and some tests
+ var env = global.zip || global;
+ env.Inflater = env._jzlib_Inflater = Inflater;
+})(this);
diff --git a/lib/mime-types.js b/lib/mime-types.js
new file mode 100644
index 0000000..b6a9aa6
--- /dev/null
+++ b/lib/mime-types.js
@@ -0,0 +1,1002 @@
+/*
+ Copyright (c) 2013 Gildas Lormeau. All rights reserved.
+
+ Redistribution and use in source and binary forms, with or without
+ modification, are permitted provided that the following conditions are met:
+
+ 1. Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+ 2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the distribution.
+
+ 3. The names of the authors may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+ THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+ INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+ INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+ INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+ OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+ LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+(function() {
+ "use strict";
+ var table = {
+ "application" : {
+ "andrew-inset" : "ez",
+ "annodex" : "anx",
+ "atom+xml" : "atom",
+ "atomcat+xml" : "atomcat",
+ "atomserv+xml" : "atomsrv",
+ "bbolin" : "lin",
+ "cap" : [ "cap", "pcap" ],
+ "cu-seeme" : "cu",
+ "davmount+xml" : "davmount",
+ "dsptype" : "tsp",
+ "ecmascript" : [ "es", "ecma" ],
+ "futuresplash" : "spl",
+ "hta" : "hta",
+ "java-archive" : "jar",
+ "java-serialized-object" : "ser",
+ "java-vm" : "class",
+ "javascript" : "js",
+ "m3g" : "m3g",
+ "mac-binhex40" : "hqx",
+ "mathematica" : [ "nb", "ma", "mb" ],
+ "msaccess" : "mdb",
+ "msword" : [ "doc", "dot" ],
+ "mxf" : "mxf",
+ "oda" : "oda",
+ "ogg" : "ogx",
+ "pdf" : "pdf",
+ "pgp-keys" : "key",
+ "pgp-signature" : [ "asc", "sig" ],
+ "pics-rules" : "prf",
+ "postscript" : [ "ps", "ai", "eps", "epsi", "epsf", "eps2", "eps3" ],
+ "rar" : "rar",
+ "rdf+xml" : "rdf",
+ "rss+xml" : "rss",
+ "rtf" : "rtf",
+ "smil" : [ "smi", "smil" ],
+ "xhtml+xml" : [ "xhtml", "xht" ],
+ "xml" : [ "xml", "xsl", "xsd" ],
+ "xspf+xml" : "xspf",
+ "zip" : "zip",
+ "vnd.android.package-archive" : "apk",
+ "vnd.cinderella" : "cdy",
+ "vnd.google-earth.kml+xml" : "kml",
+ "vnd.google-earth.kmz" : "kmz",
+ "vnd.mozilla.xul+xml" : "xul",
+ "vnd.ms-excel" : [ "xls", "xlb", "xlt", "xlm", "xla", "xlc", "xlw" ],
+ "vnd.ms-pki.seccat" : "cat",
+ "vnd.ms-pki.stl" : "stl",
+ "vnd.ms-powerpoint" : [ "ppt", "pps", "pot" ],
+ "vnd.oasis.opendocument.chart" : "odc",
+ "vnd.oasis.opendocument.database" : "odb",
+ "vnd.oasis.opendocument.formula" : "odf",
+ "vnd.oasis.opendocument.graphics" : "odg",
+ "vnd.oasis.opendocument.graphics-template" : "otg",
+ "vnd.oasis.opendocument.image" : "odi",
+ "vnd.oasis.opendocument.presentation" : "odp",
+ "vnd.oasis.opendocument.presentation-template" : "otp",
+ "vnd.oasis.opendocument.spreadsheet" : "ods",
+ "vnd.oasis.opendocument.spreadsheet-template" : "ots",
+ "vnd.oasis.opendocument.text" : "odt",
+ "vnd.oasis.opendocument.text-master" : "odm",
+ "vnd.oasis.opendocument.text-template" : "ott",
+ "vnd.oasis.opendocument.text-web" : "oth",
+ "vnd.openxmlformats-officedocument.spreadsheetml.sheet" : "xlsx",
+ "vnd.openxmlformats-officedocument.spreadsheetml.template" : "xltx",
+ "vnd.openxmlformats-officedocument.presentationml.presentation" : "pptx",
+ "vnd.openxmlformats-officedocument.presentationml.slideshow" : "ppsx",
+ "vnd.openxmlformats-officedocument.presentationml.template" : "potx",
+ "vnd.openxmlformats-officedocument.wordprocessingml.document" : "docx",
+ "vnd.openxmlformats-officedocument.wordprocessingml.template" : "dotx",
+ "vnd.smaf" : "mmf",
+ "vnd.stardivision.calc" : "sdc",
+ "vnd.stardivision.chart" : "sds",
+ "vnd.stardivision.draw" : "sda",
+ "vnd.stardivision.impress" : "sdd",
+ "vnd.stardivision.math" : [ "sdf", "smf" ],
+ "vnd.stardivision.writer" : [ "sdw", "vor" ],
+ "vnd.stardivision.writer-global" : "sgl",
+ "vnd.sun.xml.calc" : "sxc",
+ "vnd.sun.xml.calc.template" : "stc",
+ "vnd.sun.xml.draw" : "sxd",
+ "vnd.sun.xml.draw.template" : "std",
+ "vnd.sun.xml.impress" : "sxi",
+ "vnd.sun.xml.impress.template" : "sti",
+ "vnd.sun.xml.math" : "sxm",
+ "vnd.sun.xml.writer" : "sxw",
+ "vnd.sun.xml.writer.global" : "sxg",
+ "vnd.sun.xml.writer.template" : "stw",
+ "vnd.symbian.install" : [ "sis", "sisx" ],
+ "vnd.visio" : [ "vsd", "vst", "vss", "vsw" ],
+ "vnd.wap.wbxml" : "wbxml",
+ "vnd.wap.wmlc" : "wmlc",
+ "vnd.wap.wmlscriptc" : "wmlsc",
+ "vnd.wordperfect" : "wpd",
+ "vnd.wordperfect5.1" : "wp5",
+ "x-123" : "wk",
+ "x-7z-compressed" : "7z",
+ "x-abiword" : "abw",
+ "x-apple-diskimage" : "dmg",
+ "x-bcpio" : "bcpio",
+ "x-bittorrent" : "torrent",
+ "x-cbr" : [ "cbr", "cba", "cbt", "cb7" ],
+ "x-cbz" : "cbz",
+ "x-cdf" : [ "cdf", "cda" ],
+ "x-cdlink" : "vcd",
+ "x-chess-pgn" : "pgn",
+ "x-cpio" : "cpio",
+ "x-csh" : "csh",
+ "x-debian-package" : [ "deb", "udeb" ],
+ "x-director" : [ "dcr", "dir", "dxr", "cst", "cct", "cxt", "w3d", "fgd", "swa" ],
+ "x-dms" : "dms",
+ "x-doom" : "wad",
+ "x-dvi" : "dvi",
+ "x-httpd-eruby" : "rhtml",
+ "x-font" : "pcf.Z",
+ "x-freemind" : "mm",
+ "x-gnumeric" : "gnumeric",
+ "x-go-sgf" : "sgf",
+ "x-graphing-calculator" : "gcf",
+ "x-gtar" : [ "gtar", "taz" ],
+ "x-hdf" : "hdf",
+ "x-httpd-php" : [ "phtml", "pht", "php" ],
+ "x-httpd-php-source" : "phps",
+ "x-httpd-php3" : "php3",
+ "x-httpd-php3-preprocessed" : "php3p",
+ "x-httpd-php4" : "php4",
+ "x-httpd-php5" : "php5",
+ "x-ica" : "ica",
+ "x-info" : "info",
+ "x-internet-signup" : [ "ins", "isp" ],
+ "x-iphone" : "iii",
+ "x-iso9660-image" : "iso",
+ "x-java-jnlp-file" : "jnlp",
+ "x-jmol" : "jmz",
+ "x-killustrator" : "kil",
+ "x-koan" : [ "skp", "skd", "skt", "skm" ],
+ "x-kpresenter" : [ "kpr", "kpt" ],
+ "x-kword" : [ "kwd", "kwt" ],
+ "x-latex" : "latex",
+ "x-lha" : "lha",
+ "x-lyx" : "lyx",
+ "x-lzh" : "lzh",
+ "x-lzx" : "lzx",
+ "x-maker" : [ "frm", "maker", "frame", "fm", "fb", "book", "fbdoc" ],
+ "x-ms-wmd" : "wmd",
+ "x-ms-wmz" : "wmz",
+ "x-msdos-program" : [ "com", "exe", "bat", "dll" ],
+ "x-msi" : "msi",
+ "x-netcdf" : [ "nc", "cdf" ],
+ "x-ns-proxy-autoconfig" : [ "pac", "dat" ],
+ "x-nwc" : "nwc",
+ "x-object" : "o",
+ "x-oz-application" : "oza",
+ "x-pkcs7-certreqresp" : "p7r",
+ "x-python-code" : [ "pyc", "pyo" ],
+ "x-qgis" : [ "qgs", "shp", "shx" ],
+ "x-quicktimeplayer" : "qtl",
+ "x-redhat-package-manager" : "rpm",
+ "x-ruby" : "rb",
+ "x-sh" : "sh",
+ "x-shar" : "shar",
+ "x-shockwave-flash" : [ "swf", "swfl" ],
+ "x-silverlight" : "scr",
+ "x-stuffit" : "sit",
+ "x-sv4cpio" : "sv4cpio",
+ "x-sv4crc" : "sv4crc",
+ "x-tar" : "tar",
+ "x-tcl" : "tcl",
+ "x-tex-gf" : "gf",
+ "x-tex-pk" : "pk",
+ "x-texinfo" : [ "texinfo", "texi" ],
+ "x-trash" : [ "~", "%", "bak", "old", "sik" ],
+ "x-troff" : [ "t", "tr", "roff" ],
+ "x-troff-man" : "man",
+ "x-troff-me" : "me",
+ "x-troff-ms" : "ms",
+ "x-ustar" : "ustar",
+ "x-wais-source" : "src",
+ "x-wingz" : "wz",
+ "x-x509-ca-cert" : [ "crt", "der", "cer" ],
+ "x-xcf" : "xcf",
+ "x-xfig" : "fig",
+ "x-xpinstall" : "xpi",
+ "applixware" : "aw",
+ "atomsvc+xml" : "atomsvc",
+ "ccxml+xml" : "ccxml",
+ "cdmi-capability" : "cdmia",
+ "cdmi-container" : "cdmic",
+ "cdmi-domain" : "cdmid",
+ "cdmi-object" : "cdmio",
+ "cdmi-queue" : "cdmiq",
+ "docbook+xml" : "dbk",
+ "dssc+der" : "dssc",
+ "dssc+xml" : "xdssc",
+ "emma+xml" : "emma",
+ "epub+zip" : "epub",
+ "exi" : "exi",
+ "font-tdpfr" : "pfr",
+ "gml+xml" : "gml",
+ "gpx+xml" : "gpx",
+ "gxf" : "gxf",
+ "hyperstudio" : "stk",
+ "inkml+xml" : [ "ink", "inkml" ],
+ "ipfix" : "ipfix",
+ "json" : "json",
+ "jsonml+json" : "jsonml",
+ "lost+xml" : "lostxml",
+ "mads+xml" : "mads",
+ "marc" : "mrc",
+ "marcxml+xml" : "mrcx",
+ "mathml+xml" : "mathml",
+ "mbox" : "mbox",
+ "mediaservercontrol+xml" : "mscml",
+ "metalink+xml" : "metalink",
+ "metalink4+xml" : "meta4",
+ "mets+xml" : "mets",
+ "mods+xml" : "mods",
+ "mp21" : [ "m21", "mp21" ],
+ "mp4" : "mp4s",
+ "oebps-package+xml" : "opf",
+ "omdoc+xml" : "omdoc",
+ "onenote" : [ "onetoc", "onetoc2", "onetmp", "onepkg" ],
+ "oxps" : "oxps",
+ "patch-ops-error+xml" : "xer",
+ "pgp-encrypted" : "pgp",
+ "pkcs10" : "p10",
+ "pkcs7-mime" : [ "p7m", "p7c" ],
+ "pkcs7-signature" : "p7s",
+ "pkcs8" : "p8",
+ "pkix-attr-cert" : "ac",
+ "pkix-crl" : "crl",
+ "pkix-pkipath" : "pkipath",
+ "pkixcmp" : "pki",
+ "pls+xml" : "pls",
+ "prs.cww" : "cww",
+ "pskc+xml" : "pskcxml",
+ "reginfo+xml" : "rif",
+ "relax-ng-compact-syntax" : "rnc",
+ "resource-lists+xml" : "rl",
+ "resource-lists-diff+xml" : "rld",
+ "rls-services+xml" : "rs",
+ "rpki-ghostbusters" : "gbr",
+ "rpki-manifest" : "mft",
+ "rpki-roa" : "roa",
+ "rsd+xml" : "rsd",
+ "sbml+xml" : "sbml",
+ "scvp-cv-request" : "scq",
+ "scvp-cv-response" : "scs",
+ "scvp-vp-request" : "spq",
+ "scvp-vp-response" : "spp",
+ "sdp" : "sdp",
+ "set-payment-initiation" : "setpay",
+ "set-registration-initiation" : "setreg",
+ "shf+xml" : "shf",
+ "sparql-query" : "rq",
+ "sparql-results+xml" : "srx",
+ "srgs" : "gram",
+ "srgs+xml" : "grxml",
+ "sru+xml" : "sru",
+ "ssdl+xml" : "ssdl",
+ "ssml+xml" : "ssml",
+ "tei+xml" : [ "tei", "teicorpus" ],
+ "thraud+xml" : "tfi",
+ "timestamped-data" : "tsd",
+ "vnd.3gpp.pic-bw-large" : "plb",
+ "vnd.3gpp.pic-bw-small" : "psb",
+ "vnd.3gpp.pic-bw-var" : "pvb",
+ "vnd.3gpp2.tcap" : "tcap",
+ "vnd.3m.post-it-notes" : "pwn",
+ "vnd.accpac.simply.aso" : "aso",
+ "vnd.accpac.simply.imp" : "imp",
+ "vnd.acucobol" : "acu",
+ "vnd.acucorp" : [ "atc", "acutc" ],
+ "vnd.adobe.air-application-installer-package+zip" : "air",
+ "vnd.adobe.formscentral.fcdt" : "fcdt",
+ "vnd.adobe.fxp" : [ "fxp", "fxpl" ],
+ "vnd.adobe.xdp+xml" : "xdp",
+ "vnd.adobe.xfdf" : "xfdf",
+ "vnd.ahead.space" : "ahead",
+ "vnd.airzip.filesecure.azf" : "azf",
+ "vnd.airzip.filesecure.azs" : "azs",
+ "vnd.amazon.ebook" : "azw",
+ "vnd.americandynamics.acc" : "acc",
+ "vnd.amiga.ami" : "ami",
+ "vnd.anser-web-certificate-issue-initiation" : "cii",
+ "vnd.anser-web-funds-transfer-initiation" : "fti",
+ "vnd.antix.game-component" : "atx",
+ "vnd.apple.installer+xml" : "mpkg",
+ "vnd.apple.mpegurl" : "m3u8",
+ "vnd.aristanetworks.swi" : "swi",
+ "vnd.astraea-software.iota" : "iota",
+ "vnd.audiograph" : "aep",
+ "vnd.blueice.multipass" : "mpm",
+ "vnd.bmi" : "bmi",
+ "vnd.businessobjects" : "rep",
+ "vnd.chemdraw+xml" : "cdxml",
+ "vnd.chipnuts.karaoke-mmd" : "mmd",
+ "vnd.claymore" : "cla",
+ "vnd.cloanto.rp9" : "rp9",
+ "vnd.clonk.c4group" : [ "c4g", "c4d", "c4f", "c4p", "c4u" ],
+ "vnd.cluetrust.cartomobile-config" : "c11amc",
+ "vnd.cluetrust.cartomobile-config-pkg" : "c11amz",
+ "vnd.commonspace" : "csp",
+ "vnd.contact.cmsg" : "cdbcmsg",
+ "vnd.cosmocaller" : "cmc",
+ "vnd.crick.clicker" : "clkx",
+ "vnd.crick.clicker.keyboard" : "clkk",
+ "vnd.crick.clicker.palette" : "clkp",
+ "vnd.crick.clicker.template" : "clkt",
+ "vnd.crick.clicker.wordbank" : "clkw",
+ "vnd.criticaltools.wbs+xml" : "wbs",
+ "vnd.ctc-posml" : "pml",
+ "vnd.cups-ppd" : "ppd",
+ "vnd.curl.car" : "car",
+ "vnd.curl.pcurl" : "pcurl",
+ "vnd.dart" : "dart",
+ "vnd.data-vision.rdz" : "rdz",
+ "vnd.dece.data" : [ "uvf", "uvvf", "uvd", "uvvd" ],
+ "vnd.dece.ttml+xml" : [ "uvt", "uvvt" ],
+ "vnd.dece.unspecified" : [ "uvx", "uvvx" ],
+ "vnd.dece.zip" : [ "uvz", "uvvz" ],
+ "vnd.denovo.fcselayout-link" : "fe_launch",
+ "vnd.dna" : "dna",
+ "vnd.dolby.mlp" : "mlp",
+ "vnd.dpgraph" : "dpg",
+ "vnd.dreamfactory" : "dfac",
+ "vnd.ds-keypoint" : "kpxx",
+ "vnd.dvb.ait" : "ait",
+ "vnd.dvb.service" : "svc",
+ "vnd.dynageo" : "geo",
+ "vnd.ecowin.chart" : "mag",
+ "vnd.enliven" : "nml",
+ "vnd.epson.esf" : "esf",
+ "vnd.epson.msf" : "msf",
+ "vnd.epson.quickanime" : "qam",
+ "vnd.epson.salt" : "slt",
+ "vnd.epson.ssf" : "ssf",
+ "vnd.eszigno3+xml" : [ "es3", "et3" ],
+ "vnd.ezpix-album" : "ez2",
+ "vnd.ezpix-package" : "ez3",
+ "vnd.fdf" : "fdf",
+ "vnd.fdsn.mseed" : "mseed",
+ "vnd.fdsn.seed" : [ "seed", "dataless" ],
+ "vnd.flographit" : "gph",
+ "vnd.fluxtime.clip" : "ftc",
+ "vnd.framemaker" : [ "fm", "frame", "maker", "book" ],
+ "vnd.frogans.fnc" : "fnc",
+ "vnd.frogans.ltf" : "ltf",
+ "vnd.fsc.weblaunch" : "fsc",
+ "vnd.fujitsu.oasys" : "oas",
+ "vnd.fujitsu.oasys2" : "oa2",
+ "vnd.fujitsu.oasys3" : "oa3",
+ "vnd.fujitsu.oasysgp" : "fg5",
+ "vnd.fujitsu.oasysprs" : "bh2",
+ "vnd.fujixerox.ddd" : "ddd",
+ "vnd.fujixerox.docuworks" : "xdw",
+ "vnd.fujixerox.docuworks.binder" : "xbd",
+ "vnd.fuzzysheet" : "fzs",
+ "vnd.genomatix.tuxedo" : "txd",
+ "vnd.geogebra.file" : "ggb",
+ "vnd.geogebra.tool" : "ggt",
+ "vnd.geometry-explorer" : [ "gex", "gre" ],
+ "vnd.geonext" : "gxt",
+ "vnd.geoplan" : "g2w",
+ "vnd.geospace" : "g3w",
+ "vnd.gmx" : "gmx",
+ "vnd.grafeq" : [ "gqf", "gqs" ],
+ "vnd.groove-account" : "gac",
+ "vnd.groove-help" : "ghf",
+ "vnd.groove-identity-message" : "gim",
+ "vnd.groove-injector" : "grv",
+ "vnd.groove-tool-message" : "gtm",
+ "vnd.groove-tool-template" : "tpl",
+ "vnd.groove-vcard" : "vcg",
+ "vnd.hal+xml" : "hal",
+ "vnd.handheld-entertainment+xml" : "zmm",
+ "vnd.hbci" : "hbci",
+ "vnd.hhe.lesson-player" : "les",
+ "vnd.hp-hpgl" : "hpgl",
+ "vnd.hp-hpid" : "hpid",
+ "vnd.hp-hps" : "hps",
+ "vnd.hp-jlyt" : "jlt",
+ "vnd.hp-pcl" : "pcl",
+ "vnd.hp-pclxl" : "pclxl",
+ "vnd.hydrostatix.sof-data" : "sfd-hdstx",
+ "vnd.ibm.minipay" : "mpy",
+ "vnd.ibm.modcap" : [ "afp", "listafp", "list3820" ],
+ "vnd.ibm.rights-management" : "irm",
+ "vnd.ibm.secure-container" : "sc",
+ "vnd.iccprofile" : [ "icc", "icm" ],
+ "vnd.igloader" : "igl",
+ "vnd.immervision-ivp" : "ivp",
+ "vnd.immervision-ivu" : "ivu",
+ "vnd.insors.igm" : "igm",
+ "vnd.intercon.formnet" : [ "xpw", "xpx" ],
+ "vnd.intergeo" : "i2g",
+ "vnd.intu.qbo" : "qbo",
+ "vnd.intu.qfx" : "qfx",
+ "vnd.ipunplugged.rcprofile" : "rcprofile",
+ "vnd.irepository.package+xml" : "irp",
+ "vnd.is-xpr" : "xpr",
+ "vnd.isac.fcs" : "fcs",
+ "vnd.jam" : "jam",
+ "vnd.jcp.javame.midlet-rms" : "rms",
+ "vnd.jisp" : "jisp",
+ "vnd.joost.joda-archive" : "joda",
+ "vnd.kahootz" : [ "ktz", "ktr" ],
+ "vnd.kde.karbon" : "karbon",
+ "vnd.kde.kchart" : "chrt",
+ "vnd.kde.kformula" : "kfo",
+ "vnd.kde.kivio" : "flw",
+ "vnd.kde.kontour" : "kon",
+ "vnd.kde.kpresenter" : [ "kpr", "kpt" ],
+ "vnd.kde.kspread" : "ksp",
+ "vnd.kde.kword" : [ "kwd", "kwt" ],
+ "vnd.kenameaapp" : "htke",
+ "vnd.kidspiration" : "kia",
+ "vnd.kinar" : [ "kne", "knp" ],
+ "vnd.koan" : [ "skp", "skd", "skt", "skm" ],
+ "vnd.kodak-descriptor" : "sse",
+ "vnd.las.las+xml" : "lasxml",
+ "vnd.llamagraphics.life-balance.desktop" : "lbd",
+ "vnd.llamagraphics.life-balance.exchange+xml" : "lbe",
+ "vnd.lotus-1-2-3" : "123",
+ "vnd.lotus-approach" : "apr",
+ "vnd.lotus-freelance" : "pre",
+ "vnd.lotus-notes" : "nsf",
+ "vnd.lotus-organizer" : "org",
+ "vnd.lotus-screencam" : "scm",
+ "vnd.lotus-wordpro" : "lwp",
+ "vnd.macports.portpkg" : "portpkg",
+ "vnd.mcd" : "mcd",
+ "vnd.medcalcdata" : "mc1",
+ "vnd.mediastation.cdkey" : "cdkey",
+ "vnd.mfer" : "mwf",
+ "vnd.mfmp" : "mfm",
+ "vnd.micrografx.flo" : "flo",
+ "vnd.micrografx.igx" : "igx",
+ "vnd.mif" : "mif",
+ "vnd.mobius.daf" : "daf",
+ "vnd.mobius.dis" : "dis",
+ "vnd.mobius.mbk" : "mbk",
+ "vnd.mobius.mqy" : "mqy",
+ "vnd.mobius.msl" : "msl",
+ "vnd.mobius.plc" : "plc",
+ "vnd.mobius.txf" : "txf",
+ "vnd.mophun.application" : "mpn",
+ "vnd.mophun.certificate" : "mpc",
+ "vnd.ms-artgalry" : "cil",
+ "vnd.ms-cab-compressed" : "cab",
+ "vnd.ms-excel.addin.macroenabled.12" : "xlam",
+ "vnd.ms-excel.sheet.binary.macroenabled.12" : "xlsb",
+ "vnd.ms-excel.sheet.macroenabled.12" : "xlsm",
+ "vnd.ms-excel.template.macroenabled.12" : "xltm",
+ "vnd.ms-fontobject" : "eot",
+ "vnd.ms-htmlhelp" : "chm",
+ "vnd.ms-ims" : "ims",
+ "vnd.ms-lrm" : "lrm",
+ "vnd.ms-officetheme" : "thmx",
+ "vnd.ms-powerpoint.addin.macroenabled.12" : "ppam",
+ "vnd.ms-powerpoint.presentation.macroenabled.12" : "pptm",
+ "vnd.ms-powerpoint.slide.macroenabled.12" : "sldm",
+ "vnd.ms-powerpoint.slideshow.macroenabled.12" : "ppsm",
+ "vnd.ms-powerpoint.template.macroenabled.12" : "potm",
+ "vnd.ms-project" : [ "mpp", "mpt" ],
+ "vnd.ms-word.document.macroenabled.12" : "docm",
+ "vnd.ms-word.template.macroenabled.12" : "dotm",
+ "vnd.ms-works" : [ "wps", "wks", "wcm", "wdb" ],
+ "vnd.ms-wpl" : "wpl",
+ "vnd.ms-xpsdocument" : "xps",
+ "vnd.mseq" : "mseq",
+ "vnd.musician" : "mus",
+ "vnd.muvee.style" : "msty",
+ "vnd.mynfc" : "taglet",
+ "vnd.neurolanguage.nlu" : "nlu",
+ "vnd.nitf" : [ "ntf", "nitf" ],
+ "vnd.noblenet-directory" : "nnd",
+ "vnd.noblenet-sealer" : "nns",
+ "vnd.noblenet-web" : "nnw",
+ "vnd.nokia.n-gage.data" : "ngdat",
+ "vnd.nokia.n-gage.symbian.install" : "n-gage",
+ "vnd.nokia.radio-preset" : "rpst",
+ "vnd.nokia.radio-presets" : "rpss",
+ "vnd.novadigm.edm" : "edm",
+ "vnd.novadigm.edx" : "edx",
+ "vnd.novadigm.ext" : "ext",
+ "vnd.oasis.opendocument.chart-template" : "otc",
+ "vnd.oasis.opendocument.formula-template" : "odft",
+ "vnd.oasis.opendocument.image-template" : "oti",
+ "vnd.olpc-sugar" : "xo",
+ "vnd.oma.dd2+xml" : "dd2",
+ "vnd.openofficeorg.extension" : "oxt",
+ "vnd.openxmlformats-officedocument.presentationml.slide" : "sldx",
+ "vnd.osgeo.mapguide.package" : "mgp",
+ "vnd.osgi.dp" : "dp",
+ "vnd.osgi.subsystem" : "esa",
+ "vnd.palm" : [ "pdb", "pqa", "oprc" ],
+ "vnd.pawaafile" : "paw",
+ "vnd.pg.format" : "str",
+ "vnd.pg.osasli" : "ei6",
+ "vnd.picsel" : "efif",
+ "vnd.pmi.widget" : "wg",
+ "vnd.pocketlearn" : "plf",
+ "vnd.powerbuilder6" : "pbd",
+ "vnd.previewsystems.box" : "box",
+ "vnd.proteus.magazine" : "mgz",
+ "vnd.publishare-delta-tree" : "qps",
+ "vnd.pvi.ptid1" : "ptid",
+ "vnd.quark.quarkxpress" : [ "qxd", "qxt", "qwd", "qwt", "qxl", "qxb" ],
+ "vnd.realvnc.bed" : "bed",
+ "vnd.recordare.musicxml" : "mxl",
+ "vnd.recordare.musicxml+xml" : "musicxml",
+ "vnd.rig.cryptonote" : "cryptonote",
+ "vnd.rn-realmedia" : "rm",
+ "vnd.rn-realmedia-vbr" : "rmvb",
+ "vnd.route66.link66+xml" : "link66",
+ "vnd.sailingtracker.track" : "st",
+ "vnd.seemail" : "see",
+ "vnd.sema" : "sema",
+ "vnd.semd" : "semd",
+ "vnd.semf" : "semf",
+ "vnd.shana.informed.formdata" : "ifm",
+ "vnd.shana.informed.formtemplate" : "itp",
+ "vnd.shana.informed.interchange" : "iif",
+ "vnd.shana.informed.package" : "ipk",
+ "vnd.simtech-mindmapper" : [ "twd", "twds" ],
+ "vnd.smart.teacher" : "teacher",
+ "vnd.solent.sdkm+xml" : [ "sdkm", "sdkd" ],
+ "vnd.spotfire.dxp" : "dxp",
+ "vnd.spotfire.sfs" : "sfs",
+ "vnd.stepmania.package" : "smzip",
+ "vnd.stepmania.stepchart" : "sm",
+ "vnd.sus-calendar" : [ "sus", "susp" ],
+ "vnd.svd" : "svd",
+ "vnd.syncml+xml" : "xsm",
+ "vnd.syncml.dm+wbxml" : "bdm",
+ "vnd.syncml.dm+xml" : "xdm",
+ "vnd.tao.intent-module-archive" : "tao",
+ "vnd.tcpdump.pcap" : [ "pcap", "cap", "dmp" ],
+ "vnd.tmobile-livetv" : "tmo",
+ "vnd.trid.tpt" : "tpt",
+ "vnd.triscape.mxs" : "mxs",
+ "vnd.trueapp" : "tra",
+ "vnd.ufdl" : [ "ufd", "ufdl" ],
+ "vnd.uiq.theme" : "utz",
+ "vnd.umajin" : "umj",
+ "vnd.unity" : "unityweb",
+ "vnd.uoml+xml" : "uoml",
+ "vnd.vcx" : "vcx",
+ "vnd.visionary" : "vis",
+ "vnd.vsf" : "vsf",
+ "vnd.webturbo" : "wtb",
+ "vnd.wolfram.player" : "nbp",
+ "vnd.wqd" : "wqd",
+ "vnd.wt.stf" : "stf",
+ "vnd.xara" : "xar",
+ "vnd.xfdl" : "xfdl",
+ "vnd.yamaha.hv-dic" : "hvd",
+ "vnd.yamaha.hv-script" : "hvs",
+ "vnd.yamaha.hv-voice" : "hvp",
+ "vnd.yamaha.openscoreformat" : "osf",
+ "vnd.yamaha.openscoreformat.osfpvg+xml" : "osfpvg",
+ "vnd.yamaha.smaf-audio" : "saf",
+ "vnd.yamaha.smaf-phrase" : "spf",
+ "vnd.yellowriver-custom-menu" : "cmp",
+ "vnd.zul" : [ "zir", "zirz" ],
+ "vnd.zzazz.deck+xml" : "zaz",
+ "voicexml+xml" : "vxml",
+ "widget" : "wgt",
+ "winhlp" : "hlp",
+ "wsdl+xml" : "wsdl",
+ "wspolicy+xml" : "wspolicy",
+ "x-ace-compressed" : "ace",
+ "x-authorware-bin" : [ "aab", "x32", "u32", "vox" ],
+ "x-authorware-map" : "aam",
+ "x-authorware-seg" : "aas",
+ "x-blorb" : [ "blb", "blorb" ],
+ "x-bzip" : "bz",
+ "x-bzip2" : [ "bz2", "boz" ],
+ "x-cfs-compressed" : "cfs",
+ "x-chat" : "chat",
+ "x-conference" : "nsc",
+ "x-dgc-compressed" : "dgc",
+ "x-dtbncx+xml" : "ncx",
+ "x-dtbook+xml" : "dtb",
+ "x-dtbresource+xml" : "res",
+ "x-eva" : "eva",
+ "x-font-bdf" : "bdf",
+ "x-font-ghostscript" : "gsf",
+ "x-font-linux-psf" : "psf",
+ "x-font-otf" : "otf",
+ "x-font-pcf" : "pcf",
+ "x-font-snf" : "snf",
+ "x-font-ttf" : [ "ttf", "ttc" ],
+ "x-font-type1" : [ "pfa", "pfb", "pfm", "afm" ],
+ "x-font-woff" : "woff",
+ "x-freearc" : "arc",
+ "x-gca-compressed" : "gca",
+ "x-glulx" : "ulx",
+ "x-gramps-xml" : "gramps",
+ "x-install-instructions" : "install",
+ "x-lzh-compressed" : [ "lzh", "lha" ],
+ "x-mie" : "mie",
+ "x-mobipocket-ebook" : [ "prc", "mobi" ],
+ "x-ms-application" : "application",
+ "x-ms-shortcut" : "lnk",
+ "x-ms-xbap" : "xbap",
+ "x-msbinder" : "obd",
+ "x-mscardfile" : "crd",
+ "x-msclip" : "clp",
+ "x-msdownload" : [ "exe", "dll", "com", "bat", "msi" ],
+ "x-msmediaview" : [ "mvb", "m13", "m14" ],
+ "x-msmetafile" : [ "wmf", "wmz", "emf", "emz" ],
+ "x-msmoney" : "mny",
+ "x-mspublisher" : "pub",
+ "x-msschedule" : "scd",
+ "x-msterminal" : "trm",
+ "x-mswrite" : "wri",
+ "x-nzb" : "nzb",
+ "x-pkcs12" : [ "p12", "pfx" ],
+ "x-pkcs7-certificates" : [ "p7b", "spc" ],
+ "x-research-info-systems" : "ris",
+ "x-silverlight-app" : "xap",
+ "x-sql" : "sql",
+ "x-stuffitx" : "sitx",
+ "x-subrip" : "srt",
+ "x-t3vm-image" : "t3",
+ "x-tads" : "gam",
+ "x-tex" : "tex",
+ "x-tex-tfm" : "tfm",
+ "x-tgif" : "obj",
+ "x-xliff+xml" : "xlf",
+ "x-xz" : "xz",
+ "x-zmachine" : [ "z1", "z2", "z3", "z4", "z5", "z6", "z7", "z8" ],
+ "xaml+xml" : "xaml",
+ "xcap-diff+xml" : "xdf",
+ "xenc+xml" : "xenc",
+ "xml-dtd" : "dtd",
+ "xop+xml" : "xop",
+ "xproc+xml" : "xpl",
+ "xslt+xml" : "xslt",
+ "xv+xml" : [ "mxml", "xhvml", "xvml", "xvm" ],
+ "yang" : "yang",
+ "yin+xml" : "yin",
+ "envoy" : "evy",
+ "fractals" : "fif",
+ "internet-property-stream" : "acx",
+ "olescript" : "axs",
+ "vnd.ms-outlook" : "msg",
+ "vnd.ms-pkicertstore" : "sst",
+ "x-compress" : "z",
+ "x-compressed" : "tgz",
+ "x-gzip" : "gz",
+ "x-perfmon" : [ "pma", "pmc", "pml", "pmr", "pmw" ],
+ "x-pkcs7-mime" : [ "p7c", "p7m" ],
+ "ynd.ms-pkipko" : "pko"
+ },
+ "audio" : {
+ "amr" : "amr",
+ "amr-wb" : "awb",
+ "annodex" : "axa",
+ "basic" : [ "au", "snd" ],
+ "flac" : "flac",
+ "midi" : [ "mid", "midi", "kar", "rmi" ],
+ "mpeg" : [ "mpga", "mpega", "mp2", "mp3", "m4a", "mp2a", "m2a", "m3a" ],
+ "mpegurl" : "m3u",
+ "ogg" : [ "oga", "ogg", "spx" ],
+ "prs.sid" : "sid",
+ "x-aiff" : [ "aif", "aiff", "aifc" ],
+ "x-gsm" : "gsm",
+ "x-ms-wma" : "wma",
+ "x-ms-wax" : "wax",
+ "x-pn-realaudio" : "ram",
+ "x-realaudio" : "ra",
+ "x-sd2" : "sd2",
+ "x-wav" : "wav",
+ "adpcm" : "adp",
+ "mp4" : "mp4a",
+ "s3m" : "s3m",
+ "silk" : "sil",
+ "vnd.dece.audio" : [ "uva", "uvva" ],
+ "vnd.digital-winds" : "eol",
+ "vnd.dra" : "dra",
+ "vnd.dts" : "dts",
+ "vnd.dts.hd" : "dtshd",
+ "vnd.lucent.voice" : "lvp",
+ "vnd.ms-playready.media.pya" : "pya",
+ "vnd.nuera.ecelp4800" : "ecelp4800",
+ "vnd.nuera.ecelp7470" : "ecelp7470",
+ "vnd.nuera.ecelp9600" : "ecelp9600",
+ "vnd.rip" : "rip",
+ "webm" : "weba",
+ "x-aac" : "aac",
+ "x-caf" : "caf",
+ "x-matroska" : "mka",
+ "x-pn-realaudio-plugin" : "rmp",
+ "xm" : "xm",
+ "mid" : [ "mid", "rmi" ]
+ },
+ "chemical" : {
+ "x-alchemy" : "alc",
+ "x-cache" : [ "cac", "cache" ],
+ "x-cache-csf" : "csf",
+ "x-cactvs-binary" : [ "cbin", "cascii", "ctab" ],
+ "x-cdx" : "cdx",
+ "x-chem3d" : "c3d",
+ "x-cif" : "cif",
+ "x-cmdf" : "cmdf",
+ "x-cml" : "cml",
+ "x-compass" : "cpa",
+ "x-crossfire" : "bsd",
+ "x-csml" : [ "csml", "csm" ],
+ "x-ctx" : "ctx",
+ "x-cxf" : [ "cxf", "cef" ],
+ "x-embl-dl-nucleotide" : [ "emb", "embl" ],
+ "x-gamess-input" : [ "inp", "gam", "gamin" ],
+ "x-gaussian-checkpoint" : [ "fch", "fchk" ],
+ "x-gaussian-cube" : "cub",
+ "x-gaussian-input" : [ "gau", "gjc", "gjf" ],
+ "x-gaussian-log" : "gal",
+ "x-gcg8-sequence" : "gcg",
+ "x-genbank" : "gen",
+ "x-hin" : "hin",
+ "x-isostar" : [ "istr", "ist" ],
+ "x-jcamp-dx" : [ "jdx", "dx" ],
+ "x-kinemage" : "kin",
+ "x-macmolecule" : "mcm",
+ "x-macromodel-input" : [ "mmd", "mmod" ],
+ "x-mdl-molfile" : "mol",
+ "x-mdl-rdfile" : "rd",
+ "x-mdl-rxnfile" : "rxn",
+ "x-mdl-sdfile" : [ "sd", "sdf" ],
+ "x-mdl-tgf" : "tgf",
+ "x-mmcif" : "mcif",
+ "x-mol2" : "mol2",
+ "x-molconn-Z" : "b",
+ "x-mopac-graph" : "gpt",
+ "x-mopac-input" : [ "mop", "mopcrt", "mpc", "zmt" ],
+ "x-mopac-out" : "moo",
+ "x-ncbi-asn1" : "asn",
+ "x-ncbi-asn1-ascii" : [ "prt", "ent" ],
+ "x-ncbi-asn1-binary" : [ "val", "aso" ],
+ "x-pdb" : [ "pdb", "ent" ],
+ "x-rosdal" : "ros",
+ "x-swissprot" : "sw",
+ "x-vamas-iso14976" : "vms",
+ "x-vmd" : "vmd",
+ "x-xtel" : "xtel",
+ "x-xyz" : "xyz"
+ },
+ "image" : {
+ "gif" : "gif",
+ "ief" : "ief",
+ "jpeg" : [ "jpeg", "jpg", "jpe" ],
+ "pcx" : "pcx",
+ "png" : "png",
+ "svg+xml" : [ "svg", "svgz" ],
+ "tiff" : [ "tiff", "tif" ],
+ "vnd.djvu" : [ "djvu", "djv" ],
+ "vnd.wap.wbmp" : "wbmp",
+ "x-canon-cr2" : "cr2",
+ "x-canon-crw" : "crw",
+ "x-cmu-raster" : "ras",
+ "x-coreldraw" : "cdr",
+ "x-coreldrawpattern" : "pat",
+ "x-coreldrawtemplate" : "cdt",
+ "x-corelphotopaint" : "cpt",
+ "x-epson-erf" : "erf",
+ "x-icon" : "ico",
+ "x-jg" : "art",
+ "x-jng" : "jng",
+ "x-nikon-nef" : "nef",
+ "x-olympus-orf" : "orf",
+ "x-photoshop" : "psd",
+ "x-portable-anymap" : "pnm",
+ "x-portable-bitmap" : "pbm",
+ "x-portable-graymap" : "pgm",
+ "x-portable-pixmap" : "ppm",
+ "x-rgb" : "rgb",
+ "x-xbitmap" : "xbm",
+ "x-xpixmap" : "xpm",
+ "x-xwindowdump" : "xwd",
+ "bmp" : "bmp",
+ "cgm" : "cgm",
+ "g3fax" : "g3",
+ "ktx" : "ktx",
+ "prs.btif" : "btif",
+ "sgi" : "sgi",
+ "vnd.dece.graphic" : [ "uvi", "uvvi", "uvg", "uvvg" ],
+ "vnd.dwg" : "dwg",
+ "vnd.dxf" : "dxf",
+ "vnd.fastbidsheet" : "fbs",
+ "vnd.fpx" : "fpx",
+ "vnd.fst" : "fst",
+ "vnd.fujixerox.edmics-mmr" : "mmr",
+ "vnd.fujixerox.edmics-rlc" : "rlc",
+ "vnd.ms-modi" : "mdi",
+ "vnd.ms-photo" : "wdp",
+ "vnd.net-fpx" : "npx",
+ "vnd.xiff" : "xif",
+ "webp" : "webp",
+ "x-3ds" : "3ds",
+ "x-cmx" : "cmx",
+ "x-freehand" : [ "fh", "fhc", "fh4", "fh5", "fh7" ],
+ "x-pict" : [ "pic", "pct" ],
+ "x-tga" : "tga",
+ "cis-cod" : "cod",
+ "pipeg" : "jfif"
+ },
+ "message" : {
+ "rfc822" : [ "eml", "mime", "mht", "mhtml", "nws" ]
+ },
+ "model" : {
+ "iges" : [ "igs", "iges" ],
+ "mesh" : [ "msh", "mesh", "silo" ],
+ "vrml" : [ "wrl", "vrml" ],
+ "x3d+vrml" : [ "x3dv", "x3dvz" ],
+ "x3d+xml" : [ "x3d", "x3dz" ],
+ "x3d+binary" : [ "x3db", "x3dbz" ],
+ "vnd.collada+xml" : "dae",
+ "vnd.dwf" : "dwf",
+ "vnd.gdl" : "gdl",
+ "vnd.gtw" : "gtw",
+ "vnd.mts" : "mts",
+ "vnd.vtu" : "vtu"
+ },
+ "text" : {
+ "cache-manifest" : [ "manifest", "appcache" ],
+ "calendar" : [ "ics", "icz", "ifb" ],
+ "css" : "css",
+ "csv" : "csv",
+ "h323" : "323",
+ "html" : [ "html", "htm", "shtml", "stm" ],
+ "iuls" : "uls",
+ "mathml" : "mml",
+ "plain" : [ "txt", "text", "brf", "conf", "def", "list", "log", "in", "bas" ],
+ "richtext" : "rtx",
+ "scriptlet" : [ "sct", "wsc" ],
+ "texmacs" : [ "tm", "ts" ],
+ "tab-separated-values" : "tsv",
+ "vnd.sun.j2me.app-descriptor" : "jad",
+ "vnd.wap.wml" : "wml",
+ "vnd.wap.wmlscript" : "wmls",
+ "x-bibtex" : "bib",
+ "x-boo" : "boo",
+ "x-c++hdr" : [ "h++", "hpp", "hxx", "hh" ],
+ "x-c++src" : [ "c++", "cpp", "cxx", "cc" ],
+ "x-component" : "htc",
+ "x-dsrc" : "d",
+ "x-diff" : [ "diff", "patch" ],
+ "x-haskell" : "hs",
+ "x-java" : "java",
+ "x-literate-haskell" : "lhs",
+ "x-moc" : "moc",
+ "x-pascal" : [ "p", "pas" ],
+ "x-pcs-gcd" : "gcd",
+ "x-perl" : [ "pl", "pm" ],
+ "x-python" : "py",
+ "x-scala" : "scala",
+ "x-setext" : "etx",
+ "x-tcl" : [ "tcl", "tk" ],
+ "x-tex" : [ "tex", "ltx", "sty", "cls" ],
+ "x-vcalendar" : "vcs",
+ "x-vcard" : "vcf",
+ "n3" : "n3",
+ "prs.lines.tag" : "dsc",
+ "sgml" : [ "sgml", "sgm" ],
+ "troff" : [ "t", "tr", "roff", "man", "me", "ms" ],
+ "turtle" : "ttl",
+ "uri-list" : [ "uri", "uris", "urls" ],
+ "vcard" : "vcard",
+ "vnd.curl" : "curl",
+ "vnd.curl.dcurl" : "dcurl",
+ "vnd.curl.scurl" : "scurl",
+ "vnd.curl.mcurl" : "mcurl",
+ "vnd.dvb.subtitle" : "sub",
+ "vnd.fly" : "fly",
+ "vnd.fmi.flexstor" : "flx",
+ "vnd.graphviz" : "gv",
+ "vnd.in3d.3dml" : "3dml",
+ "vnd.in3d.spot" : "spot",
+ "x-asm" : [ "s", "asm" ],
+ "x-c" : [ "c", "cc", "cxx", "cpp", "h", "hh", "dic" ],
+ "x-fortran" : [ "f", "for", "f77", "f90" ],
+ "x-opml" : "opml",
+ "x-nfo" : "nfo",
+ "x-sfv" : "sfv",
+ "x-uuencode" : "uu",
+ "webviewhtml" : "htt"
+ },
+ "video" : {
+ "3gpp" : "3gp",
+ "annodex" : "axv",
+ "dl" : "dl",
+ "dv" : [ "dif", "dv" ],
+ "fli" : "fli",
+ "gl" : "gl",
+ "mpeg" : [ "mpeg", "mpg", "mpe", "m1v", "m2v", "mp2", "mpa", "mpv2" ],
+ "mp4" : [ "mp4", "mp4v", "mpg4" ],
+ "quicktime" : [ "qt", "mov" ],
+ "ogg" : "ogv",
+ "vnd.mpegurl" : [ "mxu", "m4u" ],
+ "x-flv" : "flv",
+ "x-la-asf" : [ "lsf", "lsx" ],
+ "x-mng" : "mng",
+ "x-ms-asf" : [ "asf", "asx", "asr" ],
+ "x-ms-wm" : "wm",
+ "x-ms-wmv" : "wmv",
+ "x-ms-wmx" : "wmx",
+ "x-ms-wvx" : "wvx",
+ "x-msvideo" : "avi",
+ "x-sgi-movie" : "movie",
+ "x-matroska" : [ "mpv", "mkv", "mk3d", "mks" ],
+ "3gpp2" : "3g2",
+ "h261" : "h261",
+ "h263" : "h263",
+ "h264" : "h264",
+ "jpeg" : "jpgv",
+ "jpm" : [ "jpm", "jpgm" ],
+ "mj2" : [ "mj2", "mjp2" ],
+ "vnd.dece.hd" : [ "uvh", "uvvh" ],
+ "vnd.dece.mobile" : [ "uvm", "uvvm" ],
+ "vnd.dece.pd" : [ "uvp", "uvvp" ],
+ "vnd.dece.sd" : [ "uvs", "uvvs" ],
+ "vnd.dece.video" : [ "uvv", "uvvv" ],
+ "vnd.dvb.file" : "dvb",
+ "vnd.fvt" : "fvt",
+ "vnd.ms-playready.media.pyv" : "pyv",
+ "vnd.uvvu.mp4" : [ "uvu", "uvvu" ],
+ "vnd.vivo" : "viv",
+ "webm" : "webm",
+ "x-f4v" : "f4v",
+ "x-m4v" : "m4v",
+ "x-ms-vob" : "vob",
+ "x-smv" : "smv"
+ },
+ "x-conference" : {
+ "x-cooltalk" : "ice"
+ },
+ "x-world" : {
+ "x-vrml" : [ "vrm", "vrml", "wrl", "flr", "wrz", "xaf", "xof" ]
+ }
+ };
+
+ var mimeTypes = (function() {
+ var type, subtype, val, index, mimeTypes = {};
+ for (type in table) {
+ if (table.hasOwnProperty(type)) {
+ for (subtype in table[type]) {
+ if (table[type].hasOwnProperty(subtype)) {
+ val = table[type][subtype];
+ if (typeof val == "string") {
+ mimeTypes[val] = type + "/" + subtype;
+ } else {
+ for (index = 0; index < val.length; index++) {
+ mimeTypes[val[index]] = type + "/" + subtype;
+ }
+ }
+ }
+ }
+ }
+ }
+ return mimeTypes;
+ })();
+
+ zip.getMimeType = function(filename) {
+ var defaultValue = "application/octet-stream";
+ return filename && mimeTypes[filename.split(".").pop().toLowerCase()] || defaultValue;
+ };
+
+})();
diff --git a/lib/pako/codecs.js b/lib/pako/codecs.js
new file mode 100644
index 0000000..9ce80fc
--- /dev/null
+++ b/lib/pako/codecs.js
@@ -0,0 +1,64 @@
+/// wrapper for pako (https://github.com/nodeca/pako)
+
+/* globals pako */
+(function(global) {
+ "use strict";
+
+ function Codec(isDeflater, options) {
+ var newOptions = { raw: true, chunkSize: 1024 * 1024 };
+ if (options && typeof options.level === 'number')
+ newOptions.level = options.level;
+ this._backEnd = isDeflater?
+ new pako.Deflate(newOptions) :
+ new pako.Inflate(newOptions);
+ this._chunks = [];
+ this._dataLength = 0;
+ this._backEnd.onData = this._onData.bind(this);
+ }
+ Codec.prototype._onData = function _onData(chunk) {
+ this._chunks.push(chunk);
+ this._dataLength += chunk.length;
+ };
+ Codec.prototype._fetchData = function _fetchData() {
+ var be = this._backEnd;
+ if (be.err !== 0)
+ throw new Error(be.msg);
+ var chunks = this._chunks;
+ var data;
+ if (chunks.length === 1)
+ data = chunks[0];
+ else if (chunks.length > 1) {
+ data = new Uint8Array(this._dataLength);
+ for (var i = 0, n = chunks.length, off = 0; i < n; i++) {
+ var chunk = chunks[i];
+ data.set(chunk, off);
+ off += chunk.length;
+ }
+ }
+ chunks.length = 0;
+ this._dataLength = 0;
+ return data;
+ };
+ Codec.prototype.append = function append(bytes, onprogress) {
+ this._backEnd.push(bytes, false);
+ return this._fetchData();
+ };
+ Codec.prototype.flush = function flush() {
+ this._backEnd.push(new Uint8Array(0), true);
+ return this._fetchData();
+ };
+
+ function Deflater(options) {
+ Codec.call(this, true, options);
+ }
+ Deflater.prototype = Object.create(Codec.prototype);
+ function Inflater() {
+ Codec.call(this, false);
+ }
+ Inflater.prototype = Object.create(Codec.prototype);
+
+ // 'zip' may not be defined in z-worker and some tests
+ var env = global.zip || global;
+ env.Deflater = env._pako_Deflater = Deflater;
+ env.Inflater = env._pako_Inflater = Inflater;
+})(this);
\ No newline at end of file
diff --git a/lib/z-worker.js b/lib/z-worker.js
new file mode 100644
index 0000000..3e4019e
--- /dev/null
+++ b/lib/z-worker.js
@@ -0,0 +1,153 @@
+/* jshint worker:true */
+(function main(global) {
+ "use strict";
+
+ if (global.zWorkerInitialized)
+ throw new Error('z-worker.js should be run only once');
+ global.zWorkerInitialized = true;
+
+ addEventListener("message", function(event) {
+ var message = event.data, type = message.type, sn = message.sn;
+ var handler = handlers[type];
+ if (handler) {
+ try {
+ handler(message);
+ } catch (e) {
+ onError(type, sn, e);
+ }
+ }
+ //for debug
+ //postMessage({type: 'echo', originalType: type, sn: sn});
+ });
+
+ var handlers = {
+ importScripts: doImportScripts,
+ newTask: newTask,
+ append: processData,
+ flush: processData,
+ };
+
+ // deflater/inflater tasks indexed by serial numbers
+ var tasks = {};
+
+ function doImportScripts(msg) {
+ if (msg.scripts && msg.scripts.length > 0)
+ importScripts.apply(undefined, msg.scripts);
+ postMessage({type: 'importScripts'});
+ }
+
+ function newTask(msg) {
+ var CodecClass = global[msg.codecClass];
+ var sn = msg.sn;
+ if (tasks[sn])
+ throw Error('duplicated sn');
+ tasks[sn] = {
+ codec: new CodecClass(msg.options),
+ crcInput: msg.crcType === 'input',
+ crcOutput: msg.crcType === 'output',
+ crc: new Crc32(),
+ };
+ postMessage({type: 'newTask', sn: sn});
+ }
+
+ // performance may not be supported
+ var now = global.performance ? global.performance.now.bind(global.performance) : Date.now;
+
+ function processData(msg) {
+ var sn = msg.sn, type = msg.type, input = msg.data;
+ var task = tasks[sn];
+ // allow creating codec on first append
+ if (!task && msg.codecClass) {
+ newTask(msg);
+ task = tasks[sn];
+ }
+ var isAppend = type === 'append';
+ var start = now();
+ var output;
+ if (isAppend) {
+ try {
+ output = task.codec.append(input, function onprogress(loaded) {
+ postMessage({type: 'progress', sn: sn, loaded: loaded});
+ });
+ } catch (e) {
+ delete tasks[sn];
+ throw e;
+ }
+ } else {
+ delete tasks[sn];
+ output = task.codec.flush();
+ }
+ var codecTime = now() - start;
+
+ start = now();
+ if (input && task.crcInput)
+ task.crc.append(input);
+ if (output && task.crcOutput)
+ task.crc.append(output);
+ var crcTime = now() - start;
+
+ var rmsg = {type: type, sn: sn, codecTime: codecTime, crcTime: crcTime};
+ var transferables = [];
+ if (output) {
+ rmsg.data = output;
+ transferables.push(output.buffer);
+ }
+ if (!isAppend && (task.crcInput || task.crcOutput))
+ rmsg.crc = task.crc.get();
+
+ // posting a message with transferables will fail on IE10
+ try {
+ postMessage(rmsg, transferables);
+ } catch(ex) {
+ postMessage(rmsg); // retry without transferables
+ }
+ }
+
+ function onError(type, sn, e) {
+ var msg = {
+ type: type,
+ sn: sn,
+ error: formatError(e)
+ };
+ postMessage(msg);
+ }
+
+ function formatError(e) {
+ return { message: e.message, stack: e.stack };
+ }
+
+ // Crc32 code copied from file zip.js
+ function Crc32() {
+ this.crc = -1;
+ }
+ Crc32.prototype.append = function append(data) {
+ var crc = this.crc | 0, table = this.table;
+ for (var offset = 0, len = data.length | 0; offset < len; offset++)
+ crc = (crc >>> 8) ^ table[(crc ^ data[offset]) & 0xFF];
+ this.crc = crc;
+ };
+ Crc32.prototype.get = function get() {
+ return ~this.crc;
+ };
+ Crc32.prototype.table = (function() {
+ var i, j, t, table = []; // Uint32Array is actually slower than []
+ for (i = 0; i < 256; i++) {
+ t = i;
+ for (j = 0; j < 8; j++)
+ if (t & 1)
+ t = (t >>> 1) ^ 0xEDB88320;
+ else
+ t = t >>> 1;
+ table[i] = t;
+ }
+ return table;
+ })();
+
+ // "no-op" codec
+ function NOOP() {}
+ global.NOOP = NOOP;
+ NOOP.prototype.append = function append(bytes, onprogress) {
+ return bytes;
+ };
+ NOOP.prototype.flush = function flush() {};
+})(this);
diff --git a/lib/zip-ext.js b/lib/zip-ext.js
new file mode 100644
index 0000000..0e08dae
--- /dev/null
+++ b/lib/zip-ext.js
@@ -0,0 +1,242 @@
+/*
+ Copyright (c) 2013 Gildas Lormeau. All rights reserved.
+
+ Redistribution and use in source and binary forms, with or without
+ modification, are permitted provided that the following conditions are met:
+
+ 1. Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+ 2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the distribution.
+
+ 3. The names of the authors may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+ THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+ INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+ INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+ INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+ OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+ LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+(function() {
+ "use strict";
+
+ var ERR_HTTP_RANGE = "HTTP Range not supported.";
+
+ var Reader = zip.Reader;
+ var Writer = zip.Writer;
+
+ var ZipDirectoryEntry;
+
+ var appendABViewSupported;
+ try {
+ appendABViewSupported = new Blob([ new DataView(new ArrayBuffer(0)) ]).size === 0;
+ } catch (e) {
+ }
+
+ function HttpReader(url) {
+ var that = this;
+
+ function getData(callback, onerror) {
+ var request;
+ if (!that.data) {
+ request = new XMLHttpRequest();
+ request.addEventListener("load", function() {
+ if (!that.size)
+ that.size = Number(request.getResponseHeader("Content-Length"));
+ that.data = new Uint8Array(request.response);
+ callback();
+ }, false);
+ request.addEventListener("error", onerror, false);
+ request.open("GET", url);
+ request.responseType = "arraybuffer";
+ request.send();
+ } else
+ callback();
+ }
+
+ function init(callback, onerror) {
+ var request = new XMLHttpRequest();
+ request.addEventListener("load", function() {
+ that.size = Number(request.getResponseHeader("Content-Length"));
+ callback();
+ }, false);
+ request.addEventListener("error", onerror, false);
+ request.open("HEAD", url);
+ request.send();
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ getData(function() {
+ callback(new Uint8Array(that.data.subarray(index, index + length)));
+ }, onerror);
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ HttpReader.prototype = new Reader();
+ HttpReader.prototype.constructor = HttpReader;
+
+ function HttpRangeReader(url) {
+ var that = this;
+
+ function init(callback, onerror) {
+ var request = new XMLHttpRequest();
+ request.addEventListener("load", function() {
+ that.size = Number(request.getResponseHeader("Content-Length"));
+ if (request.getResponseHeader("Accept-Ranges") == "bytes")
+ callback();
+ else
+ onerror(ERR_HTTP_RANGE);
+ }, false);
+ request.addEventListener("error", onerror, false);
+ request.open("HEAD", url);
+ request.send();
+ }
+
+ function readArrayBuffer(index, length, callback, onerror) {
+ var request = new XMLHttpRequest();
+ request.open("GET", url);
+ request.responseType = "arraybuffer";
+ request.setRequestHeader("Range", "bytes=" + index + "-" + (index + length - 1));
+ request.addEventListener("load", function() {
+ callback(request.response);
+ }, false);
+ request.addEventListener("error", onerror, false);
+ request.send();
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ readArrayBuffer(index, length, function(arraybuffer) {
+ callback(new Uint8Array(arraybuffer));
+ }, onerror);
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ HttpRangeReader.prototype = new Reader();
+ HttpRangeReader.prototype.constructor = HttpRangeReader;
+
+ function ArrayBufferReader(arrayBuffer) {
+ var that = this;
+
+ function init(callback, onerror) {
+ that.size = arrayBuffer.byteLength;
+ callback();
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ callback(new Uint8Array(arrayBuffer.slice(index, index + length)));
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ ArrayBufferReader.prototype = new Reader();
+ ArrayBufferReader.prototype.constructor = ArrayBufferReader;
+
+ function ArrayBufferWriter() {
+ var array, that = this;
+
+ function init(callback, onerror) {
+ array = new Uint8Array();
+ callback();
+ }
+
+ function writeUint8Array(arr, callback, onerror) {
+ var tmpArray = new Uint8Array(array.length + arr.length);
+ tmpArray.set(array);
+ tmpArray.set(arr, array.length);
+ array = tmpArray;
+ callback();
+ }
+
+ function getData(callback) {
+ callback(array.buffer);
+ }
+
+ that.init = init;
+ that.writeUint8Array = writeUint8Array;
+ that.getData = getData;
+ }
+ ArrayBufferWriter.prototype = new Writer();
+ ArrayBufferWriter.prototype.constructor = ArrayBufferWriter;
+
+ function FileWriter(fileEntry, contentType) {
+ var writer, that = this;
+
+ function init(callback, onerror) {
+ fileEntry.createWriter(function(fileWriter) {
+ writer = fileWriter;
+ callback();
+ }, onerror);
+ }
+
+ function writeUint8Array(array, callback, onerror) {
+ var blob = new Blob([ appendABViewSupported ? array : array.buffer ], {
+ type : contentType
+ });
+ writer.onwrite = function() {
+ writer.onwrite = null;
+ callback();
+ };
+ writer.onerror = onerror;
+ writer.write(blob);
+ }
+
+ function getData(callback) {
+ fileEntry.file(callback);
+ }
+
+ that.init = init;
+ that.writeUint8Array = writeUint8Array;
+ that.getData = getData;
+ }
+ FileWriter.prototype = new Writer();
+ FileWriter.prototype.constructor = FileWriter;
+
+ zip.FileWriter = FileWriter;
+ zip.HttpReader = HttpReader;
+ zip.HttpRangeReader = HttpRangeReader;
+ zip.ArrayBufferReader = ArrayBufferReader;
+ zip.ArrayBufferWriter = ArrayBufferWriter;
+
+ if (zip.fs) {
+ ZipDirectoryEntry = zip.fs.ZipDirectoryEntry;
+ ZipDirectoryEntry.prototype.addHttpContent = function(name, URL, useRangeHeader) {
+ function addChild(parent, name, params, directory) {
+ if (parent.directory)
+ return directory ? new ZipDirectoryEntry(parent.fs, name, params, parent) : new zip.fs.ZipFileEntry(parent.fs, name, params, parent);
+ else
+ throw "Parent entry is not a directory.";
+ }
+
+ return addChild(this, name, {
+ data : URL,
+ Reader : useRangeHeader ? HttpRangeReader : HttpReader
+ });
+ };
+ ZipDirectoryEntry.prototype.importHttpContent = function(URL, useRangeHeader, onend, onerror) {
+ this.importZip(useRangeHeader ? new HttpRangeReader(URL) : new HttpReader(URL), onend, onerror);
+ };
+ zip.fs.FS.prototype.importHttpContent = function(URL, useRangeHeader, onend, onerror) {
+ this.entries = [];
+ this.root = new ZipDirectoryEntry(this);
+ this.root.importHttpContent(URL, useRangeHeader, onend, onerror);
+ };
+ }
+
+})();
diff --git a/lib/zip-fs.js b/lib/zip-fs.js
new file mode 100644
index 0000000..20b8003
--- /dev/null
+++ b/lib/zip-fs.js
@@ -0,0 +1,541 @@
+/*
+ Copyright (c) 2013 Gildas Lormeau. All rights reserved.
+
+ Redistribution and use in source and binary forms, with or without
+ modification, are permitted provided that the following conditions are met:
+
+ 1. Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+ 2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the distribution.
+
+ 3. The names of the authors may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+ THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+ INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+ INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+ INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+ OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+ LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+(function() {
+ "use strict";
+
+ var CHUNK_SIZE = 512 * 1024;
+
+ var TextWriter = zip.TextWriter, //
+ BlobWriter = zip.BlobWriter, //
+ Data64URIWriter = zip.Data64URIWriter, //
+ Reader = zip.Reader, //
+ TextReader = zip.TextReader, //
+ BlobReader = zip.BlobReader, //
+ Data64URIReader = zip.Data64URIReader, //
+ createReader = zip.createReader, //
+ createWriter = zip.createWriter;
+
+ function ZipBlobReader(entry) {
+ var that = this, blobReader;
+
+ function init(callback) {
+ that.size = entry.uncompressedSize;
+ callback();
+ }
+
+ function getData(callback) {
+ if (that.data)
+ callback();
+ else
+ entry.getData(new BlobWriter(), function(data) {
+ that.data = data;
+ blobReader = new BlobReader(data);
+ callback();
+ }, null, that.checkCrc32);
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ getData(function() {
+ blobReader.readUint8Array(index, length, callback, onerror);
+ }, onerror);
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ ZipBlobReader.prototype = new Reader();
+ ZipBlobReader.prototype.constructor = ZipBlobReader;
+ ZipBlobReader.prototype.checkCrc32 = false;
+
+ function getTotalSize(entry) {
+ var size = 0;
+
+ function process(entry) {
+ size += entry.uncompressedSize || 0;
+ entry.children.forEach(process);
+ }
+
+ process(entry);
+ return size;
+ }
+
+ function initReaders(entry, onend, onerror) {
+ var index = 0;
+
+ function next() {
+ index++;
+ if (index < entry.children.length)
+ process(entry.children[index]);
+ else
+ onend();
+ }
+
+ function process(child) {
+ if (child.directory)
+ initReaders(child, next, onerror);
+ else {
+ child.reader = new child.Reader(child.data, onerror);
+ child.reader.init(function() {
+ child.uncompressedSize = child.reader.size;
+ next();
+ });
+ }
+ }
+
+ if (entry.children.length)
+ process(entry.children[index]);
+ else
+ onend();
+ }
+
+ function detach(entry) {
+ var children = entry.parent.children;
+ children.forEach(function(child, index) {
+ if (child.id == entry.id)
+ children.splice(index, 1);
+ });
+ }
+
+ function exportZip(zipWriter, entry, onend, onprogress, totalSize) {
+ var currentIndex = 0;
+
+ function process(zipWriter, entry, onend, onprogress, totalSize) {
+ var childIndex = 0;
+
+ function exportChild() {
+ var child = entry.children[childIndex];
+ if (child)
+ zipWriter.add(child.getFullname(), child.reader, function() {
+ currentIndex += child.uncompressedSize || 0;
+ process(zipWriter, child, function() {
+ childIndex++;
+ exportChild();
+ }, onprogress, totalSize);
+ }, function(index) {
+ if (onprogress)
+ onprogress(currentIndex + index, totalSize);
+ }, {
+ directory : child.directory,
+ version : child.zipVersion
+ });
+ else
+ onend();
+ }
+
+ exportChild();
+ }
+
+ process(zipWriter, entry, onend, onprogress, totalSize);
+ }
+
+ function addFileEntry(zipEntry, fileEntry, onend, onerror) {
+ function getChildren(fileEntry, callback) {
+ if (fileEntry.isDirectory)
+ fileEntry.createReader().readEntries(callback);
+ if (fileEntry.isFile)
+ callback([]);
+ }
+
+ function process(zipEntry, fileEntry, onend) {
+ getChildren(fileEntry, function(children) {
+ var childIndex = 0;
+
+ function addChild(child) {
+ function nextChild(childFileEntry) {
+ process(childFileEntry, child, function() {
+ childIndex++;
+ processChild();
+ });
+ }
+
+ if (child.isDirectory)
+ nextChild(zipEntry.addDirectory(child.name));
+ if (child.isFile)
+ child.file(function(file) {
+ var childZipEntry = zipEntry.addBlob(child.name, file);
+ childZipEntry.uncompressedSize = file.size;
+ nextChild(childZipEntry);
+ }, onerror);
+ }
+
+ function processChild() {
+ var child = children[childIndex];
+ if (child)
+ addChild(child);
+ else
+ onend();
+ }
+
+ processChild();
+ });
+ }
+
+ if (fileEntry.isDirectory)
+ process(zipEntry, fileEntry, onend);
+ else
+ fileEntry.file(function(file) {
+ zipEntry.addBlob(fileEntry.name, file);
+ onend();
+ }, onerror);
+ }
+
+ function getFileEntry(fileEntry, entry, onend, onprogress, onerror, totalSize, checkCrc32) {
+ var currentIndex = 0;
+
+ function process(fileEntry, entry, onend, onprogress, onerror, totalSize) {
+ var childIndex = 0;
+
+ function addChild(child) {
+ function nextChild(childFileEntry) {
+ currentIndex += child.uncompressedSize || 0;
+ process(childFileEntry, child, function() {
+ childIndex++;
+ processChild();
+ }, onprogress, onerror, totalSize);
+ }
+
+ if (child.directory)
+ fileEntry.getDirectory(child.name, {
+ create : true
+ }, nextChild, onerror);
+ else
+ fileEntry.getFile(child.name, {
+ create : true
+ }, function(file) {
+ child.getData(new zip.FileWriter(file, zip.getMimeType(child.name)), nextChild, function(index) {
+ if (onprogress)
+ onprogress(currentIndex + index, totalSize);
+ }, checkCrc32);
+ }, onerror);
+ }
+
+ function processChild() {
+ var child = entry.children[childIndex];
+ if (child)
+ addChild(child);
+ else
+ onend();
+ }
+
+ processChild();
+ }
+
+ if (entry.directory)
+ process(fileEntry, entry, onend, onprogress, onerror, totalSize);
+ else
+ entry.getData(new zip.FileWriter(fileEntry, zip.getMimeType(entry.name)), onend, onprogress, checkCrc32);
+ }
+
+ function resetFS(fs) {
+ fs.entries = [];
+ fs.root = new ZipDirectoryEntry(fs);
+ }
+
+ function bufferedCopy(reader, writer, onend, onprogress, onerror) {
+ var chunkIndex = 0;
+
+ function stepCopy() {
+ var index = chunkIndex * CHUNK_SIZE;
+ if (onprogress)
+ onprogress(index, reader.size);
+ if (index < reader.size)
+ reader.readUint8Array(index, Math.min(CHUNK_SIZE, reader.size - index), function(array) {
+ writer.writeUint8Array(new Uint8Array(array), function() {
+ chunkIndex++;
+ stepCopy();
+ });
+ }, onerror);
+ else
+ writer.getData(onend);
+ }
+
+ stepCopy();
+ }
+
+ function addChild(parent, name, params, directory) {
+ if (parent.directory)
+ return directory ? new ZipDirectoryEntry(parent.fs, name, params, parent) : new ZipFileEntry(parent.fs, name, params, parent);
+ else
+ throw "Parent entry is not a directory.";
+ }
+
+ function ZipEntry() {
+ }
+
+ ZipEntry.prototype = {
+ init : function(fs, name, params, parent) {
+ var that = this;
+ if (fs.root && parent && parent.getChildByName(name))
+ throw "Entry filename already exists.";
+ if (!params)
+ params = {};
+ that.fs = fs;
+ that.name = name;
+ that.id = fs.entries.length;
+ that.parent = parent;
+ that.children = [];
+ that.zipVersion = params.zipVersion || 0x14;
+ that.uncompressedSize = 0;
+ fs.entries.push(that);
+ if (parent)
+ that.parent.children.push(that);
+ },
+ getFileEntry : function(fileEntry, onend, onprogress, onerror, checkCrc32) {
+ var that = this;
+ initReaders(that, function() {
+ getFileEntry(fileEntry, that, onend, onprogress, onerror, getTotalSize(that), checkCrc32);
+ }, onerror);
+ },
+ moveTo : function(target) {
+ var that = this;
+ if (target.directory) {
+ if (!target.isDescendantOf(that)) {
+ if (that != target) {
+ if (target.getChildByName(that.name))
+ throw "Entry filename already exists.";
+ detach(that);
+ that.parent = target;
+ target.children.push(that);
+ }
+ } else
+ throw "Entry is a ancestor of target entry.";
+ } else
+ throw "Target entry is not a directory.";
+ },
+ getFullname : function() {
+ var that = this, fullname = that.name, entry = that.parent;
+ while (entry) {
+ fullname = (entry.name ? entry.name + "/" : "") + fullname;
+ entry = entry.parent;
+ }
+ return fullname;
+ },
+ isDescendantOf : function(ancestor) {
+ var entry = this.parent;
+ while (entry && entry.id != ancestor.id)
+ entry = entry.parent;
+ return !!entry;
+ }
+ };
+ ZipEntry.prototype.constructor = ZipEntry;
+
+ var ZipFileEntryProto;
+
+ function ZipFileEntry(fs, name, params, parent) {
+ var that = this;
+ ZipEntry.prototype.init.call(that, fs, name, params, parent);
+ that.Reader = params.Reader;
+ that.Writer = params.Writer;
+ that.data = params.data;
+ if (params.getData) {
+ that.getData = params.getData;
+ }
+ }
+
+ ZipFileEntry.prototype = ZipFileEntryProto = new ZipEntry();
+ ZipFileEntryProto.constructor = ZipFileEntry;
+ ZipFileEntryProto.getData = function(writer, onend, onprogress, onerror) {
+ var that = this;
+ if (!writer || (writer.constructor == that.Writer && that.data))
+ onend(that.data);
+ else {
+ if (!that.reader)
+ that.reader = new that.Reader(that.data, onerror);
+ that.reader.init(function() {
+ writer.init(function() {
+ bufferedCopy(that.reader, writer, onend, onprogress, onerror);
+ }, onerror);
+ });
+ }
+ };
+
+ ZipFileEntryProto.getText = function(onend, onprogress, checkCrc32, encoding) {
+ this.getData(new TextWriter(encoding), onend, onprogress, checkCrc32);
+ };
+ ZipFileEntryProto.getBlob = function(mimeType, onend, onprogress, checkCrc32) {
+ this.getData(new BlobWriter(mimeType), onend, onprogress, checkCrc32);
+ };
+ ZipFileEntryProto.getData64URI = function(mimeType, onend, onprogress, checkCrc32) {
+ this.getData(new Data64URIWriter(mimeType), onend, onprogress, checkCrc32);
+ };
+
+ var ZipDirectoryEntryProto;
+
+ function ZipDirectoryEntry(fs, name, params, parent) {
+ var that = this;
+ ZipEntry.prototype.init.call(that, fs, name, params, parent);
+ that.directory = true;
+ }
+
+ ZipDirectoryEntry.prototype = ZipDirectoryEntryProto = new ZipEntry();
+ ZipDirectoryEntryProto.constructor = ZipDirectoryEntry;
+ ZipDirectoryEntryProto.addDirectory = function(name) {
+ return addChild(this, name, null, true);
+ };
+ ZipDirectoryEntryProto.addText = function(name, text) {
+ return addChild(this, name, {
+ data : text,
+ Reader : TextReader,
+ Writer : TextWriter
+ });
+ };
+ ZipDirectoryEntryProto.addBlob = function(name, blob) {
+ return addChild(this, name, {
+ data : blob,
+ Reader : BlobReader,
+ Writer : BlobWriter
+ });
+ };
+ ZipDirectoryEntryProto.addData64URI = function(name, dataURI) {
+ return addChild(this, name, {
+ data : dataURI,
+ Reader : Data64URIReader,
+ Writer : Data64URIWriter
+ });
+ };
+ ZipDirectoryEntryProto.addFileEntry = function(fileEntry, onend, onerror) {
+ addFileEntry(this, fileEntry, onend, onerror);
+ };
+ ZipDirectoryEntryProto.addData = function(name, params) {
+ return addChild(this, name, params);
+ };
+ ZipDirectoryEntryProto.importBlob = function(blob, onend, onerror) {
+ this.importZip(new BlobReader(blob), onend, onerror);
+ };
+ ZipDirectoryEntryProto.importText = function(text, onend, onerror) {
+ this.importZip(new TextReader(text), onend, onerror);
+ };
+ ZipDirectoryEntryProto.importData64URI = function(dataURI, onend, onerror) {
+ this.importZip(new Data64URIReader(dataURI), onend, onerror);
+ };
+ ZipDirectoryEntryProto.exportBlob = function(onend, onprogress, onerror) {
+ this.exportZip(new BlobWriter("application/zip"), onend, onprogress, onerror);
+ };
+ ZipDirectoryEntryProto.exportText = function(onend, onprogress, onerror) {
+ this.exportZip(new TextWriter(), onend, onprogress, onerror);
+ };
+ ZipDirectoryEntryProto.exportFileEntry = function(fileEntry, onend, onprogress, onerror) {
+ this.exportZip(new zip.FileWriter(fileEntry, "application/zip"), onend, onprogress, onerror);
+ };
+ ZipDirectoryEntryProto.exportData64URI = function(onend, onprogress, onerror) {
+ this.exportZip(new Data64URIWriter("application/zip"), onend, onprogress, onerror);
+ };
+ ZipDirectoryEntryProto.importZip = function(reader, onend, onerror) {
+ var that = this;
+ createReader(reader, function(zipReader) {
+ zipReader.getEntries(function(entries) {
+ entries.forEach(function(entry) {
+ var parent = that, path = entry.filename.split("/"), name = path.pop();
+ path.forEach(function(pathPart) {
+ parent = parent.getChildByName(pathPart) || new ZipDirectoryEntry(that.fs, pathPart, null, parent);
+ });
+ if (!entry.directory)
+ addChild(parent, name, {
+ data : entry,
+ Reader : ZipBlobReader
+ });
+ });
+ onend();
+ });
+ }, onerror);
+ };
+ ZipDirectoryEntryProto.exportZip = function(writer, onend, onprogress, onerror) {
+ var that = this;
+ initReaders(that, function() {
+ createWriter(writer, function(zipWriter) {
+ exportZip(zipWriter, that, function() {
+ zipWriter.close(onend);
+ }, onprogress, getTotalSize(that));
+ }, onerror);
+ }, onerror);
+ };
+ ZipDirectoryEntryProto.getChildByName = function(name) {
+ var childIndex, child, that = this;
+ for (childIndex = 0; childIndex < that.children.length; childIndex++) {
+ child = that.children[childIndex];
+ if (child.name == name)
+ return child;
+ }
+ };
+
+ function FS() {
+ resetFS(this);
+ }
+ FS.prototype = {
+ remove : function(entry) {
+ detach(entry);
+ this.entries[entry.id] = null;
+ },
+ find : function(fullname) {
+ var index, path = fullname.split("/"), node = this.root;
+ for (index = 0; node && index < path.length; index++)
+ node = node.getChildByName(path[index]);
+ return node;
+ },
+ getById : function(id) {
+ return this.entries[id];
+ },
+ importBlob : function(blob, onend, onerror) {
+ resetFS(this);
+ this.root.importBlob(blob, onend, onerror);
+ },
+ importText : function(text, onend, onerror) {
+ resetFS(this);
+ this.root.importText(text, onend, onerror);
+ },
+ importData64URI : function(dataURI, onend, onerror) {
+ resetFS(this);
+ this.root.importData64URI(dataURI, onend, onerror);
+ },
+ exportBlob : function(onend, onprogress, onerror) {
+ this.root.exportBlob(onend, onprogress, onerror);
+ },
+ exportText : function(onend, onprogress, onerror) {
+ this.root.exportText(onend, onprogress, onerror);
+ },
+ exportFileEntry : function(fileEntry, onend, onprogress, onerror) {
+ this.root.exportFileEntry(fileEntry, onend, onprogress, onerror);
+ },
+ exportData64URI : function(onend, onprogress, onerror) {
+ this.root.exportData64URI(onend, onprogress, onerror);
+ }
+ };
+
+ zip.fs = {
+ FS : FS,
+ ZipDirectoryEntry : ZipDirectoryEntry,
+ ZipFileEntry : ZipFileEntry
+ };
+
+ zip.getMimeType = function() {
+ return "application/octet-stream";
+ };
+
+})();
diff --git a/lib/zip.js b/lib/zip.js
new file mode 100644
index 0000000..4f6f67e
--- /dev/null
+++ b/lib/zip.js
@@ -0,0 +1,966 @@
+/*
+ Copyright (c) 2013 Gildas Lormeau. All rights reserved.
+
+ Redistribution and use in source and binary forms, with or without
+ modification, are permitted provided that the following conditions are met:
+
+ 1. Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+ 2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the distribution.
+
+ 3. The names of the authors may not be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+ THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+ INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+ INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+ INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+ OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+ LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+(function(obj) {
+ "use strict";
+
+ var ERR_BAD_FORMAT = "File format is not recognized.";
+ var ERR_CRC = "CRC failed.";
+ var ERR_ENCRYPTED = "File contains encrypted entry.";
+ var ERR_ZIP64 = "File is using Zip64 (4gb+ file size).";
+ var ERR_READ = "Error while reading zip file.";
+ var ERR_WRITE = "Error while writing zip file.";
+ var ERR_WRITE_DATA = "Error while writing file data.";
+ var ERR_READ_DATA = "Error while reading file data.";
+ var ERR_DUPLICATED_NAME = "File already exists.";
+ var CHUNK_SIZE = 512 * 1024;
+
+ var TEXT_PLAIN = "text/plain";
+
+ var appendABViewSupported;
+ try {
+ appendABViewSupported = new Blob([ new DataView(new ArrayBuffer(0)) ]).size === 0;
+ } catch (e) {
+ }
+
+ function Crc32() {
+ this.crc = -1;
+ }
+ Crc32.prototype.append = function append(data) {
+ var crc = this.crc | 0, table = this.table;
+ for (var offset = 0, len = data.length | 0; offset < len; offset++)
+ crc = (crc >>> 8) ^ table[(crc ^ data[offset]) & 0xFF];
+ this.crc = crc;
+ };
+ Crc32.prototype.get = function get() {
+ return ~this.crc;
+ };
+ Crc32.prototype.table = (function() {
+ var i, j, t, table = []; // Uint32Array is actually slower than []
+ for (i = 0; i < 256; i++) {
+ t = i;
+ for (j = 0; j < 8; j++)
+ if (t & 1)
+ t = (t >>> 1) ^ 0xEDB88320;
+ else
+ t = t >>> 1;
+ table[i] = t;
+ }
+ return table;
+ })();
+
+ // "no-op" codec
+ function NOOP() {}
+ NOOP.prototype.append = function append(bytes, onprogress) {
+ return bytes;
+ };
+ NOOP.prototype.flush = function flush() {};
+
+ function blobSlice(blob, index, length) {
+ if (index < 0 || length < 0 || index + length > blob.size)
+ throw new RangeError('offset:' + index + ', length:' + length + ', size:' + blob.size);
+ if (blob.slice)
+ return blob.slice(index, index + length);
+ else if (blob.webkitSlice)
+ return blob.webkitSlice(index, index + length);
+ else if (blob.mozSlice)
+ return blob.mozSlice(index, index + length);
+ else if (blob.msSlice)
+ return blob.msSlice(index, index + length);
+ }
+
+ function getDataHelper(byteLength, bytes) {
+ var dataBuffer, dataArray;
+ dataBuffer = new ArrayBuffer(byteLength);
+ dataArray = new Uint8Array(dataBuffer);
+ if (bytes)
+ dataArray.set(bytes, 0);
+ return {
+ buffer : dataBuffer,
+ array : dataArray,
+ view : new DataView(dataBuffer)
+ };
+ }
+
+ // Readers
+ function Reader() {
+ }
+
+ function TextReader(text) {
+ var that = this, blobReader;
+
+ function init(callback, onerror) {
+ var blob = new Blob([ text ], {
+ type : TEXT_PLAIN
+ });
+ blobReader = new BlobReader(blob);
+ blobReader.init(function() {
+ that.size = blobReader.size;
+ callback();
+ }, onerror);
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ blobReader.readUint8Array(index, length, callback, onerror);
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ TextReader.prototype = new Reader();
+ TextReader.prototype.constructor = TextReader;
+
+ function Data64URIReader(dataURI) {
+ var that = this, dataStart;
+
+ function init(callback) {
+ var dataEnd = dataURI.length;
+ while (dataURI.charAt(dataEnd - 1) == "=")
+ dataEnd--;
+ dataStart = dataURI.indexOf(",") + 1;
+ that.size = Math.floor((dataEnd - dataStart) * 0.75);
+ callback();
+ }
+
+ function readUint8Array(index, length, callback) {
+ var i, data = getDataHelper(length);
+ var start = Math.floor(index / 3) * 4;
+ var end = Math.ceil((index + length) / 3) * 4;
+ var bytes = obj.atob(dataURI.substring(start + dataStart, end + dataStart));
+ var delta = index - Math.floor(start / 4) * 3;
+ for (i = delta; i < delta + length; i++)
+ data.array[i - delta] = bytes.charCodeAt(i);
+ callback(data.array);
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ Data64URIReader.prototype = new Reader();
+ Data64URIReader.prototype.constructor = Data64URIReader;
+
+ function BlobReader(blob) {
+ var that = this;
+
+ function init(callback) {
+ that.size = blob.size;
+ callback();
+ }
+
+ function readUint8Array(index, length, callback, onerror) {
+ var reader = new FileReader();
+ reader.onload = function(e) {
+ callback(new Uint8Array(e.target.result));
+ };
+ reader.onerror = onerror;
+ try {
+ reader.readAsArrayBuffer(blobSlice(blob, index, length));
+ } catch (e) {
+ onerror(e);
+ }
+ }
+
+ that.size = 0;
+ that.init = init;
+ that.readUint8Array = readUint8Array;
+ }
+ BlobReader.prototype = new Reader();
+ BlobReader.prototype.constructor = BlobReader;
+
+ // Writers
+
+ function Writer() {
+ }
+ Writer.prototype.getData = function(callback) {
+ callback(this.data);
+ };
+
+ function TextWriter(encoding) {
+ var that = this, blob;
+
+ function init(callback) {
+ blob = new Blob([], {
+ type : TEXT_PLAIN
+ });
+ callback();
+ }
+
+ function writeUint8Array(array, callback) {
+ blob = new Blob([ blob, appendABViewSupported ? array : array.buffer ], {
+ type : TEXT_PLAIN
+ });
+ callback();
+ }
+
+ function getData(callback, onerror) {
+ var reader = new FileReader();
+ reader.onload = function(e) {
+ callback(e.target.result);
+ };
+ reader.onerror = onerror;
+ reader.readAsText(blob, encoding);
+ }
+
+ that.init = init;
+ that.writeUint8Array = writeUint8Array;
+ that.getData = getData;
+ }
+ TextWriter.prototype = new Writer();
+ TextWriter.prototype.constructor = TextWriter;
+
+ function Data64URIWriter(contentType) {
+ var that = this, data = "", pending = "";
+
+ function init(callback) {
+ data += "data:" + (contentType || "") + ";base64,";
+ callback();
+ }
+
+ function writeUint8Array(array, callback) {
+ var i, delta = pending.length, dataString = pending;
+ pending = "";
+ for (i = 0; i < (Math.floor((delta + array.length) / 3) * 3) - delta; i++)
+ dataString += String.fromCharCode(array[i]);
+ for (; i < array.length; i++)
+ pending += String.fromCharCode(array[i]);
+ if (dataString.length > 2)
+ data += obj.btoa(dataString);
+ else
+ pending = dataString;
+ callback();
+ }
+
+ function getData(callback) {
+ callback(data + obj.btoa(pending));
+ }
+
+ that.init = init;
+ that.writeUint8Array = writeUint8Array;
+ that.getData = getData;
+ }
+ Data64URIWriter.prototype = new Writer();
+ Data64URIWriter.prototype.constructor = Data64URIWriter;
+
+ function BlobWriter(contentType) {
+ var blob, that = this;
+
+ function init(callback) {
+ blob = new Blob([], {
+ type : contentType
+ });
+ callback();
+ }
+
+ function writeUint8Array(array, callback) {
+ blob = new Blob([ blob, appendABViewSupported ? array : array.buffer ], {
+ type : contentType
+ });
+ callback();
+ }
+
+ function getData(callback) {
+ callback(blob);
+ }
+
+ that.init = init;
+ that.writeUint8Array = writeUint8Array;
+ that.getData = getData;
+ }
+ BlobWriter.prototype = new Writer();
+ BlobWriter.prototype.constructor = BlobWriter;
+
+ /**
+ * inflate/deflate core functions
+ * @param worker {Worker} web worker for the task.
+ * @param initialMessage {Object} initial message to be sent to the worker. should contain
+ * sn(serial number for distinguishing multiple tasks sent to the worker), and codecClass.
+ * This function may add more properties before sending.
+ */
+ function launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror) {
+ var chunkIndex = 0, index, outputSize, sn = initialMessage.sn, crc;
+
+ function onflush() {
+ worker.removeEventListener('message', onmessage, false);
+ onend(outputSize, crc);
+ }
+
+ function onmessage(event) {
+ var message = event.data, data = message.data, err = message.error;
+ if (err) {
+ err.toString = function () { return 'Error: ' + this.message; };
+ onreaderror(err);
+ return;
+ }
+ if (message.sn !== sn)
+ return;
+ if (typeof message.codecTime === 'number')
+ worker.codecTime += message.codecTime; // should be before onflush()
+ if (typeof message.crcTime === 'number')
+ worker.crcTime += message.crcTime;
+
+ switch (message.type) {
+ case 'append':
+ if (data) {
+ outputSize += data.length;
+ writer.writeUint8Array(data, function() {
+ step();
+ }, onwriteerror);
+ } else
+ step();
+ break;
+ case 'flush':
+ crc = message.crc;
+ if (data) {
+ outputSize += data.length;
+ writer.writeUint8Array(data, function() {
+ onflush();
+ }, onwriteerror);
+ } else
+ onflush();
+ break;
+ case 'progress':
+ if (onprogress)
+ onprogress(index + message.loaded, size);
+ break;
+ case 'importScripts': //no need to handle here
+ case 'newTask':
+ case 'echo':
+ break;
+ default:
+ console.warn('zip.js:launchWorkerProcess: unknown message: ', message);
+ }
+ }
+
+ function step() {
+ index = chunkIndex * CHUNK_SIZE;
+ // use `<=` instead of `<`, because `size` may be 0.
+ if (index <= size) {
+ reader.readUint8Array(offset + index, Math.min(CHUNK_SIZE, size - index), function(array) {
+ if (onprogress)
+ onprogress(index, size);
+ var msg = index === 0 ? initialMessage : {sn : sn};
+ msg.type = 'append';
+ msg.data = array;
+
+ // posting a message with transferables will fail on IE10
+ try {
+ worker.postMessage(msg, [array.buffer]);
+ } catch(ex) {
+ worker.postMessage(msg); // retry without transferables
+ }
+ chunkIndex++;
+ }, onreaderror);
+ } else {
+ worker.postMessage({
+ sn: sn,
+ type: 'flush'
+ });
+ }
+ }
+
+ outputSize = 0;
+ worker.addEventListener('message', onmessage, false);
+ step();
+ }
+
+ function launchProcess(process, reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror) {
+ var chunkIndex = 0, index, outputSize = 0,
+ crcInput = crcType === 'input',
+ crcOutput = crcType === 'output',
+ crc = new Crc32();
+ function step() {
+ var outputData;
+ index = chunkIndex * CHUNK_SIZE;
+ if (index < size)
+ reader.readUint8Array(offset + index, Math.min(CHUNK_SIZE, size - index), function(inputData) {
+ var outputData;
+ try {
+ outputData = process.append(inputData, function(loaded) {
+ if (onprogress)
+ onprogress(index + loaded, size);
+ });
+ } catch (e) {
+ onreaderror(e);
+ return;
+ }
+ if (outputData) {
+ outputSize += outputData.length;
+ writer.writeUint8Array(outputData, function() {
+ chunkIndex++;
+ setTimeout(step, 1);
+ }, onwriteerror);
+ if (crcOutput)
+ crc.append(outputData);
+ } else {
+ chunkIndex++;
+ setTimeout(step, 1);
+ }
+ if (crcInput)
+ crc.append(inputData);
+ if (onprogress)
+ onprogress(index, size);
+ }, onreaderror);
+ else {
+ try {
+ outputData = process.flush();
+ } catch (e) {
+ onreaderror(e);
+ return;
+ }
+ if (outputData) {
+ if (crcOutput)
+ crc.append(outputData);
+ outputSize += outputData.length;
+ writer.writeUint8Array(outputData, function() {
+ onend(outputSize, crc.get());
+ }, onwriteerror);
+ } else
+ onend(outputSize, crc.get());
+ }
+ }
+
+ step();
+ }
+
+ function inflate(worker, sn, reader, writer, offset, size, computeCrc32, onend, onprogress, onreaderror, onwriteerror) {
+ var crcType = computeCrc32 ? 'output' : 'none';
+ if (obj.zip.useWebWorkers) {
+ var initialMessage = {
+ sn: sn,
+ codecClass: 'Inflater',
+ crcType: crcType,
+ };
+ launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror);
+ } else
+ launchProcess(new obj.zip.Inflater(), reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror);
+ }
+
+ function deflate(worker, sn, reader, writer, level, onend, onprogress, onreaderror, onwriteerror) {
+ var crcType = 'input';
+ if (obj.zip.useWebWorkers) {
+ var initialMessage = {
+ sn: sn,
+ options: {level: level},
+ codecClass: 'Deflater',
+ crcType: crcType,
+ };
+ launchWorkerProcess(worker, initialMessage, reader, writer, 0, reader.size, onprogress, onend, onreaderror, onwriteerror);
+ } else
+ launchProcess(new obj.zip.Deflater(), reader, writer, 0, reader.size, crcType, onprogress, onend, onreaderror, onwriteerror);
+ }
+
+ function copy(worker, sn, reader, writer, offset, size, computeCrc32, onend, onprogress, onreaderror, onwriteerror) {
+ var crcType = 'input';
+ if (obj.zip.useWebWorkers && computeCrc32) {
+ var initialMessage = {
+ sn: sn,
+ codecClass: 'NOOP',
+ crcType: crcType,
+ };
+ launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror);
+ } else
+ launchProcess(new NOOP(), reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror);
+ }
+
+ // ZipReader
+
+ function decodeASCII(str) {
+ var i, out = "", charCode, extendedASCII = [ '\u00C7', '\u00FC', '\u00E9', '\u00E2', '\u00E4', '\u00E0', '\u00E5', '\u00E7', '\u00EA', '\u00EB',
+ '\u00E8', '\u00EF', '\u00EE', '\u00EC', '\u00C4', '\u00C5', '\u00C9', '\u00E6', '\u00C6', '\u00F4', '\u00F6', '\u00F2', '\u00FB', '\u00F9',
+ '\u00FF', '\u00D6', '\u00DC', '\u00F8', '\u00A3', '\u00D8', '\u00D7', '\u0192', '\u00E1', '\u00ED', '\u00F3', '\u00FA', '\u00F1', '\u00D1',
+ '\u00AA', '\u00BA', '\u00BF', '\u00AE', '\u00AC', '\u00BD', '\u00BC', '\u00A1', '\u00AB', '\u00BB', '_', '_', '_', '\u00A6', '\u00A6',
+ '\u00C1', '\u00C2', '\u00C0', '\u00A9', '\u00A6', '\u00A6', '+', '+', '\u00A2', '\u00A5', '+', '+', '-', '-', '+', '-', '+', '\u00E3',
+ '\u00C3', '+', '+', '-', '-', '\u00A6', '-', '+', '\u00A4', '\u00F0', '\u00D0', '\u00CA', '\u00CB', '\u00C8', 'i', '\u00CD', '\u00CE',
+ '\u00CF', '+', '+', '_', '_', '\u00A6', '\u00CC', '_', '\u00D3', '\u00DF', '\u00D4', '\u00D2', '\u00F5', '\u00D5', '\u00B5', '\u00FE',
+ '\u00DE', '\u00DA', '\u00DB', '\u00D9', '\u00FD', '\u00DD', '\u00AF', '\u00B4', '\u00AD', '\u00B1', '_', '\u00BE', '\u00B6', '\u00A7',
+ '\u00F7', '\u00B8', '\u00B0', '\u00A8', '\u00B7', '\u00B9', '\u00B3', '\u00B2', '_', ' ' ];
+ for (i = 0; i < str.length; i++) {
+ charCode = str.charCodeAt(i) & 0xFF;
+ if (charCode > 127)
+ out += extendedASCII[charCode - 128];
+ else
+ out += String.fromCharCode(charCode);
+ }
+ return out;
+ }
+
+ function decodeUTF8(string) {
+ return decodeURIComponent(escape(string));
+ }
+
+ function getString(bytes) {
+ var i, str = "";
+ for (i = 0; i < bytes.length; i++)
+ str += String.fromCharCode(bytes[i]);
+ return str;
+ }
+
+ function getDate(timeRaw) {
+ var date = (timeRaw & 0xffff0000) >> 16, time = timeRaw & 0x0000ffff;
+ try {
+ return new Date(1980 + ((date & 0xFE00) >> 9), ((date & 0x01E0) >> 5) - 1, date & 0x001F, (time & 0xF800) >> 11, (time & 0x07E0) >> 5,
+ (time & 0x001F) * 2, 0);
+ } catch (e) {
+ }
+ }
+
+ function readCommonHeader(entry, data, index, centralDirectory, onerror) {
+ entry.version = data.view.getUint16(index, true);
+ entry.bitFlag = data.view.getUint16(index + 2, true);
+ entry.compressionMethod = data.view.getUint16(index + 4, true);
+ entry.lastModDateRaw = data.view.getUint32(index + 6, true);
+ entry.lastModDate = getDate(entry.lastModDateRaw);
+ if ((entry.bitFlag & 0x01) === 0x01) {
+ onerror(ERR_ENCRYPTED);
+ return;
+ }
+ if (centralDirectory || (entry.bitFlag & 0x0008) != 0x0008) {
+ entry.crc32 = data.view.getUint32(index + 10, true);
+ entry.compressedSize = data.view.getUint32(index + 14, true);
+ entry.uncompressedSize = data.view.getUint32(index + 18, true);
+ }
+ if (entry.compressedSize === 0xFFFFFFFF || entry.uncompressedSize === 0xFFFFFFFF) {
+ onerror(ERR_ZIP64);
+ return;
+ }
+ entry.filenameLength = data.view.getUint16(index + 22, true);
+ entry.extraFieldLength = data.view.getUint16(index + 24, true);
+ }
+
+ function createZipReader(reader, callback, onerror) {
+ var inflateSN = 0;
+
+ function Entry() {
+ }
+
+ Entry.prototype.getData = function(writer, onend, onprogress, checkCrc32) {
+ var that = this;
+
+ function testCrc32(crc32) {
+ var dataCrc32 = getDataHelper(4);
+ dataCrc32.view.setUint32(0, crc32);
+ return that.crc32 == dataCrc32.view.getUint32(0);
+ }
+
+ function getWriterData(uncompressedSize, crc32) {
+ if (checkCrc32 && !testCrc32(crc32))
+ onerror(ERR_CRC);
+ else
+ writer.getData(function(data) {
+ onend(data);
+ });
+ }
+
+ function onreaderror(err) {
+ onerror(err || ERR_READ_DATA);
+ }
+
+ function onwriteerror(err) {
+ onerror(err || ERR_WRITE_DATA);
+ }
+
+ reader.readUint8Array(that.offset, 30, function(bytes) {
+ var data = getDataHelper(bytes.length, bytes), dataOffset;
+ if (data.view.getUint32(0) != 0x504b0304) {
+ onerror(ERR_BAD_FORMAT);
+ return;
+ }
+ readCommonHeader(that, data, 4, false, onerror);
+ dataOffset = that.offset + 30 + that.filenameLength + that.extraFieldLength;
+ writer.init(function() {
+ if (that.compressionMethod === 0)
+ copy(that._worker, inflateSN++, reader, writer, dataOffset, that.compressedSize, checkCrc32, getWriterData, onprogress, onreaderror, onwriteerror);
+ else
+ inflate(that._worker, inflateSN++, reader, writer, dataOffset, that.compressedSize, checkCrc32, getWriterData, onprogress, onreaderror, onwriteerror);
+ }, onwriteerror);
+ }, onreaderror);
+ };
+
+ function seekEOCDR(eocdrCallback) {
+ // "End of central directory record" is the last part of a zip archive, and is at least 22 bytes long.
+ // Zip file comment is the last part of EOCDR and has max length of 64KB,
+ // so we only have to search the last 64K + 22 bytes of a archive for EOCDR signature (0x06054b50).
+ var EOCDR_MIN = 22;
+ if (reader.size < EOCDR_MIN) {
+ onerror(ERR_BAD_FORMAT);
+ return;
+ }
+ var ZIP_COMMENT_MAX = 256 * 256, EOCDR_MAX = EOCDR_MIN + ZIP_COMMENT_MAX;
+
+ // In most cases, the EOCDR is EOCDR_MIN bytes long
+ doSeek(EOCDR_MIN, function() {
+ // If not found, try within EOCDR_MAX bytes
+ doSeek(Math.min(EOCDR_MAX, reader.size), function() {
+ onerror(ERR_BAD_FORMAT);
+ });
+ });
+
+ // seek last length bytes of file for EOCDR
+ function doSeek(length, eocdrNotFoundCallback) {
+ reader.readUint8Array(reader.size - length, length, function(bytes) {
+ for (var i = bytes.length - EOCDR_MIN; i >= 0; i--) {
+ if (bytes[i] === 0x50 && bytes[i + 1] === 0x4b && bytes[i + 2] === 0x05 && bytes[i + 3] === 0x06) {
+ eocdrCallback(new DataView(bytes.buffer, i, EOCDR_MIN));
+ return;
+ }
+ }
+ eocdrNotFoundCallback();
+ }, function() {
+ onerror(ERR_READ);
+ });
+ }
+ }
+
+ var zipReader = {
+ getEntries : function(callback) {
+ var worker = this._worker;
+ // look for End of central directory record
+ seekEOCDR(function(dataView) {
+ var datalength, fileslength;
+ datalength = dataView.getUint32(16, true);
+ fileslength = dataView.getUint16(8, true);
+ if (datalength < 0 || datalength >= reader.size) {
+ onerror(ERR_BAD_FORMAT);
+ return;
+ }
+ reader.readUint8Array(datalength, reader.size - datalength, function(bytes) {
+ var i, index = 0, entries = [], entry, filename, comment, data = getDataHelper(bytes.length, bytes);
+ for (i = 0; i < fileslength; i++) {
+ entry = new Entry();
+ entry._worker = worker;
+ if (data.view.getUint32(index) != 0x504b0102) {
+ onerror(ERR_BAD_FORMAT);
+ return;
+ }
+ readCommonHeader(entry, data, index + 6, true, onerror);
+ entry.commentLength = data.view.getUint16(index + 32, true);
+ entry.directory = ((data.view.getUint8(index + 38) & 0x10) == 0x10);
+ entry.offset = data.view.getUint32(index + 42, true);
+ filename = getString(data.array.subarray(index + 46, index + 46 + entry.filenameLength));
+ entry.filename = ((entry.bitFlag & 0x0800) === 0x0800) ? decodeUTF8(filename) : decodeASCII(filename);
+ if (!entry.directory && entry.filename.charAt(entry.filename.length - 1) == "/")
+ entry.directory = true;
+ comment = getString(data.array.subarray(index + 46 + entry.filenameLength + entry.extraFieldLength, index + 46
+ + entry.filenameLength + entry.extraFieldLength + entry.commentLength));
+ entry.comment = ((entry.bitFlag & 0x0800) === 0x0800) ? decodeUTF8(comment) : decodeASCII(comment);
+ entries.push(entry);
+ index += 46 + entry.filenameLength + entry.extraFieldLength + entry.commentLength;
+ }
+ callback(entries);
+ }, function() {
+ onerror(ERR_READ);
+ });
+ });
+ },
+ close : function(callback) {
+ if (this._worker) {
+ this._worker.terminate();
+ this._worker = null;
+ }
+ if (callback)
+ callback();
+ },
+ _worker: null
+ };
+
+ if (!obj.zip.useWebWorkers)
+ callback(zipReader);
+ else {
+ createWorker('inflater',
+ function(worker) {
+ zipReader._worker = worker;
+ callback(zipReader);
+ },
+ function(err) {
+ onerror(err);
+ }
+ );
+ }
+ }
+
+ // ZipWriter
+
+ function encodeUTF8(string) {
+ return unescape(encodeURIComponent(string));
+ }
+
+ function getBytes(str) {
+ var i, array = [];
+ for (i = 0; i < str.length; i++)
+ array.push(str.charCodeAt(i));
+ return array;
+ }
+
+ function createZipWriter(writer, callback, onerror, dontDeflate) {
+ var files = {}, filenames = [], datalength = 0;
+ var deflateSN = 0;
+
+ function onwriteerror(err) {
+ onerror(err || ERR_WRITE);
+ }
+
+ function onreaderror(err) {
+ onerror(err || ERR_READ_DATA);
+ }
+
+ var zipWriter = {
+ add : function(name, reader, onend, onprogress, options) {
+ var header, filename, date;
+ var worker = this._worker;
+
+ function writeHeader(callback) {
+ var data;
+ date = options.lastModDate || new Date();
+ header = getDataHelper(26);
+ files[name] = {
+ headerArray : header.array,
+ directory : options.directory,
+ filename : filename,
+ offset : datalength,
+ comment : getBytes(encodeUTF8(options.comment || ""))
+ };
+ header.view.setUint32(0, 0x14000808);
+ if (options.version)
+ header.view.setUint8(0, options.version);
+ if (!dontDeflate && options.level !== 0 && !options.directory)
+ header.view.setUint16(4, 0x0800);
+ header.view.setUint16(6, (((date.getHours() << 6) | date.getMinutes()) << 5) | date.getSeconds() / 2, true);
+ header.view.setUint16(8, ((((date.getFullYear() - 1980) << 4) | (date.getMonth() + 1)) << 5) | date.getDate(), true);
+ header.view.setUint16(22, filename.length, true);
+ data = getDataHelper(30 + filename.length);
+ data.view.setUint32(0, 0x504b0304);
+ data.array.set(header.array, 4);
+ data.array.set(filename, 30);
+ datalength += data.array.length;
+ writer.writeUint8Array(data.array, callback, onwriteerror);
+ }
+
+ function writeFooter(compressedLength, crc32) {
+ var footer = getDataHelper(16);
+ datalength += compressedLength || 0;
+ footer.view.setUint32(0, 0x504b0708);
+ if (typeof crc32 != "undefined") {
+ header.view.setUint32(10, crc32, true);
+ footer.view.setUint32(4, crc32, true);
+ }
+ if (reader) {
+ footer.view.setUint32(8, compressedLength, true);
+ header.view.setUint32(14, compressedLength, true);
+ footer.view.setUint32(12, reader.size, true);
+ header.view.setUint32(18, reader.size, true);
+ }
+ writer.writeUint8Array(footer.array, function() {
+ datalength += 16;
+ onend();
+ }, onwriteerror);
+ }
+
+ function writeFile() {
+ options = options || {};
+ name = name.trim();
+ if (options.directory && name.charAt(name.length - 1) != "/")
+ name += "/";
+ if (files.hasOwnProperty(name)) {
+ onerror(ERR_DUPLICATED_NAME);
+ return;
+ }
+ filename = getBytes(encodeUTF8(name));
+ filenames.push(name);
+ writeHeader(function() {
+ if (reader)
+ if (dontDeflate || options.level === 0)
+ copy(worker, deflateSN++, reader, writer, 0, reader.size, true, writeFooter, onprogress, onreaderror, onwriteerror);
+ else
+ deflate(worker, deflateSN++, reader, writer, options.level, writeFooter, onprogress, onreaderror, onwriteerror);
+ else
+ writeFooter();
+ }, onwriteerror);
+ }
+
+ if (reader)
+ reader.init(writeFile, onreaderror);
+ else
+ writeFile();
+ },
+ close : function(callback) {
+ if (this._worker) {
+ this._worker.terminate();
+ this._worker = null;
+ }
+
+ var data, length = 0, index = 0, indexFilename, file;
+ for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
+ file = files[filenames[indexFilename]];
+ length += 46 + file.filename.length + file.comment.length;
+ }
+ data = getDataHelper(length + 22);
+ for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
+ file = files[filenames[indexFilename]];
+ data.view.setUint32(index, 0x504b0102);
+ data.view.setUint16(index + 4, 0x1400);
+ data.array.set(file.headerArray, index + 6);
+ data.view.setUint16(index + 32, file.comment.length, true);
+ if (file.directory)
+ data.view.setUint8(index + 38, 0x10);
+ data.view.setUint32(index + 42, file.offset, true);
+ data.array.set(file.filename, index + 46);
+ data.array.set(file.comment, index + 46 + file.filename.length);
+ index += 46 + file.filename.length + file.comment.length;
+ }
+ data.view.setUint32(index, 0x504b0506);
+ data.view.setUint16(index + 8, filenames.length, true);
+ data.view.setUint16(index + 10, filenames.length, true);
+ data.view.setUint32(index + 12, length, true);
+ data.view.setUint32(index + 16, datalength, true);
+ writer.writeUint8Array(data.array, function() {
+ writer.getData(callback);
+ }, onwriteerror);
+ },
+ _worker: null
+ };
+
+ if (!obj.zip.useWebWorkers)
+ callback(zipWriter);
+ else {
+ createWorker('deflater',
+ function(worker) {
+ zipWriter._worker = worker;
+ callback(zipWriter);
+ },
+ function(err) {
+ onerror(err);
+ }
+ );
+ }
+ }
+
+ function resolveURLs(urls) {
+ var a = document.createElement('a');
+ return urls.map(function(url) {
+ a.href = url;
+ return a.href;
+ });
+ }
+
+ var DEFAULT_WORKER_SCRIPTS = {
+ deflater: ['z-worker.js', 'deflate.js'],
+ inflater: ['z-worker.js', 'inflate.js']
+ };
+ function createWorker(type, callback, onerror) {
+ if (obj.zip.workerScripts !== null && obj.zip.workerScriptsPath !== null) {
+ onerror(new Error('Either zip.workerScripts or zip.workerScriptsPath may be set, not both.'));
+ return;
+ }
+ var scripts;
+ if (obj.zip.workerScripts) {
+ scripts = obj.zip.workerScripts[type];
+ if (!Array.isArray(scripts)) {
+ onerror(new Error('zip.workerScripts.' + type + ' is not an array!'));
+ return;
+ }
+ scripts = resolveURLs(scripts);
+ } else {
+ scripts = DEFAULT_WORKER_SCRIPTS[type].slice(0);
+ scripts[0] = (obj.zip.workerScriptsPath || '') + scripts[0];
+ }
+ var worker = new Worker(scripts[0]);
+ // record total consumed time by inflater/deflater/crc32 in this worker
+ worker.codecTime = worker.crcTime = 0;
+ worker.postMessage({ type: 'importScripts', scripts: scripts.slice(1) });
+ worker.addEventListener('message', onmessage);
+ function onmessage(ev) {
+ var msg = ev.data;
+ if (msg.error) {
+ worker.terminate(); // should before onerror(), because onerror() may throw.
+ onerror(msg.error);
+ return;
+ }
+ if (msg.type === 'importScripts') {
+ worker.removeEventListener('message', onmessage);
+ worker.removeEventListener('error', errorHandler);
+ callback(worker);
+ }
+ }
+ // catch entry script loading error and other unhandled errors
+ worker.addEventListener('error', errorHandler);
+ function errorHandler(err) {
+ worker.terminate();
+ onerror(err);
+ }
+ }
+
+ function onerror_default(error) {
+ console.error(error);
+ }
+ obj.zip = {
+ Reader : Reader,
+ Writer : Writer,
+ BlobReader : BlobReader,
+ Data64URIReader : Data64URIReader,
+ TextReader : TextReader,
+ BlobWriter : BlobWriter,
+ Data64URIWriter : Data64URIWriter,
+ TextWriter : TextWriter,
+ createReader : function(reader, callback, onerror) {
+ onerror = onerror || onerror_default;
+
+ reader.init(function() {
+ createZipReader(reader, callback, onerror);
+ }, onerror);
+ },
+ createWriter : function(writer, callback, onerror, dontDeflate) {
+ onerror = onerror || onerror_default;
+ dontDeflate = !!dontDeflate;
+
+ writer.init(function() {
+ createZipWriter(writer, callback, onerror, dontDeflate);
+ }, onerror);
+ },
+ useWebWorkers : true,
+ /**
+ * Directory containing the default worker scripts (z-worker.js, deflate.js, and inflate.js), relative to current base url.
+ * E.g.: zip.workerScripts = './';
+ */
+ workerScriptsPath : null,
+ /**
+ * Advanced option to control which scripts are loaded in the Web worker. If this option is specified, then workerScriptsPath must not be set.
+ * workerScripts.deflater/workerScripts.inflater should be arrays of urls to scripts for deflater/inflater, respectively.
+ * Scripts in the array are executed in order, and the first one should be z-worker.js, which is used to start the worker.
+ * All urls are relative to current base url.
+ * E.g.:
+ * zip.workerScripts = {
+ * deflater: ['z-worker.js', 'deflate.js'],
+ * inflater: ['z-worker.js', 'inflate.js']
+ * };
+ */
+ workerScripts : null,
+ };
+
+})(this);
diff --git a/lib/zlib-asm/codecs.js b/lib/zlib-asm/codecs.js
new file mode 100644
index 0000000..a142014
--- /dev/null
+++ b/lib/zlib-asm/codecs.js
@@ -0,0 +1,49 @@
+/// wrapper for zlib-asm (https://github.com/ukyo/zlib-asm)
+
+/* globals zlib */
+(function(global) {
+ "use strict";
+
+ function Codec(isDeflater, options) {
+ this._isDeflater = isDeflater;
+ if (options && typeof options.level === 'number')
+ this.level = options.level;
+ this._inputLength = 0;
+ this._input = [];
+ }
+ Codec.prototype.append = function append(bytes, onprogress) {
+ this._inputLength += bytes.length;
+ this._input.push(bytes);
+ };
+ Codec.prototype.flush = function flush() {
+ var bytes;
+ var input = this._input;
+ if (input.length === 1)
+ bytes = input[0];
+ else {
+ bytes = new Uint8Array(this._inputLength);
+ for (var i = 0, n = input.length, off = 0; i < n; i++) {
+ var slice = input[i];
+ bytes.set(slice, off);
+ off += slice.length;
+ }
+ }
+ return this._isDeflater ?
+ zlib.rawDeflate(bytes, this.level) :
+ zlib.rawInflate(bytes);
+ };
+
+ function Deflater(options) {
+ Codec.call(this, true, options);
+ }
+ Deflater.prototype = Object.create(Codec.prototype);
+ function Inflater() {
+ Codec.call(this, false);
+ }
+ Inflater.prototype = Object.create(Codec.prototype);
+
+ // 'zip' may not be defined in z-worker and some tests
+ var env = global.zip || global;
+ env.Deflater = env._zlib_asm_Deflater = Deflater;
+ env.Inflater = env._zlib_asm_Inflater = Inflater;
+})(this);
\ No newline at end of file
diff --git a/old/index.html b/old/index.html
new file mode 100644
index 0000000..ee1aa9f
--- /dev/null
+++ b/old/index.html
@@ -0,0 +1,385 @@
+
+
+
+
+
+ 0x40
+
+
+
+
+
+
+
+
+
+ 0x00
+
+
+
+
+
stop
+
play
+
prev
+
next
+
B=0x0000
+
T=0x0000
+
Madeon - Finale
+
Megumi
+
white
+
+
+
\ No newline at end of file
diff --git a/old/index_ideal.html b/old/index_ideal.html
new file mode 100644
index 0000000..28487a2
--- /dev/null
+++ b/old/index_ideal.html
@@ -0,0 +1,57 @@
+
+
+
+
+
+ 0x40
+
+
+
+
+
+
+
+
+
+
+
+
+ 0x00
+
+
+
+
+
stop
+
play
+
prev
+
next
+
B=0x0000
+
T=0x0000
+
Madeon - Finale
+
Megumi
+
white
+
+
+
\ No newline at end of file
diff --git a/0x40.js b/old/js/0x40.js
similarity index 100%
rename from 0x40.js
rename to old/js/0x40.js
diff --git a/old/js/assetManager.js b/old/js/assetManager.js
new file mode 100644
index 0000000..0fd5a26
--- /dev/null
+++ b/old/js/assetManager.js
@@ -0,0 +1,19 @@
+var assets = {}
+
+assets.loadLocal = function(file) {
+
+}
+
+assets.loadRemote = function(file) {
+
+}
+
+// called by loadRemote/Local
+assets.decodeZip = function(zip) {
+
+}
+
+// Given a buffer
+assets.decodeSong = function(index, callback) {
+
+}
\ No newline at end of file
diff --git a/audioUtils.js b/old/js/audioUtils.js
similarity index 100%
rename from audioUtils.js
rename to old/js/audioUtils.js
diff --git a/old/js/respack.js b/old/js/respack.js
new file mode 100644
index 0000000..7fb81d3
--- /dev/null
+++ b/old/js/respack.js
@@ -0,0 +1,217 @@
+/* README for mon
+ * download zip.js from here http://gildas-lormeau.github.io/zip.js/
+ * place all .js from zip.js in lib/
+ * place this file also somewhere in the webserver dir
+ * add lib/zip.js, lib/zip-fs.js and this file as scripts
+ * call respackInit() from somewhere.
+ * Enjoy the filepicker.
+ *
+ * Limitations:
+ * - does not load images yet
+ * - cannot unload respacks
+ *
+ */
+
+function respackParseSongs(songsxml){
+ var oParser = new DOMParser();
+ var oDOM = oParser.parseFromString(songsxml, "text/xml");
+ var songList = [];
+ if(oDOM.documentElement.nodeName !== "songs"){
+ console.log("Songs.xml error, corrupt file?")
+ return songList;
+ }
+
+ var domsonglist = oDOM.documentElement.children;
+ for (var i = 0; i < domsonglist.length; i++){
+ var filename = domsonglist[i].attributes[0].value + ".mp3";
+ var rhythm = domsonglist[i].getElementsByTagName('rhythm')[0].textContent;
+ var title = domsonglist[i].getElementsByTagName('title')[0].textContent;
+ var source = domsonglist[i].getElementsByTagName('source')[0] && domsonglist[i].getElementsByTagName('source')[0].textContent;
+ var buildUp = domsonglist[i].getElementsByTagName('buildup')[0] && domsonglist[i].getElementsByTagName('buildup')[0].textContent;
+ var buildUpRhythm = domsonglist[i].getElementsByTagName('buildupRhythm')[0] && domsonglist[i].getElementsByTagName('buildupRhythm')[0].textContent;
+
+ if (buildUp && !buildUpRhythm){
+ buildUpRhythm = "."; //Add a empty rhythm
+ }
+
+
+ var result = {
+ pack: packno,
+ file: filename,
+ name: title,
+ source: source,
+ rhythm: rhythm
+ }
+ if(buildUp){
+ result.buildUp = buildUp + ".mp3";
+ result.buildUpRhythm = buildUpRhythm;
+ }
+ songList[i] = result;
+ }
+
+ return songList;
+
+}
+
+
+function respackFindBuildUpData(song, songsDir, onend){
+ songsDir.children.forEach(function(entry) {
+ var buildupFile = song.buildUp;
+ if (buildupFile === entry.name) {
+
+ entry.getData64URI("audio/mpeg", function(data) {
+
+ onend(data);
+ });
+ }
+ });
+}
+
+function respackFindSongData(song, songsDir, onend){
+ songsDir.children.forEach(function(entry) {
+ var songfile = song.file;
+ if (songfile === entry.name) {
+
+ entry.getData64URI("audio/mpeg", function(data) {
+
+ onend(data);
+ });
+ }
+
+ })
+}
+
+
+function respackIsLoaded(packname){
+ for (var i = 0; i < respacksLoaded.length; i++) {
+ if(respacksLoaded[i].name === packname ){
+ return true;
+ }
+ }
+ return false;
+}
+
+function respackInit(audio) {
+
+ //Add array.find() for chrome
+ if (!Array.prototype.find) {
+ Array.prototype.find = function(predicate) {
+ if (this == null) {
+ throw new TypeError('Array.prototype.find called on null or undefined');
+ }
+ if (typeof predicate !== 'function') {
+ throw new TypeError('predicate must be a function');
+ }
+ var list = Object(this);
+ var length = list.length >>> 0;
+ var thisArg = arguments[1];
+ var value;
+
+ for (var i = 0; i < length; i++) {
+ value = list[i];
+ if (predicate.call(thisArg, value, i, list)) {
+ return value;
+ }
+ }
+ return undefined;
+ };
+ }
+
+ zip.workerScriptsPath = 'lib/';
+ respacksLoaded = [];
+ packno = 4;
+ function addGlobalSong(song){
+ audio.songs[audio.songs.length] = song;
+ }
+ var fileInput = document.getElementById("respack-input");
+ if(!audio || !audio.songs ){
+ console.log("invalid audio object, bailing...");
+ if(fileInput){
+ fileInput.remove();
+ }
+
+ }
+
+
+ function respackHandleZipFile(event) {
+ var fileInput = document.getElementById("respack-input");
+
+ var fs = new zip.fs.FS();
+
+ fs.importBlob( fileInput.files[0], function() {
+
+ if(fs.root.children.length === 1){
+ rootdir = fs.root.children[0]
+ var basename = rootdir.name;
+ //Packshit respack is giving me headaches, this "fixes" the directory structure like Packshit.zip/Packshit/Packshit/songs.xml
+ if(rootdir.children.length == 1 && rootdir.children[0].name == basename){
+ rootdir = rootdir.children[0];
+ }
+ if(respackIsLoaded(basename)){
+ alert("respack " + basename + " already loaded!");
+ return;
+ }else{
+ respacksLoaded[respacksLoaded.length] = {name: basename, packno: packno};
+ }
+ console.log("Loading " + basename+".zip");
+ //Find files of iterest
+ var songsXML = rootdir.children.find(function(element,index, array){
+ if( element.name.toLowerCase() === "songs.xml"){
+ return element;
+ }
+ return false;
+ });
+
+ var songsDir = rootdir.children.find(function(element,index, array){
+ if( element.directory && (element.name.toLowerCase() === "songs" || element.name.toLowerCase() === "loops") ){
+ return element;
+ }
+ return false;
+ });
+ songsXML.getText(function(text){
+ text = text.replace(/&/g, '&'); //XML parser will complain about a bare '&', found in the Packshit respack
+ var songList = respackParseSongs(text);
+ songList.forEach(function(song, index, arr){
+ if(song.buildUp){
+ respackFindBuildUpData(song, songsDir, function(data){
+ song.buildUp = data;
+
+ });
+ }
+ respackFindSongData(song, songsDir, function(data){
+ song.file = data;
+ addGlobalSong(song);
+ });
+
+ if( index === (arr.length-1)){
+
+ packno += 1;
+ fs = null; //or somethin
+ }
+ });
+
+ });
+
+ }
+
+ }, function(error) {
+ // onerror callback
+ console.log("Error loading zip file!");
+ });
+ this.value = null; //Chrom*
+ return false; //Firefox
+ }
+ if(!fileInput){
+ //create a file input
+ fileInput = document.createElement("input");
+ fileInput.type ="file"
+ fileInput.accept="application/zip"
+ fileInput.id="respack-input"
+ var controls = document.getElementById("controls");
+ controls.appendChild(fileInput);
+
+ }
+ fileInput.addEventListener('change', respackHandleZipFile, false);
+
+
+}
diff --git a/waifuCanvas.js b/old/js/waifuCanvas.js
similarity index 99%
rename from waifuCanvas.js
rename to old/js/waifuCanvas.js
index 060e21d..0029e7e 100644
--- a/waifuCanvas.js
+++ b/old/js/waifuCanvas.js
@@ -26,7 +26,7 @@ waifuCanvas.init = function() {
canvas = document.getElementById("waifu").getContext("2d");
window.addEventListener('resize', waifuCanvas.resize);
waifuCanvas.resize();
- canvas.drawImage(waifuImgs[0], 0, 0);
+ //canvas.drawImage(waifuImgs[0], 0, 0);
}
waifuCanvas.resize = function() {
diff --git a/old/js/waifus.js b/old/js/waifus.js
new file mode 100644
index 0000000..f36ce7d
--- /dev/null
+++ b/old/js/waifus.js
@@ -0,0 +1,113 @@
+waifus = {}
+
+waifus.defaults = {
+ respacks: ["respacks/default.zip"],
+ preloadBuild: null,
+ preloadLoop: null,
+ preloadOutro: null,
+ customColourSet: null,
+ blurQuality: 2, // low/med/high/extreme 0-3
+
+ // UI accessible config
+ // Autosong stuff is a todo
+ smartAlign: true,
+ blendMode: "hard-light", // hard-light, TODO: plain, alpha
+ blurAmount: 1, // 0,1,2,3 = off,low,med,high
+ blurDecay: 2, // 0,1,2,3 = slow,med,fast,faster!
+ colourSet: "normal", // normal, pastel, 420, custom
+ // scaleImages, nah
+ blackoutUI: false,
+ // channel selection, nah
+ playBuildups: "on" // off, once, on
+}
+
+waifus.init = function() {
+ console.log("Initialising 0x40");
+ // merge defaults and user set stuff
+ waifus.config = waifus.config || {};
+ for(var attr in waifus.defaults) {
+ if(!waifus.config[attr])
+ waifus.config[attr] = waifus.defaults[attr];
+ }
+
+ waifus.updateBlurQuality();
+ waifus.updateBlurAmount();
+ waifus.updateBlurDecay();
+ waifuCanvas.blendMode = waifus.config.blendMode;
+ waifuCanvas.init();
+
+ waifus.initPreloader();
+ if(waifus.config.preloadBuild || waifus.config.preloadLoop || waifus.config.preloadOutro)
+ waifus.initPreloadSong();
+
+ for(var i in waifus.config.respacks) {
+ waifus.loadRespack(i);
+ }
+ console.log("Initialisation complete");
+}
+
+
+waifus.initPreloadSong = function() {
+
+}
+
+waifus.initPreloader = function() {
+ waifus.preload = [];
+ waifus.leftToLoad = waifus.config.respacks.length;
+ for(var i in waifus.config.respacks) {
+ waifus.preload[i] = 0;
+ }
+}
+
+waifus.updatePreloader = function() {
+ var total = 0;
+ for(var i in waifus.config.respacks) {
+ total += waifus.preload[i];
+ }
+ total /= waifus.config.respacks.length;
+ total = Math.floor(total * 0x40);
+ document.getElementById("preloader").innerHTML = '0x' + pad(total.toString(16), 2);
+}
+
+waifus.loadRespack = function(i) {
+ var respack = waifus.config.respacks[i];
+ console.log("Loading " + respack);
+ var req = new XMLHttpRequest();
+ req.open('GET', respack, true);
+ req.responseType = 'arraybuffer';
+ req.onload = function() {
+ console.log(respack + " downloaded");
+ waifus.onRespackLoad(req.response);
+ };
+ req.onprogress = function(evt) {
+ if (evt.lengthComputable) {
+ waifus.preload[i] = evt.loaded / evt.total;
+ waifus.updatePreloader();
+ }
+ }
+ req.send();
+}
+
+waifus.onRespackLoad = function(buff) {
+ processRespack(buff);
+ waifus.leftToLoad--;
+ if(!waifus.leftToLoad) {
+ console.log("All zips downloaded");
+ document.getElementById("preloader").style.color = "#0F0";
+ }
+}
+
+waifus.updateBlurQuality = function() {
+ var iterations = [5, 15, 31, 65];
+ waifuCanvas.setBlurIterations(iterations[waifus.config.blurQuality]);
+}
+
+waifus.updateBlurAmount = function() {
+ var amount = [0, 15, 30, 60];
+ waifuCanvas.setBlurAmount(amount[waifus.config.blurAmount]);
+}
+
+waifus.updateBlurDecay = function() {
+ var decay = [10, 15, 25, 45];
+ waifuCanvas.setBlurDecay(decay[waifus.config.blurDecay]);
+}
\ No newline at end of file
diff --git a/style.css b/old/style.css
similarity index 100%
rename from style.css
rename to old/style.css
diff --git a/orig.txt b/orig.txt
deleted file mode 100644
index 2c5eddf..0000000
--- a/orig.txt
+++ /dev/null
@@ -1,1886 +0,0 @@
-//class HuesReloadedRe
-package
-{
- import flash.display.*;
- import flash.events.*;
- import flash.filters.*;
- import flash.geom.*;
- import flash.media.*;
- import flash.net.*;
- import flash.ui.*;
- import flash.utils.*;
-
- public class HuesReloadedRe extends flash.display.MovieClip implements HuesCore
- {
- public function HuesReloadedRe()
- {
- this.currentColors = this.oldColors;
- this.respackBytes = new flash.utils.ByteArray();
- this.uiArray = [];
- this.lastSongArray = [];
- this.lastImageArray = [];
- this.sampleTargetArray = new flash.utils.ByteArray();
- super();
- Logging.log("0x40 Hues " + this.versionText + " (" + this.versionDate + ") - start your engines!");
- this._settings = new HuesSettings();
- this.oldAutoSong = this.settings.autosong;
- this.outputSound = new flash.media.Sound();
- this.outputSound.addEventListener(flash.events.SampleDataEvent.SAMPLE_DATA, this.sampler);
- this.resourceManager = new ResourceManager();
- this.builtIn = new BuiltResourcePack();
- this.builtIn.addEventListener(flash.events.Event.COMPLETE, this.respackLoaded);
- this.blackOverlay = new flash.display.MovieClip();
- this.blackOverlay.graphics.beginFill(0);
- this.blackOverlay.graphics.drawRect(0, 0, 10, 10);
- this.blackOverlay.graphics.endFill();
- this.blackOverlay.width = 1280;
- this.blackOverlay.height = 720;
- this.blackOverlay.mouseEnabled = false;
- this.blackOverlay.mouseChildren = false;
- this.blackOverlay.alpha = 0;
- this.blackOverlayColorTransform = new flash.geom.ColorTransform();
- this.destBMD = new flash.display.BitmapData(1280, 720);
- this.colorBMD = new flash.display.BitmapData(1280, 720, false, 4294967295);
- this.destBM = new flash.display.Bitmap(this.destBMD);
- this.destBM.smoothing = true;
- this.colorBM = new flash.display.Bitmap(this.colorBMD);
- this.colorBM.smoothing = false;
- this.colorBM.alpha = 0.7;
- this.colorBM.blendMode = flash.display.BlendMode.HARDLIGHT;
- this.colorBM.width = 1280;
- this.colorBM.height = 720;
- this.imageHolder = new flash.display.MovieClip();
- this.imageHolder.addChild(this.destBM);
- this.imageHolder.addChild(this.colorBM);
- addChildAt(this.imageHolder, 0);
- addChildAt(this.blackOverlay, (this.numChildren - 1));
- this.blurFilter = new flash.filters.BlurFilter(0, 0, 1);
- this.userInterface = this.retroui as HuesUserInterface;
- this.userInterface.connectCore(this);
- this.window.connectCore(this);
- this.uiArray.push(new RetroUI(), new WeedUI(), new ModernUI(), new XmasUI());
- this.settings.addCallback(this.settingsUpdated);
- this.addEventListener(flash.events.Event.ADDED_TO_STAGE, this.onAddedToStage);
- this.window.hide();
- this.builtIn.ready();
- return;
- }
-
- public function loadLocal():void
- {
- this.loadLocalMulti();
- return;
- }
-
- public function loadLocalMulti():void
- {
- trace("Opening multiple");
- this.files = new flash.net.FileReferenceList();
- var loc1:*=new flash.net.FileFilter("Resource pack (*.zip)", "*.zip");
- this.files.addEventListener(flash.events.Event.SELECT, this.selectedMultipleFiles);
- this.files.browse([loc1]);
- return;
- }
-
- internal function selectedMultipleFiles(arg1:flash.events.Event=null):*
- {
- var loc1:*=undefined;
- if (arg1)
- {
- arg1.target.removeEventListener(flash.events.Event.SELECT, this.selectedMultipleFiles);
- }
- if (this.loadedFiles < this.files.fileList.length)
- {
- loc1 = this.files.fileList[this.loadedFiles];
- trace("@@@ Loading file", this.loadedFiles, "of", this.files.fileList.length, loc1.name);
- loc1.addEventListener(flash.events.Event.COMPLETE, this.loadedFile);
- loc1.load();
- }
- else
- {
- trace("@@@ All loaded!");
- this.loadedFiles = 0;
- this.files = null;
- }
- return;
- }
-
- internal function loadedFile(arg1:flash.events.Event):*
- {
- var loc2:*;
- var loc3:*=((loc2 = this).loadedFiles + 1);
- loc2.loadedFiles = loc3;
- var loc1:*=new ResourcePack();
- this.window.loadingLocal(loc1);
- loc1.addEventListener(flash.events.Event.COMPLETE, this.respackLoaded);
- loc1.loadFileReference(arg1.target);
- arg1.target.removeEventListener(flash.events.Event.COMPLETE, this.loadedFile);
- this.selectedMultipleFiles();
- return;
- }
-
- public function getCurrentMode():String
- {
- return this.isFullAuto ? "FULL AUTO" : "NORMAL";
- }
-
- public function get colors():Array
- {
- return this.currentColors;
- }
-
- public function getColorNames():Array
- {
- return this.oldColorNames;
- }
-
- public function get version():String
- {
- return "5.11";
- }
-
- public function get flavor():String
- {
- return null;
- }
-
- public function get versionDate():String
- {
- return "14.03.2015";
- }
-
- public function get versionText():String
- {
- return this.flavor ? this.version + " " + this.flavor : this.version;
- }
-
- internal function increaseVolume():*
- {
- this.volumeMuted = false;
- this.currentVolume = this.currentVolume + 4;
- if (this.currentVolume > 100)
- {
- this.currentVolume = 100;
- }
- this.updateVolume();
- return;
- }
-
- internal function toggleMute():*
- {
- this.volumeMuted = !this.volumeMuted;
- this.updateVolume();
- return;
- }
-
- internal function updateVolume():*
- {
- var loc1:*=undefined;
- if (this.soundChannel)
- {
- loc1 = this.soundChannel.soundTransform;
- loc1.volume = this.volumeMuted ? 0 : this.currentVolume / 100;
- this.soundChannel.soundTransform = loc1;
- }
- return;
- }
-
- public function get settings():HuesSettings
- {
- return this._settings;
- }
-
- public function nextSong():void
- {
- this.lastSongArray = [];
- var loc1:*=(this.songIndex + 1 + this.resourceManager.enabledSongCount) % this.resourceManager.enabledSongCount;
- this.setSong(loc1);
- return;
- }
-
- public function previousSong():void
- {
- this.lastSongArray = [];
- var loc1:*=((this.songIndex - 1) + this.resourceManager.enabledSongCount) % this.resourceManager.enabledSongCount;
- this.setSong(loc1);
- return;
- }
-
- public function get resManager():ResourceManager
- {
- return this.resourceManager;
- }
-
- public function getSongIndex():Number
- {
- return this.songIndex;
- }
-
- public function setSong(arg1:Number):void
- {
- var loc1:*=undefined;
- var loc2:*=undefined;
- this.songIndex = arg1;
- this.currentSong = this.resourceManager.enabledSongs[this.songIndex];
- trace("Next song:", this.songIndex, this.currentSong, this.currentSong ? this.currentSong.name : "");
- if (this.currentSong == undefined)
- {
- this.currentSong = {"name":"None", "title":"None", "rhythm":".", "source":null, "crc":"none", "sound":null, "enabled":true, "filename":"none"};
- }
- this.userInterface.setSongText();
- this.loopCount = 0;
- if (this.currentSong.buildup)
- {
- if (this.currentSong.force)
- {
- if (this.currentSong.force != "song")
- {
- this.currentSong.buildupPlayed = false;
- this.doBuildup = true;
- }
- else
- {
- this.currentSong.buildupPlayed = true;
- this.doBuildup = false;
- }
- this.currentSong.force = null;
- }
- else
- {
- var loc3:*=this.settings.buildups;
- switch (loc3)
- {
- case "off":
- {
- this.currentSong.buildupPlayed = true;
- this.doBuildup = false;
- break;
- }
- case "on":
- {
- this.currentSong.buildupPlayed = false;
- this.doBuildup = true;
- break;
- }
- case "once":
- {
- this.doBuildup = !this.currentSong.buildupPlayed;
- break;
- }
- }
- }
- loc1 = this.calculateSongLength(this.currentSong.sound) / this.currentSong.rhythm.length;
- loc2 = int(this.calculateSongLength(this.currentSong.buildup) / loc1) + 1;
- if (this.currentSong.buildupRhythm == null)
- {
- this.currentSong.buildupRhythm = "";
- }
- if (this.currentSong.buildupRhythm.length < loc2)
- {
- trace("Filling buildup beatmap");
- while (this.currentSong.buildupRhythm.length < loc2)
- {
- this.currentSong.buildupRhythm = this.currentSong.buildupRhythm + ".";
- }
- }
- trace("Buildup length:", loc2);
- }
- this.restartAudio();
- return;
- }
-
- internal function decreaseVolume():*
- {
- this.volumeMuted = false;
- this.currentVolume = this.currentVolume - 4;
- if (this.currentVolume < 0)
- {
- this.currentVolume = 0;
- }
- this.updateVolume();
- return;
- }
-
- public function randomSong():void
- {
- var loc2:*=undefined;
- var loc1:*=int(Math.random() * this.resourceManager.enabledSongCount);
- if (loc1 == this.songIndex && this.resourceManager.enabledSongCount > 1 || !(this.lastSongArray.indexOf(loc1) == -1))
- {
- this.randomSong();
- }
- else
- {
- trace("Randoming a song!");
- this.setSong(loc1);
- this.lastSongArray.push(loc1);
- loc2 = Math.min(5, int(this.resourceManager.enabledSongCount / 2));
- while (this.lastSongArray.length > loc2 && loc2 >= 0)
- {
- this.lastSongArray.shift();
- }
- }
- return;
- }
-
- internal function onLoop():*
- {
- this.loopCount = this.loopCount + 1;
- var loc1:*=this.settings.autosong;
- switch (loc1)
- {
- case "loop":
- {
- trace("Checking loops");
- if (this.loopCount >= this.settings.autosongDelay)
- {
- this.startSongChangeFade();
- }
- break;
- }
- case "time":
- {
- trace("Checking times");
- if (this.currentSong.sound && this.calculateSongLength(this.currentSong.sound) / 1000 * this.loopCount >= this.settings.autosongDelay * 60)
- {
- this.startSongChangeFade();
- }
- break;
- }
- }
- return;
- }
-
- public function startSongChangeFade():void
- {
- var loc1:*;
- this.fadeDirection = loc1 = true;
- this.fadeOut = loc1;
- return;
- }
-
- public function restartAudio():*
- {
- this.resetAudio();
- this.doBlackout = false;
- this.soundChannel = this.outputSound.play();
- this.updateVolume();
- return;
- }
-
- public function songDataUpdated():void
- {
- if (this.currentSong)
- {
- this.samples = this.calculateSamples(this.currentSong.sound);
- this.beatLength = this.calculateSongLength(this.currentSong.sound) / this.currentSong.rhythm.length;
- this.userInterface.updateLists();
- this.userInterface.updateTexts();
- this.userInterface.setSongText();
- this.userInterface.setImageText();
- }
- else
- {
- var loc1:*;
- this.beatLength = loc1 = -1;
- this.samples = loc1;
- }
- return;
- }
-
- public function resetAudio():*
- {
- this.samplePosition = 0;
- this.lastBeat = -123;
- this.position = 0;
- this.testPosition = 0;
- this.lastSC = 0;
- this.buildupDiff = 0;
- this.songDataUpdated();
- if (this.soundChannel)
- {
- this.soundChannel.stop();
- }
- this.soundChannel = null;
- return;
- }
-
- public function randomImage():void
- {
- var loc2:*=undefined;
- var loc1:*=int(Math.random() * this.resourceManager.enabledBothCount);
- if ((loc1 == this.imageIndex || !(this.lastImageArray.indexOf(loc1) == -1)) && this.resourceManager.enabledBothCount > 1)
- {
- this.randomImage();
- }
- else
- {
- this.setImage(loc1);
- this.lastImageArray.push(loc1);
- loc2 = Math.min(20, int(this.resourceManager.enabledBothCount / 2));
- while (this.lastImageArray.length > loc2 && loc2 >= 0)
- {
- this.lastImageArray.shift();
- }
- }
- return;
- }
-
- public function getImageIndex():Number
- {
- return this.imageIndex;
- }
-
- public function setImage(arg1:Number):void
- {
- this.imageIndex = arg1;
- this.animationFrame = 0;
- this.tmpFrame = 0;
- var loc1:*=this.resourceManager.enabledBoth[this.imageIndex];
- if (loc1 == this.currentImage && !(loc1 == null))
- {
- return;
- }
- if (loc1 != undefined)
- {
- this.currentImage = loc1;
- }
- else if (!this.currentImage || !(this.currentImage.crc == "none"))
- {
- this.currentImage = {"name":"None", "fullname":"None", "align":"center", "bitmap":new flash.display.BitmapData(1280, 720, true, 16777215), "crc":"none", "source":null, "enabled":true};
- this.imageIndex = -1;
- this.lastImageArray = [];
- }
- if (this.animTimeoutID)
- {
- flash.utils.clearTimeout(this.animTimeoutID);
- }
- if (this.currentImage.bitmaps)
- {
- this.animTimeoutID = flash.utils.setTimeout(this.animationTimeOut, this.currentImage.frameDurations[0]);
- }
- this.userInterface.setImageText();
- this.drawImage();
- return;
- }
-
- public function getCurrentSong():Object
- {
- return this.currentSong;
- }
-
- public function getCurrentImage():Object
- {
- return this.currentImage;
- }
-
- internal function alignImage():*
- {
- if (!stage)
- {
- return;
- }
- if (this.settings.smartAlign && !(this.currentImage == null))
- {
- var loc1:*=this.currentImage.align;
- switch (loc1)
- {
- case "center":
- {
- this.destBM.x = (stage.stageWidth - this.destBM.width) / 2;
- break;
- }
- case "left":
- {
- this.destBM.x = 0;
- break;
- }
- case "right":
- {
- this.destBM.x = stage.stageWidth - this.destBM.width;
- break;
- }
- }
- }
- else
- {
- this.destBM.x = (stage.stageWidth - this.destBM.width) / 2;
- }
- return;
- }
-
- public function nextImage():void
- {
- this.isFullAuto = false;
- var loc1:*=(this.imageIndex + 1 + this.resourceManager.enabledBothCount) % this.resourceManager.enabledBothCount;
- this.setImage(loc1);
- this.lastImageArray = [];
- return;
- }
-
- public function previousImage():void
- {
- this.isFullAuto = false;
- var loc1:*=((this.imageIndex - 1) + this.resourceManager.enabledBothCount) % this.resourceManager.enabledBothCount;
- this.setImage(loc1);
- this.lastImageArray = [];
- return;
- }
-
- internal function randomColorIndex():*
- {
- var loc1:*=int(Math.random() * 64);
- var loc2:*=!(this.settings.blendMode == "hard") && this.colors[loc1].c == 0;
- if (loc1 == this.colorIndex || loc2)
- {
- return this.randomColorIndex();
- }
- return loc1;
- }
-
- internal function randomColor():*
- {
- var loc1:*=this.randomColorIndex();
- this.setColor(loc1);
- return;
- }
-
- internal function setColor(arg1:Number):*
- {
- this.stopFade();
- this.colorIndex = arg1;
- var loc1:*=this.colors[this.colorIndex].c;
- this.colorBMD.fillRect(this.colorBMD.rect, loc1);
- return;
- }
-
- internal function startFade(arg1:*):*
- {
- trace("started fadde");
- this.doColorFade = true;
- this.colorFadeLength = int(arg1);
- this.colorFadeStarted = flash.utils.getTimer();
- this.oldColor = this.colors[this.colorIndex].c;
- this.colorIndex = this.randomColorIndex();
- return;
- }
-
- internal function stopFade():*
- {
- this.doColorFade = false;
- this.oldColor = 16777215;
- this.colorFadeStarted = 0;
- this.colorFadeLength = 0;
- return;
- }
-
- internal function fadeColor(arg1:Number):*
- {
- var loc1:*=this.mixColors(this.oldColor, this.colors[this.colorIndex].c, arg1);
- this.colorBMD.fillRect(this.colorBMD.rect, loc1);
- return;
- }
-
- internal function mixColors(arg1:*, arg2:*, arg3:*):*
- {
- arg3 = Math.min(1, Math.max(0, arg3));
- var loc1:*=arg1 >> 16 & 255;
- var loc2:*=arg1 >> 8 & 255;
- var loc3:*=arg1 & 255;
- var loc4:*=arg2 >> 16 & 255;
- var loc5:*=arg2 >> 8 & 255;
- var loc6:*=arg2 & 255;
- var loc7:*=loc1 * (1 - arg3) + loc4 * arg3;
- var loc8:*=loc2 * (1 - arg3) + loc5 * arg3;
- var loc9:*=loc3 * (1 - arg3) + loc6 * arg3;
- var loc10:*;
- return loc10 = loc7 << 16 | loc8 << 8 | loc9;
- }
-
- public function getColorIndex():Number
- {
- return this.colorIndex;
- }
-
- public function getBlurFilter():flash.filters.BlurFilter
- {
- return this.blurFilter;
- }
-
- public function getLastBeat():Number
- {
- return Math.max(this.lastBeat, 0);
- }
-
- public function getCurrentBeat():Number
- {
- if (!this.currentSong || !this.soundChannel)
- {
- return 0;
- }
- var loc1:*=this.currentSong.buildup ? this.calculateSongLength(this.currentSong.buildup) : 0;
- var loc2:*=this.soundChannel.position - (this.doBuildup ? loc1 : 0) < 0;
- var loc3:*=loc2 ? this.soundChannel.position - (this.doBuildup ? loc1 : 0) : (this.soundChannel.position - (this.doBuildup ? loc1 : 0)) % this.calculateSongLength(this.currentSong.sound);
- var loc4:*;
- return loc4 = Math.floor(loc3 / this.beatLength);
- }
-
- public function getSoundPosition():Number
- {
- return this.soundChannel ? (this.soundChannel.position - this.buildupDiff) % (this.beatLength * this.currentSong.rhythm.length) : 0;
- }
-
- public function getRhythm():String
- {
- return this.currentSong.buildup && !this.currentSong.buildupPlayed ? this.currentSong.buildupRhythm : this.currentSong.rhythm;
- }
-
- internal function beater():*
- {
- var loc1:*=undefined;
- var loc2:*=undefined;
- var loc3:*=undefined;
- var loc4:*=undefined;
- var loc5:*=undefined;
- var loc6:*=undefined;
- var loc7:*=undefined;
- var loc8:*=undefined;
- var loc9:*=undefined;
- var loc10:*=undefined;
- var loc11:*=undefined;
- var loc12:*=undefined;
- if (this.soundChannel && this.currentSong && this.currentSong.sound)
- {
- loc1 = this.currentSong.buildup ? this.calculateSongLength(this.currentSong.buildup) : 0;
- loc2 = this.soundChannel.position - (this.doBuildup ? loc1 : 0) < 0;
- loc3 = loc2 ? this.soundChannel.position - (this.doBuildup ? loc1 : 0) : (this.soundChannel.position - (this.doBuildup ? loc1 : 0)) % this.calculateSongLength(this.currentSong.sound);
- if ((loc4 = Math.floor(loc3 / this.beatLength)) != this.lastBeat)
- {
- loc5 = loc2 ? this.currentSong.buildupRhythm : this.currentSong.rhythm;
- loc6 = loc2 ? loc5.length + loc4 : loc4;
- this.userInterface.doBeat(loc4);
- loc7 = loc5.charAt(loc6);
- var loc13:*=loc7;
- switch (loc13)
- {
- case "X":
- case "x":
- {
- this.blurFilter.blurY = 96 * this.blurMultiplier;
- break;
- }
- case "O":
- case "o":
- {
- this.blurFilter.blurX = 96 * this.blurMultiplier;
- break;
- }
- case "+":
- {
- this.blurFilter.blurX = 96 * this.blurMultiplier;
- this.doBlackout = true;
- this.recolorBlackout(false);
- break;
- }
- case "¤":
- {
- this.blurFilter.blurX = 96 * this.blurMultiplier;
- this.doBlackout = true;
- this.recolorBlackout(true);
- break;
- }
- case "|":
- {
- this.halfBeatBlackout();
- break;
- }
- case ":":
- {
- this.randomColor();
- break;
- }
- case "*":
- {
- if (this.isFullAuto)
- {
- this.randomImage();
- }
- break;
- }
- case "=":
- {
- if (this.isFullAuto)
- {
- this.randomImage();
- }
- }
- case "~":
- {
- loc9 = loc5.substr(loc6);
- loc10 = 1;
- loc11 = false;
- for (;;)
- {
- if (loc9.length <= loc10)
- {
- if (loc11)
- {
- break;
- }
- loc9 = loc9 + this.currentSong.rhythm;
- loc11 = true;
- }
- if ((loc12 = loc9.charAt(loc10)) != ".")
- {
- break;
- }
- ++loc10;
- }
- this.stopFade();
- this.startFade(loc10 * this.beatLength);
- break;
- }
- }
- if ([".", "+", "|", "¤"].indexOf(loc7) == -1)
- {
- this.doBlackout = false;
- }
- if ((loc8 = [".", "+", ":", "*", "X", "O", "~", "="]).indexOf(loc7) == -1)
- {
- this.randomColor();
- if (this.isFullAuto)
- {
- this.randomImage();
- }
- }
- this.lastBeat = loc4;
- }
- }
- return;
- }
-
- internal function halfBeatBlackout():*
- {
- this.doBlackout = true;
- this.recolorBlackout(false);
- flash.utils.setTimeout(this.stopBlackout, this.beatLength / 1.7);
- return;
- }
-
- internal function recolorBlackout(arg1:Boolean=false):*
- {
- var loc1:*=arg1 ? 16777215 : 0;
- if (this.blackOverlayColorTransform.color != loc1)
- {
- this.blackOverlayColorTransform.color = loc1;
- this.blackOverlay.transform.colorTransform = this.blackOverlayColorTransform;
- }
- return;
- }
-
- public function get isFullAuto():Boolean
- {
- return this._isFullAuto;
- }
-
- public function set isFullAuto(arg1:Boolean):*
- {
- this._isFullAuto = arg1;
- if (this.userInterface)
- {
- this.userInterface.modeUpdated();
- }
- return;
- }
-
- public function setModeAuto():void
- {
- this.isFullAuto = true;
- return;
- }
-
- public function setModeNormal():void
- {
- this.isFullAuto = false;
- return;
- }
-
- internal function stopBlackout():*
- {
- this.doBlackout = false;
- return;
- }
-
- internal function animationTimeOut():*
- {
- this.animationFrame = this.animationFrame + 1;
- if (this.animationFrame >= this.currentImage.bitmaps.length)
- {
- this.animationFrame = 0;
- }
- var loc1:*=this.currentImage.frameDurations[this.animationFrame];
- this.animTimeoutID = flash.utils.setTimeout(this.animationTimeOut, loc1);
- this.drawImage();
- return;
- }
-
- public function setVolume(arg1:Number):void
- {
- this.currentVolume = Math.max(0, Math.min(100, arg1));
- this.updateVolume();
- return;
- }
-
- public function setMute(arg1:Boolean):void
- {
- this.volumeMuted = arg1;
- this.updateVolume();
- return;
- }
-
- public function getVolume():Number
- {
- return this.currentVolume;
- }
-
- public function getVolumeMuted():Boolean
- {
- return this.volumeMuted;
- }
-
- internal function enterFrame(arg1:flash.events.Event):*
- {
- var loc1:*=undefined;
- var loc2:*=undefined;
- var loc3:*=undefined;
- var loc4:*=undefined;
- var loc5:*=undefined;
- var loc6:*=undefined;
- this.doImageBlurring();
- this.setTexts();
- if (this.doColorFade)
- {
- loc1 = flash.utils.getTimer() - this.colorFadeStarted;
- loc2 = loc1 / this.colorFadeLength;
- if (loc2 >= 1)
- {
- this.stopFade();
- }
- else
- {
- this.fadeColor(loc2);
- }
- }
- if (this.fadeOut)
- {
- loc3 = 0.02;
- loc4 = this.fadeDirection ? -loc3 : loc3;
- if (this.soundChannel)
- {
- loc5 = this.soundChannel.soundTransform;
- loc6 = Math.max(0, Math.min(this.currentVolume / 100, loc5.volume + loc4));
- loc5.volume = loc6;
- this.soundChannel.soundTransform = loc5;
- if (loc6 == 0)
- {
- var loc7:*;
- this.fadeOut = loc7 = false;
- this.fadeDirection = loc7;
- if (this.settings.autosongShuffle)
- {
- this.randomSong();
- }
- else
- {
- this.nextSong();
- }
- }
- }
- }
- if (this.doBlackout)
- {
- this.blackOverlay.alpha = this.blackOverlay.alpha + 0.4;
- if (this.blackOverlay.alpha > 1)
- {
- this.blackOverlay.alpha = 1;
- }
- }
- else
- {
- this.blackOverlay.alpha = 0;
- }
- return;
- }
-
- public function respackLoaded(arg1:flash.events.Event):void
- {
- var loc1:*=null;
- if (arg1 && arg1.target && arg1.target is ResourcePackInterface)
- {
- loc1 = arg1.target as ResourcePackInterface;
- this.resourceManager.addPack(loc1);
- }
- this.window.updateList(this.resManager);
- this.init();
- this.userInterface.updateLists();
- if (this.doRandom && arg1 && arg1.target && flash.utils.getQualifiedClassName(arg1.target) == "BuiltResourcePack")
- {
- this.randomSong();
- }
- return;
- }
-
- public function get userInterface():HuesUserInterface
- {
- return this._userInterface;
- }
-
- public function set userInterface(arg1:HuesUserInterface):*
- {
- this._userInterface = arg1;
- return;
- }
-
- internal function setTexts():*
- {
- this.userInterface.updateTexts();
- return;
- }
-
- internal function doImageBlurring():*
- {
- var loc1:*=this.blurFilter.blurX;
- var loc2:*=this.blurFilter.blurY;
- var loc3:*=1 + this.blurDecayMultiplier;
- if (this.blurFilter.blurX > 0)
- {
- this.blurFilter.blurX = this.blurFilter.blurX / loc3;
- }
- if (this.blurFilter.blurX <= 1)
- {
- this.blurFilter.blurX = 0;
- }
- if (this.blurFilter.blurY > 0)
- {
- this.blurFilter.blurY = this.blurFilter.blurY / loc3;
- }
- if (this.blurFilter.blurY <= 1)
- {
- this.blurFilter.blurY = 0;
- }
- if (!(this.blurFilter.blurX == loc1) || !(this.blurFilter.blurY == loc2))
- {
- this.drawImage();
- }
- return;
- }
-
- public function getWindow():flash.display.MovieClip
- {
- return this.window as flash.display.MovieClip;
- }
-
- internal function resizeBitmap(arg1:flash.display.BitmapData):*
- {
- var loc5:*=null;
- var loc1:*=arg1.rect;
- var loc2:*=this.destBMD.rect;
- var loc3:*="[" + loc1.width + "," + loc1.height + "]";
- var loc4:*="[" + loc2.width + "," + loc2.height + "]";
- if (!(loc1.width == loc2.width) || !(loc1.height == loc2.height))
- {
- trace("Creating new BMD, since", loc3, "!=", loc4);
- loc5 = new flash.display.BitmapData(loc1.width, loc1.height);
- this.destBMD.dispose();
- this.destBMD = loc5;
- this.destBM.bitmapData = this.destBMD;
- this.destBM.smoothing = this.settings.imageSmoothing;
- this.resizeListener(null);
- }
- return;
- }
-
- internal function drawImage():*
- {
- var loc1:*=null;
- if (this.currentImage.bitmap)
- {
- loc1 = this.currentImage.bitmap.clone();
- this.resizeBitmap(loc1);
- if (this.settings.blurEnabled)
- {
- this.destBMD.applyFilter(loc1, loc1.rect, new flash.geom.Point(0, 0), this.blurFilter);
- }
- else
- {
- this.destBMD.copyPixels(loc1, loc1.rect, new flash.geom.Point(0, 0));
- }
- }
- else if (this.currentImage.bitmaps)
- {
- loc1 = this.currentImage.bitmaps[this.animationFrame].clone();
- this.resizeBitmap(loc1);
- this.destBMD.fillRect(this.destBMD.rect, 16777215);
- if (this.settings.blurEnabled)
- {
- this.destBMD.applyFilter(loc1, loc1.rect, new flash.geom.Point(0, 0), this.blurFilter);
- }
- else
- {
- this.destBMD.copyPixels(loc1, loc1.rect, new flash.geom.Point(0, 0));
- }
- }
- loc1.dispose();
- loc1 = null;
- this.alignImage();
- return;
- }
-
- internal function toHex(arg1:Number, arg2:Number):*
- {
- var loc1:*=arg1.toString(16);
- while (loc1.length < arg2)
- {
- loc1 = "0" + loc1;
- }
- return loc1;
- }
-
- internal function sampler(arg1:flash.events.SampleDataEvent):*
- {
- this.extractSamples(arg1.data, this.BUFFER_SIZE);
- return;
- }
-
- internal function resizeListener(arg1:flash.events.Event):*
- {
- if (!stage)
- {
- trace("No stage!");
- return;
- }
- var loc1:*=stage.stageWidth;
- var loc2:*=stage.stageHeight;
- var loc3:*;
- var loc4:*=(loc3 = loc1 / loc2) >= 1280 / 720;
- var loc6:*;
- this.blackOverlay.width = loc6 = loc1;
- this.colorBM.width = loc6;
- this.blackOverlay.height = loc6 = loc2;
- this.colorBM.height = loc6;
- this.blackOverlay.y = loc6 = 0;
- this.blackOverlay.x = loc6 = loc6;
- this.colorBM.y = loc6 = loc6;
- this.colorBM.x = loc6;
- if (this.whiteOverlay)
- {
- this.whiteOverlay.width = loc1;
- this.whiteOverlay.height = loc2;
- }
- this.window.relativePosition(loc1, loc2);
- this.window.fitIntoView();
- var loc5:*=1;
- loc6 = this.settings.imagescaling;
- switch (loc6)
- {
- case "on":
- {
- loc5 = loc2 / this.destBMD.height;
- break;
- }
- case "semi":
- {
- loc5 = this.destBMD.height >= loc2 ? loc2 / this.destBMD.height : 1;
- break;
- }
- }
- this.destBM.scaleY = loc6 = loc5;
- this.destBM.scaleX = loc6;
- this.destBM.x = (loc1 - this.destBM.width) / 2;
- this.destBM.y = (loc2 - this.destBM.height) / 2;
- if (this.userInterface)
- {
- this.userInterface.resizeInterface(loc1, loc2);
- }
- this.alignImage();
- return;
- }
-
- internal function onAddedToStage(arg1:flash.events.Event):*
- {
- var loc1:*=null;
- var loc2:*=this;
- while (loc2.parent != null)
- {
- loc2 = loc2.parent;
- }
- if (loc2)
- {
- loc1 = flash.display.LoaderInfo(loc2.loaderInfo).parameters;
- }
- else
- {
- loc1 = flash.display.LoaderInfo(this.root.loaderInfo).parameters;
- }
- this._settings.getSettingsFromParameters(loc1);
- this.window.checkUpdates();
- stage.addEventListener(flash.events.Event.RESIZE, this.resizeListener);
- stage.scaleMode = flash.display.StageScaleMode.NO_SCALE;
- stage.align = flash.display.StageAlign.TOP_LEFT;
- stage.addEventListener(flash.events.KeyboardEvent.KEY_DOWN, this.keyboardListener);
- stage.addEventListener(flash.events.KeyboardEvent.KEY_UP, this.keyboardListener);
- stage.addEventListener(flash.events.MouseEvent.RIGHT_CLICK, this.rightClickListener);
- stage.addEventListener(flash.events.MouseEvent.MOUSE_WHEEL, this.rightClickListener);
- this.resizeListener(null);
- if (this.whiteOverlay)
- {
- this.addEventListener(flash.events.Event.ENTER_FRAME, this.fadeInListener);
- }
- this.removeEventListener(flash.events.Event.ADDED_TO_STAGE, this.onAddedToStage);
- return;
- }
-
- internal function rightClickListener(arg1:flash.events.MouseEvent):*
- {
- var loc1:*=arg1.type;
- switch (loc1)
- {
- case flash.events.MouseEvent.RIGHT_CLICK:
- {
- this.toggleWindow();
- break;
- }
- case flash.events.MouseEvent.MOUSE_WHEEL:
- {
- if (arg1.delta > 0)
- {
- this.increaseVolume();
- }
- else
- {
- this.decreaseVolume();
- }
- break;
- }
- }
- return;
- }
-
- internal function fadeInListener(arg1:flash.events.Event):*
- {
- if (this.whiteOverlay.alpha == 0)
- {
- this.whiteOverlay.visible = false;
- this.removeChild(this.whiteOverlay);
- this.whiteOverlay = null;
- this.removeEventListener(flash.events.Event.ENTER_FRAME, this.fadeInListener);
- }
- else
- {
- this.whiteOverlay.alpha = Math.max(0, this.whiteOverlay.alpha - 0.05);
- }
- return;
- }
-
- internal function calculateSamples(arg1:flash.media.Sound=null):Number
- {
- if (arg1)
- {
- return int(arg1.length * 44.1 - (arg1 is FakeSound ? 0 : this.LAME_DELAY_START + this.LAME_DELAY_END));
- }
- trace("Can\'t calculateSamples");
- return -1;
- }
-
- internal function changeUI(arg1:Number):*
- {
- var loc1:*=undefined;
- if (arg1 >= 0 && this.uiArray.length > arg1 && !(this.userInterface == this.uiArray[arg1]))
- {
- this.userInterface.disconnect();
- loc1 = this.uiArray[arg1];
- addChild(loc1 as flash.display.MovieClip);
- swapChildren(loc1, this.userInterface as flash.display.MovieClip);
- removeChild(this.userInterface as flash.display.MovieClip);
- this.userInterface = loc1;
- this.userInterface.connectCore(this);
- this.userInterface.updateLists();
- this.userInterface.updateTexts();
- this.userInterface.resourcePackLoaded();
- this.userInterface.setSongText();
- this.userInterface.setImageText();
- if (stage)
- {
- this.userInterface.resizeInterface(stage.stageWidth, stage.stageHeight);
- }
- this.currentUIIndex = arg1;
- }
- return;
- }
-
- internal function calculateSongLength(arg1:flash.media.Sound=null):Number
- {
- if (arg1)
- {
- return this.calculateSamples(arg1) / 44.1;
- }
- return -1;
- }
-
- internal function settingsUpdated():*
- {
- trace("Updating according to this.settings");
- this.destBM.smoothing = this.settings.imageSmoothing;
- if (stage)
- {
- stage.quality = this.settings.flashQuality;
- }
- this.blurMultiplier = {"low":0.5, "medium":1, "high":4}[this.settings.blurAmount];
- this.blurDecayMultiplier = {"low":0.3, "medium":0.6, "high":1, "vhigh":1.6}[this.settings.blurDecay];
- var loc1:*=new flash.media.SoundTransform();
- var loc2:*=this.settings.channels;
- switch (loc2)
- {
- case "stereo":
- {
- break;
- }
- case "left":
- {
- loc1.leftToRight = loc2 = 1;
- loc1.leftToLeft = loc2;
- loc1.rightToRight = loc2 = 0;
- loc1.rightToLeft = loc2;
- break;
- }
- case "right":
- {
- loc1.leftToRight = loc2 = 0;
- loc1.leftToLeft = loc2;
- loc1.rightToRight = loc2 = 1;
- loc1.rightToLeft = loc2;
- break;
- }
- case "mono":
- {
- loc1.leftToRight = loc2 = 0.6;
- loc1.leftToLeft = loc2;
- loc1.rightToRight = loc2 = 0.6;
- loc1.rightToLeft = loc2;
- break;
- }
- }
- flash.media.SoundMixer.soundTransform = loc1;
- loc2 = this.settings.blendMode;
- switch (loc2)
- {
- case "plain":
- {
- this.imageHolder.removeChild(this.destBM);
- this.imageHolder.removeChild(this.colorBM);
- this.imageHolder.addChild(this.colorBM);
- this.imageHolder.addChild(this.destBM);
- this.colorBM.alpha = 1;
- this.destBM.alpha = 1;
- this.colorBM.blendMode = flash.display.BlendMode.NORMAL;
- break;
- }
- case "alpha":
- {
- this.imageHolder.removeChild(this.destBM);
- this.imageHolder.removeChild(this.colorBM);
- this.imageHolder.addChild(this.colorBM);
- this.imageHolder.addChild(this.destBM);
- this.colorBM.alpha = 1;
- this.destBM.alpha = 0.7;
- this.colorBM.blendMode = flash.display.BlendMode.NORMAL;
- break;
- }
- case "hard":
- {
- this.imageHolder.removeChild(this.destBM);
- this.imageHolder.removeChild(this.colorBM);
- this.imageHolder.addChild(this.destBM);
- this.imageHolder.addChild(this.colorBM);
- this.colorBM.alpha = 0.7;
- this.destBM.alpha = 1;
- this.colorBM.blendMode = flash.display.BlendMode.HARDLIGHT;
- break;
- }
- }
- loc2 = this.settings.blackoutUI;
- switch (loc2)
- {
- case true:
- {
- removeChild(this.blackOverlay);
- addChildAt(this.blackOverlay, getChildIndex(this.window));
- break;
- }
- case false:
- {
- removeChild(this.blackOverlay);
- addChildAt(this.blackOverlay, getChildIndex(this.userInterface as flash.display.MovieClip));
- break;
- }
- }
- loc2 = this.settings.colors;
- switch (loc2)
- {
- case "normal":
- {
- this.currentColors = this.oldColors;
- break;
- }
- case "pastel":
- {
- this.currentColors = this.pastelColors;
- break;
- }
- case "gp":
- {
- this.currentColors = this.gpColors;
- break;
- }
- }
- loc2 = this.settings.ui;
- switch (loc2)
- {
- case "retro":
- {
- this.changeUI(0);
- break;
- }
- case "weed":
- {
- this.changeUI(1);
- break;
- }
- case "modern":
- {
- this.changeUI(2);
- break;
- }
- case "xmas":
- {
- this.changeUI(3);
- break;
- }
- }
- if (this.oldAutoSong == "off" && !(this.settings.autosong == "off"))
- {
- trace("Resetting loopCount since AutoSong was enabled");
- this.loopCount = 0;
- this.oldAutoSong = this.settings.autosong;
- }
- if (stage)
- {
- this.resizeListener(null);
- this.alignImage();
- }
- return;
- }
-
- public function enabledChanged():void
- {
- this.resourceManager.rebuildEnabled();
- this.userInterface.updateLists();
- return;
- }
-
- internal function extractSamples(arg1:flash.utils.ByteArray, arg2:int):void
- {
- var loc2:*=0;
- var loc3:*=0;
- var loc4:*=undefined;
- var loc5:*=undefined;
- var loc6:*=undefined;
- var loc7:*=undefined;
- var loc8:*=undefined;
- var loc9:*=undefined;
- var loc1:*=arg2;
- while (0 < arg2)
- {
- loc4 = this.currentSong.buildup && !this.currentSong.buildupPlayed;
- loc5 = this.currentSong.sound is FakeSound ? 0 : this.LAME_DELAY_START;
- loc6 = loc4 ? this.currentSong.buildup : this.currentSong.sound;
- loc7 = loc4 && this.currentSong.buildupLength ? this.currentSong.buildupLength - loc5 : this.calculateSamples(loc6);
- if (!(loc4 && this.currentSong.buildupLength))
- {
- };
- if (loc7 <= 0 || !loc6)
- {
- while (loc9 < arg2)
- {
- arg1.writeFloat(0);
- arg1.writeFloat(0);
- ++loc9;
- }
- return;
- }
- loc8 = this.samplePosition;
- if (this.samplePosition + arg2 > loc7)
- {
- loc3 = loc7 - this.samplePosition;
- loc2 = loc6.extract(arg1, loc3, this.samplePosition + loc5);
- this.samplePosition = this.samplePosition + loc2;
- arg2 = arg2 - loc2;
- if (loc4)
- {
- trace("Buildup finished");
- this.samplePosition = 0;
- this.currentSong.buildupPlayed = true;
- this.buildupDiff = loc7 / 44.1;
- }
- }
- else
- {
- loc2 = loc6.extract(arg1, arg2, this.samplePosition + loc5);
- this.samplePosition = this.samplePosition + loc2;
- arg2 = arg2 - loc2;
- }
- if (!(this.samplePosition >= loc7))
- {
- continue;
- }
- this.samplePosition = 0;
- if (loc4)
- {
- trace("Buildup finished 2");
- this.currentSong.buildupPlayed = true;
- this.buildupDiff = this.calculateSongLength(this.currentSong.buildup);
- continue;
- }
- this.onLoop();
- }
- return;
- }
-
- internal function unload():*
- {
- trace("Unloading!");
- this.resetAudio();
- flash.utils.clearInterval(this.beaterId);
- removeEventListener(flash.events.Event.ENTER_FRAME, this.enterFrame);
- trace("Unloaded!");
- return;
- }
-
- internal function variableExtractSampleData(arg1:flash.utils.ByteArray, arg2:int):void
- {
- var loc1:*=NaN;
- var loc2:*=0;
- var loc3:*=NaN;
- var loc4:*=NaN;
- var loc5:*=0;
- var loc6:*=0;
- var loc7:*=0;
- var loc8:*=undefined;
- var loc9:*=0;
- var loc10:*=NaN;
- var loc11:*=NaN;
- var loc12:*=NaN;
- var loc13:*=NaN;
- var loc14:*=0;
- while (0 < arg2)
- {
- this.sampleTargetArray.position = 0;
- loc1 = arg2 * this.playbackRate / 100;
- loc2 = this.position;
- loc4 = loc3 = this.position - loc2;
- loc5 = -1;
- loc6 = Math.ceil(loc1) + 2;
- if (loc2 + loc6 > this.samples)
- {
- trace("Over samples");
- loc8 = this.samples - loc2;
- loc7 = this.currentSong.sound.extract(this.sampleTargetArray, loc8, loc2 + this.LAME_DELAY_START);
- }
- else
- {
- loc8 = loc6;
- loc7 = this.currentSong.sound.extract(this.sampleTargetArray, loc8, loc2 + this.LAME_DELAY_START);
- }
- loc9 = loc7 != loc6 ? loc7 / (this.playbackRate / 100) : arg2;
- loc14 = 0;
- while (loc14 < loc9)
- {
- if (int(loc4) != loc5)
- {
- loc5 = loc4;
- this.sampleTargetArray.position = loc5 << 3;
- loc10 = this.sampleTargetArray.readFloat();
- loc11 = this.sampleTargetArray.readFloat();
- loc12 = this.sampleTargetArray.readFloat();
- loc13 = this.sampleTargetArray.readFloat();
- }
- arg1.writeFloat(loc10 + loc3 * (loc12 - loc10));
- arg1.writeFloat(loc11 + loc3 * (loc13 - loc11));
- loc4 = loc4 + this.playbackRate / 100;
- loc3 = loc3 + this.playbackRate / 100;
- while (loc3 >= 1)
- {
- --loc3;
- }
- ++loc14;
- }
- arg2 = arg2 - loc9;
- this.position = this.position + loc1;
- if (!(this.position >= this.samples))
- {
- continue;
- }
- trace("resettomg", this.position, arg2, loc1);
- this.position = 0;
- }
- return;
- }
-
- public function init():*
- {
- if (!this.initDone)
- {
- this.beaterId = flash.utils.setInterval(this.beater, 5);
- addEventListener(flash.events.Event.ENTER_FRAME, this.enterFrame);
- this.setSong(0);
- this.setImage(0);
- this.initDone = true;
- }
- return;
- }
-
- internal function openURL(arg1:String):void
- {
- flash.net.navigateToURL(new flash.net.URLRequest(arg1), "_blank");
- return;
- }
-
- public function openSongSource():void
- {
- if (this.currentSong && this.currentSong.source)
- {
- this.openURL(this.currentSong.source);
- }
- return;
- }
-
- public function openImageSource():void
- {
- if (this.currentImage && this.currentImage.source)
- {
- this.openURL(this.currentImage.source);
- }
- return;
- }
-
- internal function keyboardListener(arg1:flash.events.KeyboardEvent):*
- {
- if (arg1.type != flash.events.KeyboardEvent.KEY_UP)
- {
- var loc1:*=arg1.keyCode;
- switch (loc1)
- {
- case flash.ui.Keyboard.LEFT:
- {
- this.previousImage();
- break;
- }
- case flash.ui.Keyboard.RIGHT:
- {
- this.nextImage();
- break;
- }
- case flash.ui.Keyboard.UP:
- {
- this.nextSong();
- break;
- }
- case flash.ui.Keyboard.DOWN:
- {
- this.previousSong();
- break;
- }
- case flash.ui.Keyboard.F:
- {
- this.isFullAuto = !this.isFullAuto;
- break;
- }
- case flash.ui.Keyboard.NUMPAD_SUBTRACT:
- case flash.ui.Keyboard.MINUS:
- {
- this.decreaseVolume();
- break;
- }
- case flash.ui.Keyboard.NUMPAD_ADD:
- case flash.ui.Keyboard.EQUAL:
- {
- this.increaseVolume();
- break;
- }
- case flash.ui.Keyboard.M:
- {
- this.toggleMute();
- break;
- }
- case flash.ui.Keyboard.H:
- {
- this.userInterface.toggleHide();
- break;
- }
- case flash.ui.Keyboard.R:
- {
- this.window.showRespacks();
- break;
- }
- case flash.ui.Keyboard.E:
- {
- this.window.showEditor();
- break;
- }
- case flash.ui.Keyboard.O:
- {
- this.window.showOptions();
- break;
- }
- case flash.ui.Keyboard.I:
- {
- this.window.showInfo();
- break;
- }
- case flash.ui.Keyboard.NUMBER_1:
- {
- this.settings.ui = "retro";
- break;
- }
- case flash.ui.Keyboard.NUMBER_2:
- {
- this.settings.ui = "weed";
- break;
- }
- case flash.ui.Keyboard.NUMBER_3:
- {
- this.settings.ui = "modern";
- break;
- }
- case flash.ui.Keyboard.NUMBER_4:
- {
- this.settings.ui = "xmas";
- break;
- }
- case flash.ui.Keyboard.L:
- {
- this.loadLocal();
- break;
- }
- case flash.ui.Keyboard.C:
- {
- this.toggleImageList();
- break;
- }
- case flash.ui.Keyboard.S:
- {
- this.toggleSongList();
- break;
- }
- case flash.ui.Keyboard.W:
- {
- this.toggleWindow();
- break;
- }
- case flash.ui.Keyboard.SHIFT:
- {
- if (!this.lastKeyEvent || !(this.lastKeyEvent.keyCode == arg1.keyCode) || !(this.lastKeyEvent.type == arg1.type))
- {
- this.randomSong();
- }
- break;
- }
- }
- }
- this.lastKeyEvent = arg1;
- return;
- }
-
- public function toggleWindow():void
- {
- this.window.toggleHide();
- this.hideLists();
- return;
- }
-
- public function hideLists():void
- {
- this.userInterface.songList.hide();
- this.userInterface.imageList.hide();
- return;
- }
-
- public function toggleSongList():void
- {
- this.userInterface.songList.toggleHide();
- this.userInterface.imageList.hide();
- this.window.hide();
- return;
- }
-
- public function toggleImageList():void
- {
- this.userInterface.imageList.toggleHide();
- this.userInterface.songList.hide();
- this.window.hide();
- return;
- }
-
- internal const oldColorNames:Array=["black", "brick", "crimson", "red", "turtle", "sludge", "brown", "orange", "green", "grass", "maize", "citrus", "lime", "leaf", "chartreuse", "yellow", "midnight", "plum", "pomegranate", "rose", "swamp", "dust", "dirt", "blossom", "sea", "ill", "haze", "peach", "spring", "mantis", "brilliant", "canary", "navy", "grape", "mauve", "purple", "cornflower", "deep", "lilac", "lavender", "aqua", "steel", "grey", "pink", "bay", "marina", "tornado", "saltine", "blue", "twilight", "orchid", "magenta", "azure", "liberty", "royalty", "thistle", "ocean", "sky", "periwinkle", "carnation", "cyan", "turquoise", "powder", "white"];
-
- internal const oldColors:Array=[{"n":"black", "c":0}, {"n":"brick", "c":5570560}, {"n":"crimson", "c":11141120}, {"n":"red", "c":16711680}, {"n":"turtle", "c":21760}, {"n":"sludge", "c":5592320}, {"n":"brown", "c":11162880}, {"n":"orange", "c":16733440}, {"n":"green", "c":43520}, {"n":"grass", "c":5614080}, {"n":"maize", "c":11184640}, {"n":"citrus", "c":16755200}, {"n":"lime", "c":65280}, {"n":"leaf", "c":5635840}, {"n":"chartreuse", "c":11206400}, {"n":"yellow", "c":16776960}, {"n":"midnight", "c":85}, {"n":"plum", "c":5570645}, {"n":"pomegranate", "c":11141205}, {"n":"rose", "c":16711765}, {"n":"swamp", "c":21845}, {"n":"dust", "c":5592405}, {"n":"dirt", "c":11162965}, {"n":"blossom", "c":16733525}, {"n":"sea", "c":43605}, {"n":"ill", "c":5614165}, {"n":"haze", "c":11184725}, {"n":"peach", "c":16755285}, {"n":"spring", "c":65365}, {"n":"mantis", "c":5635925}, {"n":"brilliant", "c":11206485}, {"n":"canary", "c":16777045}, {"n":"navy", "c":170}, {"n":"grape", "c":5570730}, {"n":"mauve", "c":11141290}, {"n":"purple", "c":16711850}, {"n":"cornflower", "c":21930}, {"n":"deep", "c":5592490}, {"n":"lilac", "c":11163050}, {"n":"lavender", "c":16733610}, {"n":"aqua", "c":43690}, {"n":"steel", "c":5614250}, {"n":"grey", "c":11184810}, {"n":"pink", "c":16755370}, {"n":"bay", "c":65450}, {"n":"marina", "c":5636010}, {"n":"tornado", "c":11206570}, {"n":"saltine", "c":16777130}, {"n":"blue", "c":255}, {"n":"twilight", "c":5570815}, {"n":"orchid", "c":11141375}, {"n":"magenta", "c":16711935}, {"n":"azure", "c":22015}, {"n":"liberty", "c":5592575}, {"n":"royalty", "c":11163135}, {"n":"thistle", "c":16733695}, {"n":"ocean", "c":43775}, {"n":"sky", "c":5614335}, {"n":"periwinkle", "c":11184895}, {"n":"carnation", "c":16755455}, {"n":"cyan", "c":65535}, {"n":"turquoise", "c":5636095}, {"n":"powder", "c":11206655}, {"n":"white", "c":16777215}];
-
- internal const pastelColors:Array=[{"n":"Mahogany", "c":13453898}, {"n":"Banana Mania", "c":16443317}, {"n":"Beaver", "c":10453360}, {"n":"Black", "c":2302755}, {"n":"Chestnut", "c":12344664}, {"n":"Copper", "c":14521461}, {"n":"Cornflower", "c":10145515}, {"n":"Denim", "c":2845892}, {"n":"Desert Sand", "c":15715768}, {"n":"Eggplant", "c":7229792}, {"n":"Electric Lime", "c":1964308}, {"n":"Fern", "c":7453816}, {"n":"Goldenrod", "c":16570741}, {"n":"Granny Smith Apple", "c":11068576}, {"n":"Gray", "c":9802124}, {"n":"Green", "c":1879160}, {"n":"Hot Magenta", "c":16719310}, {"n":"Inch Worm", "c":11725917}, {"n":"Indigo", "c":6125259}, {"n":"Laser Lemon", "c":16645236}, {"n":"Lavender", "c":16561365}, {"n":"Macaroni and Cheese", "c":16760200}, {"n":"Manatee", "c":9935530}, {"n":"Mango Tango", "c":16745027}, {"n":"Melon", "c":16628916}, {"n":"Midnight Blue", "c":1722486}, {"n":"Neon Carrot", "c":16753475}, {"n":"Olive Green", "c":12236908}, {"n":"Orange", "c":16741688}, {"n":"Orchid", "c":15116503}, {"n":"Outer Space", "c":4278860}, {"n":"Outrageous Orange", "c":16739914}, {"n":"Pacific Blue", "c":1878473}, {"n":"Periwinkle", "c":12964070}, {"n":"Plum", "c":9323909}, {"n":"Purple Heart", "c":7619272}, {"n":"Raw Sienna", "c":14060121}, {"n":"Razzmatazz", "c":14886251}, {"n":"Red", "c":15605837}, {"n":"Robin Egg Blue", "c":2084555}, {"n":"Royal Purple", "c":7885225}, {"n":"Salmon", "c":16751530}, {"n":"Scarlet", "c":16525383}, {"n":"Sea Green", "c":10478271}, {"n":"Sepia", "c":10840399}, {"n":"Shadow", "c":9075037}, {"n":"Shamrock", "c":4574882}, {"n":"Shocking Pink", "c":16482045}, {"n":"Spring Green", "c":15526590}, {"n":"Sunset Orange", "c":16604755}, {"n":"Tan", "c":16426860}, {"n":"Tickle Me Pink", "c":16550316}, {"n":"Timberwolf", "c":14407634}, {"n":"Tropical Rain Forest", "c":1540205}, {"n":"Turquoise Blue", "c":7855591}, {"n":"Vivid Tangerine", "c":16752777}, {"n":"Vivid Violet", "c":9392285}, {"n":"White", "c":15592941}, {"n":"Wild Strawberry", "c":16728996}, {"n":"Wild Watermelon", "c":16542853}, {"n":"Wisteria", "c":13477086}, {"n":"Yellow", "c":16574595}, {"n":"Yellow Green", "c":12968836}, {"n":"Yellow Orange", "c":16758355}];
-
- internal const gpColors:Array=[{"n":"Green", "c":65280}, {"n":"Lizard", "c":5923665}, {"n":"Cactus", "c":6516567}, {"n":"Kakapo", "c":4878371}, {"n":"Wet Moss", "c":4018729}, {"n":"Tree Moss", "c":6659378}, {"n":"Lime Rind", "c":3297047}, {"n":"Flight Jacket", "c":8357752}, {"n":"Green Mist", "c":12381585}, {"n":"Holly", "c":4751892}, {"n":"Mtn Dew Bottle", "c":5732922}, {"n":"Seaweed Roll", "c":7635561}, {"n":"Neon Green", "c":8647980}, {"n":"Lichen", "c":12638639}, {"n":"Guacamole", "c":10934149}, {"n":"Pond Scum", "c":6848090}, {"n":"Douglas Fir", "c":4153387}, {"n":"Royal Palm", "c":4155430}, {"n":"Seaweed", "c":6582110}, {"n":"Noble Fir", "c":4680244}, {"n":"Green Led", "c":6159370}, {"n":"Spinach", "c":4414774}, {"n":"Frog", "c":8699498}, {"n":"Emerald", "c":6003812}, {"n":"Circuit Board", "c":3827241}, {"n":"Sapgreen", "c":3178516}, {"n":"Pool Table", "c":3258701}, {"n":"Leaf", "c":5615162}, {"n":"Grass", "c":5094707}, {"n":"Snake", "c":5860438}, {"n":"100 Euro", "c":8832636}, {"n":"Night Vision", "c":8113264}, {"n":"Purple", "c":10494192}, {"n":"Purple", "c":10170623}, {"n":"Purple", "c":9514222}, {"n":"Purple", "c":8201933}, {"n":"Purple", "c":11141375}, {"n":"Purple", "c":8388736}, {"n":"Turnip", "c":10964139}, {"n":"Violet", "c":9395865}, {"n":"Eggplant", "c":8480391}, {"n":"Grape", "c":13369599}, {"n":"Wild Violet", "c":8522683}, {"n":"Concord Grape", "c":6685080}, {"n":"Garden Plum", "c":7431037}, {"n":"Purple Fish", "c":11694758}, {"n":"Ultramarine Violet", "c":6038638}, {"n":"Purple Rose", "c":6172025}, {"n":"Sea Urchin", "c":6830686}, {"n":"Cobalt Violet Deep", "c":9511326}, {"n":"Plum", "c":9135755}, {"n":"Dark Orchid", "c":10040013}, {"n":"Violet Flower", "c":12541951}, {"n":"Purple Candy", "c":12427467}, {"n":"Deep Purple", "c":5577355}, {"n":"Thistle", "c":11882652}, {"n":"Dark Purple", "c":8855416}, {"n":"Purple Ink", "c":10251160}, {"n":"Orchid", "c":14381275}, {"n":"True Purple", "c":10027161}, {"n":"Darkmagenta", "c":9109643}, {"n":"Harold\'s Crayon", "c":11935876}, {"n":"Purple Rain", "c":6898825}, {"n":"Gold", "c":16766720}];
-
- internal const LAME_DELAY_START:Number=2258;
-
- internal const LAME_DELAY_END:Number=1000;
-
- internal const BUFFER_SIZE:int=4410;
-
- internal var currentColors:*;
-
- internal var respackBytes:flash.utils.ByteArray;
-
- internal var respack:ResourcePack;
-
- internal var _settings:HuesSettings;
-
- internal var builtIn:BuiltResourcePack;
-
- internal var resourceManager:ResourceManager;
-
- internal var outputSound:flash.media.Sound;
-
- internal var inputSound:flash.media.Sound;
-
- internal var samplePosition:*=0;
-
- internal var samples:*=-1;
-
- internal var lastBeat:*=-123;
-
- internal var beatLength:*=-1;
-
- internal var currentSong:*;
-
- internal var currentImage:*;
-
- internal var destBMD:flash.display.BitmapData;
-
- internal var colorBMD:flash.display.BitmapData;
-
- internal var destBM:flash.display.Bitmap;
-
- internal var colorBM:flash.display.Bitmap;
-
- public var whiteOverlay:flash.display.MovieClip;
-
- internal var blackOverlay:flash.display.MovieClip;
-
- internal var blackOverlayColorTransform:flash.geom.ColorTransform;
-
- internal var doBlackout:*=false;
-
- internal var doWhiteout:*=false;
-
- internal var imageHolder:flash.display.MovieClip;
-
- internal var blurFilter:flash.filters.BlurFilter;
-
- internal var blurMultiplier:Number=1;
-
- internal var blurDecayMultiplier:Number=1;
-
- internal var songIndex:*=-1;
-
- internal var colorIndex:*=0;
-
- internal var imageIndex:*=-1;
-
- internal var animationFrame:*=0;
-
- internal var _isFullAuto:*=true;
-
- internal var currentVolume:*=70;
-
- internal var volumeMuted:*=false;
-
- internal var loopCount:*=0;
-
- public var retroui:flash.display.MovieClip;
-
- public var window:RetroWindow;
-
- public var imageContainer:flash.display.MovieClip;
-
- public var overlayContainer:flash.display.MovieClip;
-
- internal var uiArray:Array;
-
- internal var doRandom:Boolean=false;
-
- internal var currentUIIndex:*=2;
-
- internal var oldAutoSong:*=null;
-
- internal var beaterId:*;
-
- internal var lastSC:*=0;
-
- internal var buildupDiff:*=0;
-
- internal var soundChannel:flash.media.SoundChannel;
-
- internal var _userInterface:HuesUserInterface;
-
- internal var doColorFade:*=false;
-
- internal var oldColor:*=16777215;
-
- internal var colorFadeStarted:*=0;
-
- internal var animTimeoutID:Number;
-
- internal var colorFadeLength:*=0;
-
- internal var tmpFrame:*=0;
-
- internal var lastImageArray:*;
-
- internal var lastSongArray:*;
-
- internal var tmp_slowdown:*=false;
-
- internal var fadeOut:*=false;
-
- internal var fadeDirection:*=false;
-
- internal var initDone:*=false;
-
- internal var lastKeyEvent:*=null;
-
- internal var files:flash.net.FileReferenceList;
-
- internal var loadedFiles:*=0;
-
- internal var testPosition:*=0;
-
- internal var sampleTargetArray:flash.utils.ByteArray;
-
- internal var position:Number=0;
-
- internal var playbackRate:Number=100;
-
- internal var doBuildup:*=false;
- }
-}
-
-