Merge pull request #44 from khaybarkhaybaryayahud/main

some games
This commit is contained in:
sky 2024-01-22 15:38:48 -05:00 committed by GitHub
commit 2a8f032e0f
92 changed files with 495224 additions and 0 deletions

161
ayun/asn1-parser.js Normal file
View File

@ -0,0 +1,161 @@
// Copyright 2018 AJ ONeal. All rights reserved
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
;(function (exports) {
'use strict';
if (!exports.ASN1) { exports.ASN1 = {}; }
if (!exports.Enc) { exports.Enc = {}; }
if (!exports.PEM) { exports.PEM = {}; }
var ASN1 = exports.ASN1;
var Enc = exports.Enc;
var PEM = exports.PEM;
//
// Parser
//
// Although I've only seen 9 max in https certificates themselves,
// but each domain list could have up to 100
ASN1.ELOOPN = 102;
ASN1.ELOOP = "uASN1.js Error: iterated over " + ASN1.ELOOPN + "+ elements (probably a malformed file)";
// I've seen https certificates go 29 deep
ASN1.EDEEPN = 60;
ASN1.EDEEP = "uASN1.js Error: element nested " + ASN1.EDEEPN + "+ layers deep (probably a malformed file)";
// Container Types are Sequence 0x30, Container Array? (0xA0, 0xA1)
// Value Types are Boolean 0x01, Integer 0x02, Null 0x05, Object ID 0x06, String 0x0C, 0x16, 0x13, 0x1e Value Array? (0x82)
// Bit String (0x03) and Octet String (0x04) may be values or containers
// Sometimes Bit String is used as a container (RSA Pub Spki)
ASN1.CTYPES = [ 0x30, 0x31, 0xa0, 0xa1 ];
ASN1.VTYPES = [ 0x01, 0x02, 0x05, 0x06, 0x0c, 0x82 ];
ASN1.parse = function parseAsn1Helper(buf) {
//var ws = ' ';
function parseAsn1(buf, depth, eager) {
if (depth.length >= ASN1.EDEEPN) { throw new Error(ASN1.EDEEP); }
var index = 2; // we know, at minimum, data starts after type (0) and lengthSize (1)
var asn1 = { type: buf[0], lengthSize: 0, length: buf[1] };
var child;
var iters = 0;
var adjust = 0;
var adjustedLen;
// Determine how many bytes the length uses, and what it is
if (0x80 & asn1.length) {
asn1.lengthSize = 0x7f & asn1.length;
// I think that buf->hex->int solves the problem of Endianness... not sure
asn1.length = parseInt(Enc.bufToHex(buf.slice(index, index + asn1.lengthSize)), 16);
index += asn1.lengthSize;
}
// High-order bit Integers have a leading 0x00 to signify that they are positive.
// Bit Streams use the first byte to signify padding, which x.509 doesn't use.
if (0x00 === buf[index] && (0x02 === asn1.type || 0x03 === asn1.type)) {
// However, 0x00 on its own is a valid number
if (asn1.length > 1) {
index += 1;
adjust = -1;
}
}
adjustedLen = asn1.length + adjust;
//console.warn(depth.join(ws) + '0x' + Enc.numToHex(asn1.type), index, 'len:', asn1.length, asn1);
function parseChildren(eager) {
asn1.children = [];
//console.warn('1 len:', (2 + asn1.lengthSize + asn1.length), 'idx:', index, 'clen:', 0);
while (iters < ASN1.ELOOPN && index < (2 + asn1.length + asn1.lengthSize)) {
iters += 1;
depth.length += 1;
child = parseAsn1(buf.slice(index, index + adjustedLen), depth, eager);
depth.length -= 1;
// The numbers don't match up exactly and I don't remember why...
// probably something with adjustedLen or some such, but the tests pass
index += (2 + child.lengthSize + child.length);
//console.warn('2 len:', (2 + asn1.lengthSize + asn1.length), 'idx:', index, 'clen:', (2 + child.lengthSize + child.length));
if (index > (2 + asn1.lengthSize + asn1.length)) {
if (!eager) { console.error(JSON.stringify(asn1, ASN1._replacer, 2)); }
throw new Error("Parse error: child value length (" + child.length
+ ") is greater than remaining parent length (" + (asn1.length - index)
+ " = " + asn1.length + " - " + index + ")");
}
asn1.children.push(child);
//console.warn(depth.join(ws) + '0x' + Enc.numToHex(asn1.type), index, 'len:', asn1.length, asn1);
}
if (index !== (2 + asn1.lengthSize + asn1.length)) {
//console.warn('index:', index, 'length:', (2 + asn1.lengthSize + asn1.length));
throw new Error("premature end-of-file");
}
if (iters >= ASN1.ELOOPN) { throw new Error(ASN1.ELOOP); }
delete asn1.value;
return asn1;
}
// Recurse into types that are _always_ containers
if (-1 !== ASN1.CTYPES.indexOf(asn1.type)) { return parseChildren(eager); }
// Return types that are _always_ values
asn1.value = buf.slice(index, index + adjustedLen);
if (-1 !== ASN1.VTYPES.indexOf(asn1.type)) { return asn1; }
// For ambigious / unknown types, recurse and return on failure
// (and return child array size to zero)
try { return parseChildren(true); }
catch(e) { asn1.children.length = 0; return asn1; }
}
var asn1 = parseAsn1(buf, []);
var len = buf.byteLength || buf.length;
if (len !== 2 + asn1.lengthSize + asn1.length) {
throw new Error("Length of buffer does not match length of ASN.1 sequence.");
}
return asn1;
};
ASN1._replacer = function (k, v) {
if ('type' === k) { return '0x' + Enc.numToHex(v); }
if (v && 'value' === k) { return '0x' + Enc.bufToHex(v.data || v); }
return v;
};
// don't replace the full parseBlock, if it exists
PEM.parseBlock = PEM.parseBlock || function (str) {
var der = str.split(/\n/).filter(function (line) {
return !/-----/.test(line);
}).join('');
return { der: Enc.base64ToBuf(der) };
};
Enc.base64ToBuf = function (b64) {
return Enc.binToBuf(atob(b64));
};
Enc.binToBuf = function (bin) {
var arr = bin.split('').map(function (ch) {
return ch.charCodeAt(0);
});
return 'undefined' !== typeof Uint8Array ? new Uint8Array(arr) : arr;
};
Enc.bufToHex = function (u8) {
var hex = [];
var i, h;
var len = (u8.byteLength || u8.length);
for (i = 0; i < len; i += 1) {
h = u8[i].toString(16);
if (h.length % 2) { h = '0' + h; }
hex.push(h);
}
return hex.join('').toLowerCase();
};
Enc.numToHex = function (d) {
d = d.toString(16);
if (d.length % 2) {
return '0' + d;
}
return d;
};
}('undefined' !== typeof window ? window : module.exports));

BIN
ayun/assets.epk Normal file

Binary file not shown.

341449
ayun/classes.js Normal file

File diff suppressed because one or more lines are too long

3
ayun/cover.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 6.6 KiB

84
ayun/favicon.ico Normal file
View File

@ -0,0 +1,84 @@
<!DOCTYPE html>
<html>
<head>
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-261780635-1"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-261780635-1');
</script>
<!-- Google Tag Manager -->
<script>(function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
'https://www.googletagmanager.com/gtm.js?id='+i+dl;f.parentNode.insertBefore(j,f);
})(window,document,'script','dataLayer','GTM-TG85STRS');</script>
<!-- End Google Tag Manager -->
<title>Play Minecraft Online! | Online mc</title>
<link rel="icon" href="/icon.png">
<meta name="description" content="Play Minecraft (eaglercraft) Online For Free!" />
<meta name="keywords" content="free minecraft online, eaglercraft, minecraft, onlinemc, Onlinemc, ayuncraft, Ayuncraft, mc, eaglercraftx, eaglercraft shaders, eaglercraft download,eaglercraft, eaglercraftx, minecraft, 1.8, 1.8.8, eaglercraft, eaglercraft minecraft, eaglercraft 3kh0, eaglercraft github, eaglercraft singleplayer, eaglercraft hacks, eaglercraft 1.8, eaglercraft server, eaglecraft game, eaglercraft alternative, eaglercraft alt links, eaglercraft asspixel, eaglercraft anarchy server, eaglercraft aeon, eaglercraft asspixel server ip, eaglercraft all links, eaglercraft aeon server, eaglercraft beta, eaglecraft boats, eaglecraft boats for sale, eaglecraft boats for sale bc, eaglecraft bedwars server, eaglecraft boats inc, eaglercraft beta servers, eaglercraft blocked, eaglercraft client, eaglecraft custom boats, eaglecraft campbell river, eaglercraft creative, eaglercraft commands, eaglercraft code, eaglercraft controls, eaglercraft com, eaglercraft download, eaglercraft discord, eaglercraft dupe glitch, eaglercraft download file, eaglercraft download google drive, eaglercraft dmca, eaglercraft deleted, eaglercraft down, eaglercraft embed, eaglercraft epk file download, eaglercraft end of stream, eaglecraft apk download, eaglecraft eagtek, eaglercraft exploit, eaglecraft eggs, eaglercraft epk world file download, eaglercraft file, eaglercraft file download, eaglercraft free, eaglercraft forums, eaglercraft fork, eaglercraft faq, eaglercraft fps, eaglecraft free download, eaglercraft github servers, eaglercraft google drive, eaglercraft github lax1dude, eaglercraft github download, eaglercraft github.io, eaglercraft google sites, minecraft, applet, singleplayer, replit, html5, chromebook, lax1dude, games, eaglercraft singleplayer, eagler, minecraft for free, free minecraft">
<meta property="og:title" content="Online mc" />
<meta property="og:image" content="/icon.png" />
<meta property="og:description" content="Online Minecraft" />
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/css/bootstrap.min.css">
<link href="/style.css" rel="stylesheet" />
<link href="/darkmode.css" rel="stylesheet" />
</head>
<body class="diff-bg">
<div id="loading"></div>
<div class="container container-table">
<div class="row vertical-center-row">
<div class="col-12">
</a>
<div class="d-flex justify-content-center">
</div>
<div class="col-12">
<div class="button-area d-block">
<h1 class="dark-text">404 Not Found!</h1>
<div class="splash-404">...</div>
<h3>&nbsp;</h3>
<a href="/">click here to go back to the main page!</a>
<img class="creeper ms-auto d-block" src="creeper.webp" alt=""/>
</div>
<div class="mc-footer d-flex justify-content-between">
<div class="footer-copyright text-end text-white">This website is not affiliated with Mojang.</div>
<div class="mc-dd">
</div>
</div>
</div>
</div>
</div>
</div>
<script defer src="/_vercel/insights/script.js"></script>
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/js/bootstrap.bundle.min.js"></script>
<script>
window.onload = function () {
//display loader on page load
$('#loading').fadeOut();
}
setTimeout(function () {
//display loader on page load
$('#loading').fadeOut();
}, 12 * 1000);
</script>
<script src="/darkmode.js"></script>
<script src="/splash-404.js"></script>
</body>
</html>

44
ayun/index.html Normal file
View File

@ -0,0 +1,44 @@
<!DOCTYPE html>
<html>
<head>
<title>ayuncraft</title>
<script type="text/javascript" src="jsbn/jsbn.js"></script>
<script type="text/javascript" src="jsbn/jsbn2.js"></script>
<script type="text/javascript" src="jsbn/base64.js"></script>
<script type="text/javascript" src="jsbn/prng4.js"></script>
<script type="text/javascript" src="jsbn/rng.js"></script>
<script type="text/javascript" src="jsbn/rsa.js"></script>
<script type="text/javascript" src="jsbn/rsa2.js"></script>
<script type="text/javascript" src="asn1-parser.js"></script>
<script type="text/javascript" src="music.js"></script>
<script type="text/javascript" src="classes.js?t=updateme0"></script>
<script type="text/javascript">
//window.eag_self_proxy=true;//set to true to set the default proxy to the current url
//window.eag_proxy_list="";//set to a string of comma-separated proxy ip:port combinations for a custom proxy list
var name="PixelCraft",motd="A public 1.5.2 server",ip="pixelcraft.me";
window.addEventListener("load", function(){
document.requestPointerLock=document.requestPointerLock||function(){};
document.exitPointerLock=document.exitPointerLock||function(){};
window.minecraftOpts = [
"game_frame","assets.epk",
btoa(atob("CgAACQAHc2VydmVycwoAAAABCAAKZm9yY2VkTU9URABtb3RkaGVyZQEAC2hpZGVBZGRyZXNzAQgAAmlwAGlwaGVyZQgABG5hbWUAbmFtZWhlcmUAAA==").replace("motdhere",String.fromCharCode(motd.length)+motd).replace("namehere",String.fromCharCode(name.length)+name).replace("iphere",String.fromCharCode(ip.length)+ip))
];
(function(){
var q = window.location.search;
if(typeof q === 'string' && q.startsWith("?")) {
q = new URLSearchParams(q);
var s = q.get("server");
if(s) window.minecraftOpts.push(s);
}
})();
main();
});
</script>
</head>
<body style="margin:0px;width:100vw;height:100vh;" id="game_frame">
</body>
</html>

71
ayun/jsbn/base64.js Normal file
View File

@ -0,0 +1,71 @@
var b64map="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
var b64padchar="=";
function hex2b64(h) {
var i;
var c;
var ret = "";
for(i = 0; i+3 <= h.length; i+=3) {
c = parseInt(h.substring(i,i+3),16);
ret += b64map.charAt(c >> 6) + b64map.charAt(c & 63);
}
if(i+1 == h.length) {
c = parseInt(h.substring(i,i+1),16);
ret += b64map.charAt(c << 2);
}
else if(i+2 == h.length) {
c = parseInt(h.substring(i,i+2),16);
ret += b64map.charAt(c >> 2) + b64map.charAt((c & 3) << 4);
}
while((ret.length & 3) > 0) ret += b64padchar;
return ret;
}
// convert a base64 string to hex
function b64tohex(s) {
var ret = ""
var i;
var k = 0; // b64 state, 0-3
var slop;
for(i = 0; i < s.length; ++i) {
if(s.charAt(i) == b64padchar) break;
v = b64map.indexOf(s.charAt(i));
if(v < 0) continue;
if(k == 0) {
ret += int2char(v >> 2);
slop = v & 3;
k = 1;
}
else if(k == 1) {
ret += int2char((slop << 2) | (v >> 4));
slop = v & 0xf;
k = 2;
}
else if(k == 2) {
ret += int2char(slop);
ret += int2char(v >> 2);
slop = v & 3;
k = 3;
}
else {
ret += int2char((slop << 2) | (v >> 4));
ret += int2char(v & 0xf);
k = 0;
}
}
if(k == 1)
ret += int2char(slop << 2);
return ret;
}
// convert a base64 string to a byte/number array
function b64toBA(s) {
//piggyback on b64tohex for now, optimize later
var h = b64tohex(s);
var i;
var a = new Array();
for(i = 0; 2*i < h.length; ++i) {
a[i] = parseInt(h.substring(2*i,2*i+2),16);
}
return a;
}

559
ayun/jsbn/jsbn.js Normal file
View File

@ -0,0 +1,559 @@
// Copyright (c) 2005 Tom Wu
// All Rights Reserved.
// See "LICENSE" for details.
// Basic JavaScript BN library - subset useful for RSA encryption.
// Bits per digit
var dbits;
// JavaScript engine analysis
var canary = 0xdeadbeefcafe;
var j_lm = ((canary&0xffffff)==0xefcafe);
// (public) Constructor
function BigInteger(a,b,c) {
if(a != null)
if("number" == typeof a) this.fromNumber(a,b,c);
else if(b == null && "string" != typeof a) this.fromString(a,256);
else this.fromString(a,b);
}
// return new, unset BigInteger
function nbi() { return new BigInteger(null); }
// am: Compute w_j += (x*this_i), propagate carries,
// c is initial carry, returns final carry.
// c < 3*dvalue, x < 2*dvalue, this_i < dvalue
// We need to select the fastest one that works in this environment.
// am1: use a single mult and divide to get the high bits,
// max digit bits should be 26 because
// max internal value = 2*dvalue^2-2*dvalue (< 2^53)
function am1(i,x,w,j,c,n) {
while(--n >= 0) {
var v = x*this[i++]+w[j]+c;
c = Math.floor(v/0x4000000);
w[j++] = v&0x3ffffff;
}
return c;
}
// am2 avoids a big mult-and-extract completely.
// Max digit bits should be <= 30 because we do bitwise ops
// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)
function am2(i,x,w,j,c,n) {
var xl = x&0x7fff, xh = x>>15;
while(--n >= 0) {
var l = this[i]&0x7fff;
var h = this[i++]>>15;
var m = xh*l+h*xl;
l = xl*l+((m&0x7fff)<<15)+w[j]+(c&0x3fffffff);
c = (l>>>30)+(m>>>15)+xh*h+(c>>>30);
w[j++] = l&0x3fffffff;
}
return c;
}
// Alternately, set max digit bits to 28 since some
// browsers slow down when dealing with 32-bit numbers.
function am3(i,x,w,j,c,n) {
var xl = x&0x3fff, xh = x>>14;
while(--n >= 0) {
var l = this[i]&0x3fff;
var h = this[i++]>>14;
var m = xh*l+h*xl;
l = xl*l+((m&0x3fff)<<14)+w[j]+c;
c = (l>>28)+(m>>14)+xh*h;
w[j++] = l&0xfffffff;
}
return c;
}
if(j_lm && (navigator.appName == "Microsoft Internet Explorer")) {
BigInteger.prototype.am = am2;
dbits = 30;
}
else if(j_lm && (navigator.appName != "Netscape")) {
BigInteger.prototype.am = am1;
dbits = 26;
}
else { // Mozilla/Netscape seems to prefer am3
BigInteger.prototype.am = am3;
dbits = 28;
}
BigInteger.prototype.DB = dbits;
BigInteger.prototype.DM = ((1<<dbits)-1);
BigInteger.prototype.DV = (1<<dbits);
var BI_FP = 52;
BigInteger.prototype.FV = Math.pow(2,BI_FP);
BigInteger.prototype.F1 = BI_FP-dbits;
BigInteger.prototype.F2 = 2*dbits-BI_FP;
// Digit conversions
var BI_RM = "0123456789abcdefghijklmnopqrstuvwxyz";
var BI_RC = new Array();
var rr,vv;
rr = "0".charCodeAt(0);
for(vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;
rr = "a".charCodeAt(0);
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
rr = "A".charCodeAt(0);
for(vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;
function int2char(n) { return BI_RM.charAt(n); }
function intAt(s,i) {
var c = BI_RC[s.charCodeAt(i)];
return (c==null)?-1:c;
}
// (protected) copy this to r
function bnpCopyTo(r) {
for(var i = this.t-1; i >= 0; --i) r[i] = this[i];
r.t = this.t;
r.s = this.s;
}
// (protected) set from integer value x, -DV <= x < DV
function bnpFromInt(x) {
this.t = 1;
this.s = (x<0)?-1:0;
if(x > 0) this[0] = x;
else if(x < -1) this[0] = x+this.DV;
else this.t = 0;
}
// return bigint initialized to value
function nbv(i) { var r = nbi(); r.fromInt(i); return r; }
// (protected) set from string and radix
function bnpFromString(s,b) {
var k;
if(b == 16) k = 4;
else if(b == 8) k = 3;
else if(b == 256) k = 8; // byte array
else if(b == 2) k = 1;
else if(b == 32) k = 5;
else if(b == 4) k = 2;
else { this.fromRadix(s,b); return; }
this.t = 0;
this.s = 0;
var i = s.length, mi = false, sh = 0;
while(--i >= 0) {
var x = (k==8)?s[i]&0xff:intAt(s,i);
if(x < 0) {
if(s.charAt(i) == "-") mi = true;
continue;
}
mi = false;
if(sh == 0)
this[this.t++] = x;
else if(sh+k > this.DB) {
this[this.t-1] |= (x&((1<<(this.DB-sh))-1))<<sh;
this[this.t++] = (x>>(this.DB-sh));
}
else
this[this.t-1] |= x<<sh;
sh += k;
if(sh >= this.DB) sh -= this.DB;
}
if(k == 8 && (s[0]&0x80) != 0) {
this.s = -1;
if(sh > 0) this[this.t-1] |= ((1<<(this.DB-sh))-1)<<sh;
}
this.clamp();
if(mi) BigInteger.ZERO.subTo(this,this);
}
// (protected) clamp off excess high words
function bnpClamp() {
var c = this.s&this.DM;
while(this.t > 0 && this[this.t-1] == c) --this.t;
}
// (public) return string representation in given radix
function bnToString(b) {
if(this.s < 0) return "-"+this.negate().toString(b);
var k;
if(b == 16) k = 4;
else if(b == 8) k = 3;
else if(b == 2) k = 1;
else if(b == 32) k = 5;
else if(b == 4) k = 2;
else return this.toRadix(b);
var km = (1<<k)-1, d, m = false, r = "", i = this.t;
var p = this.DB-(i*this.DB)%k;
if(i-- > 0) {
if(p < this.DB && (d = this[i]>>p) > 0) { m = true; r = int2char(d); }
while(i >= 0) {
if(p < k) {
d = (this[i]&((1<<p)-1))<<(k-p);
d |= this[--i]>>(p+=this.DB-k);
}
else {
d = (this[i]>>(p-=k))&km;
if(p <= 0) { p += this.DB; --i; }
}
if(d > 0) m = true;
if(m) r += int2char(d);
}
}
return m?r:"0";
}
// (public) -this
function bnNegate() { var r = nbi(); BigInteger.ZERO.subTo(this,r); return r; }
// (public) |this|
function bnAbs() { return (this.s<0)?this.negate():this; }
// (public) return + if this > a, - if this < a, 0 if equal
function bnCompareTo(a) {
var r = this.s-a.s;
if(r != 0) return r;
var i = this.t;
r = i-a.t;
if(r != 0) return (this.s<0)?-r:r;
while(--i >= 0) if((r=this[i]-a[i]) != 0) return r;
return 0;
}
// returns bit length of the integer x
function nbits(x) {
var r = 1, t;
if((t=x>>>16) != 0) { x = t; r += 16; }
if((t=x>>8) != 0) { x = t; r += 8; }
if((t=x>>4) != 0) { x = t; r += 4; }
if((t=x>>2) != 0) { x = t; r += 2; }
if((t=x>>1) != 0) { x = t; r += 1; }
return r;
}
// (public) return the number of bits in "this"
function bnBitLength() {
if(this.t <= 0) return 0;
return this.DB*(this.t-1)+nbits(this[this.t-1]^(this.s&this.DM));
}
// (protected) r = this << n*DB
function bnpDLShiftTo(n,r) {
var i;
for(i = this.t-1; i >= 0; --i) r[i+n] = this[i];
for(i = n-1; i >= 0; --i) r[i] = 0;
r.t = this.t+n;
r.s = this.s;
}
// (protected) r = this >> n*DB
function bnpDRShiftTo(n,r) {
for(var i = n; i < this.t; ++i) r[i-n] = this[i];
r.t = Math.max(this.t-n,0);
r.s = this.s;
}
// (protected) r = this << n
function bnpLShiftTo(n,r) {
var bs = n%this.DB;
var cbs = this.DB-bs;
var bm = (1<<cbs)-1;
var ds = Math.floor(n/this.DB), c = (this.s<<bs)&this.DM, i;
for(i = this.t-1; i >= 0; --i) {
r[i+ds+1] = (this[i]>>cbs)|c;
c = (this[i]&bm)<<bs;
}
for(i = ds-1; i >= 0; --i) r[i] = 0;
r[ds] = c;
r.t = this.t+ds+1;
r.s = this.s;
r.clamp();
}
// (protected) r = this >> n
function bnpRShiftTo(n,r) {
r.s = this.s;
var ds = Math.floor(n/this.DB);
if(ds >= this.t) { r.t = 0; return; }
var bs = n%this.DB;
var cbs = this.DB-bs;
var bm = (1<<bs)-1;
r[0] = this[ds]>>bs;
for(var i = ds+1; i < this.t; ++i) {
r[i-ds-1] |= (this[i]&bm)<<cbs;
r[i-ds] = this[i]>>bs;
}
if(bs > 0) r[this.t-ds-1] |= (this.s&bm)<<cbs;
r.t = this.t-ds;
r.clamp();
}
// (protected) r = this - a
function bnpSubTo(a,r) {
var i = 0, c = 0, m = Math.min(a.t,this.t);
while(i < m) {
c += this[i]-a[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
if(a.t < this.t) {
c -= a.s;
while(i < this.t) {
c += this[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
c += this.s;
}
else {
c += this.s;
while(i < a.t) {
c -= a[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
c -= a.s;
}
r.s = (c<0)?-1:0;
if(c < -1) r[i++] = this.DV+c;
else if(c > 0) r[i++] = c;
r.t = i;
r.clamp();
}
// (protected) r = this * a, r != this,a (HAC 14.12)
// "this" should be the larger one if appropriate.
function bnpMultiplyTo(a,r) {
var x = this.abs(), y = a.abs();
var i = x.t;
r.t = i+y.t;
while(--i >= 0) r[i] = 0;
for(i = 0; i < y.t; ++i) r[i+x.t] = x.am(0,y[i],r,i,0,x.t);
r.s = 0;
r.clamp();
if(this.s != a.s) BigInteger.ZERO.subTo(r,r);
}
// (protected) r = this^2, r != this (HAC 14.16)
function bnpSquareTo(r) {
var x = this.abs();
var i = r.t = 2*x.t;
while(--i >= 0) r[i] = 0;
for(i = 0; i < x.t-1; ++i) {
var c = x.am(i,x[i],r,2*i,0,1);
if((r[i+x.t]+=x.am(i+1,2*x[i],r,2*i+1,c,x.t-i-1)) >= x.DV) {
r[i+x.t] -= x.DV;
r[i+x.t+1] = 1;
}
}
if(r.t > 0) r[r.t-1] += x.am(i,x[i],r,2*i,0,1);
r.s = 0;
r.clamp();
}
// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)
// r != q, this != m. q or r may be null.
function bnpDivRemTo(m,q,r) {
var pm = m.abs();
if(pm.t <= 0) return;
var pt = this.abs();
if(pt.t < pm.t) {
if(q != null) q.fromInt(0);
if(r != null) this.copyTo(r);
return;
}
if(r == null) r = nbi();
var y = nbi(), ts = this.s, ms = m.s;
var nsh = this.DB-nbits(pm[pm.t-1]); // normalize modulus
if(nsh > 0) { pm.lShiftTo(nsh,y); pt.lShiftTo(nsh,r); }
else { pm.copyTo(y); pt.copyTo(r); }
var ys = y.t;
var y0 = y[ys-1];
if(y0 == 0) return;
var yt = y0*(1<<this.F1)+((ys>1)?y[ys-2]>>this.F2:0);
var d1 = this.FV/yt, d2 = (1<<this.F1)/yt, e = 1<<this.F2;
var i = r.t, j = i-ys, t = (q==null)?nbi():q;
y.dlShiftTo(j,t);
if(r.compareTo(t) >= 0) {
r[r.t++] = 1;
r.subTo(t,r);
}
BigInteger.ONE.dlShiftTo(ys,t);
t.subTo(y,y); // "negative" y so we can replace sub with am later
while(y.t < ys) y[y.t++] = 0;
while(--j >= 0) {
// Estimate quotient digit
var qd = (r[--i]==y0)?this.DM:Math.floor(r[i]*d1+(r[i-1]+e)*d2);
if((r[i]+=y.am(0,qd,r,j,0,ys)) < qd) { // Try it out
y.dlShiftTo(j,t);
r.subTo(t,r);
while(r[i] < --qd) r.subTo(t,r);
}
}
if(q != null) {
r.drShiftTo(ys,q);
if(ts != ms) BigInteger.ZERO.subTo(q,q);
}
r.t = ys;
r.clamp();
if(nsh > 0) r.rShiftTo(nsh,r); // Denormalize remainder
if(ts < 0) BigInteger.ZERO.subTo(r,r);
}
// (public) this mod a
function bnMod(a) {
var r = nbi();
this.abs().divRemTo(a,null,r);
if(this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r,r);
return r;
}
// Modular reduction using "classic" algorithm
function Classic(m) { this.m = m; }
function cConvert(x) {
if(x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);
else return x;
}
function cRevert(x) { return x; }
function cReduce(x) { x.divRemTo(this.m,null,x); }
function cMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
function cSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
Classic.prototype.convert = cConvert;
Classic.prototype.revert = cRevert;
Classic.prototype.reduce = cReduce;
Classic.prototype.mulTo = cMulTo;
Classic.prototype.sqrTo = cSqrTo;
// (protected) return "-1/this % 2^DB"; useful for Mont. reduction
// justification:
// xy == 1 (mod m)
// xy = 1+km
// xy(2-xy) = (1+km)(1-km)
// x[y(2-xy)] = 1-k^2m^2
// x[y(2-xy)] == 1 (mod m^2)
// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2
// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.
// JS multiply "overflows" differently from C/C++, so care is needed here.
function bnpInvDigit() {
if(this.t < 1) return 0;
var x = this[0];
if((x&1) == 0) return 0;
var y = x&3; // y == 1/x mod 2^2
y = (y*(2-(x&0xf)*y))&0xf; // y == 1/x mod 2^4
y = (y*(2-(x&0xff)*y))&0xff; // y == 1/x mod 2^8
y = (y*(2-(((x&0xffff)*y)&0xffff)))&0xffff; // y == 1/x mod 2^16
// last step - calculate inverse mod DV directly;
// assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints
y = (y*(2-x*y%this.DV))%this.DV; // y == 1/x mod 2^dbits
// we really want the negative inverse, and -DV < y < DV
return (y>0)?this.DV-y:-y;
}
// Montgomery reduction
function Montgomery(m) {
this.m = m;
this.mp = m.invDigit();
this.mpl = this.mp&0x7fff;
this.mph = this.mp>>15;
this.um = (1<<(m.DB-15))-1;
this.mt2 = 2*m.t;
}
// xR mod m
function montConvert(x) {
var r = nbi();
x.abs().dlShiftTo(this.m.t,r);
r.divRemTo(this.m,null,r);
if(x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r,r);
return r;
}
// x/R mod m
function montRevert(x) {
var r = nbi();
x.copyTo(r);
this.reduce(r);
return r;
}
// x = x/R mod m (HAC 14.32)
function montReduce(x) {
while(x.t <= this.mt2) // pad x so am has enough room later
x[x.t++] = 0;
for(var i = 0; i < this.m.t; ++i) {
// faster way of calculating u0 = x[i]*mp mod DV
var j = x[i]&0x7fff;
var u0 = (j*this.mpl+(((j*this.mph+(x[i]>>15)*this.mpl)&this.um)<<15))&x.DM;
// use am to combine the multiply-shift-add into one call
j = i+this.m.t;
x[j] += this.m.am(0,u0,x,i,0,this.m.t);
// propagate carry
while(x[j] >= x.DV) { x[j] -= x.DV; x[++j]++; }
}
x.clamp();
x.drShiftTo(this.m.t,x);
if(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
}
// r = "x^2/R mod m"; x != r
function montSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
// r = "xy/R mod m"; x,y != r
function montMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
Montgomery.prototype.convert = montConvert;
Montgomery.prototype.revert = montRevert;
Montgomery.prototype.reduce = montReduce;
Montgomery.prototype.mulTo = montMulTo;
Montgomery.prototype.sqrTo = montSqrTo;
// (protected) true iff this is even
function bnpIsEven() { return ((this.t>0)?(this[0]&1):this.s) == 0; }
// (protected) this^e, e < 2^32, doing sqr and mul with "r" (HAC 14.79)
function bnpExp(e,z) {
if(e > 0xffffffff || e < 1) return BigInteger.ONE;
var r = nbi(), r2 = nbi(), g = z.convert(this), i = nbits(e)-1;
g.copyTo(r);
while(--i >= 0) {
z.sqrTo(r,r2);
if((e&(1<<i)) > 0) z.mulTo(r2,g,r);
else { var t = r; r = r2; r2 = t; }
}
return z.revert(r);
}
// (public) this^e % m, 0 <= e < 2^32
function bnModPowInt(e,m) {
var z;
if(e < 256 || m.isEven()) z = new Classic(m); else z = new Montgomery(m);
return this.exp(e,z);
}
// protected
BigInteger.prototype.copyTo = bnpCopyTo;
BigInteger.prototype.fromInt = bnpFromInt;
BigInteger.prototype.fromString = bnpFromString;
BigInteger.prototype.clamp = bnpClamp;
BigInteger.prototype.dlShiftTo = bnpDLShiftTo;
BigInteger.prototype.drShiftTo = bnpDRShiftTo;
BigInteger.prototype.lShiftTo = bnpLShiftTo;
BigInteger.prototype.rShiftTo = bnpRShiftTo;
BigInteger.prototype.subTo = bnpSubTo;
BigInteger.prototype.multiplyTo = bnpMultiplyTo;
BigInteger.prototype.squareTo = bnpSquareTo;
BigInteger.prototype.divRemTo = bnpDivRemTo;
BigInteger.prototype.invDigit = bnpInvDigit;
BigInteger.prototype.isEven = bnpIsEven;
BigInteger.prototype.exp = bnpExp;
// public
BigInteger.prototype.toString = bnToString;
BigInteger.prototype.negate = bnNegate;
BigInteger.prototype.abs = bnAbs;
BigInteger.prototype.compareTo = bnCompareTo;
BigInteger.prototype.bitLength = bnBitLength;
BigInteger.prototype.mod = bnMod;
BigInteger.prototype.modPowInt = bnModPowInt;
// "constants"
BigInteger.ZERO = nbv(0);
BigInteger.ONE = nbv(1);

656
ayun/jsbn/jsbn2.js Normal file
View File

@ -0,0 +1,656 @@
// Copyright (c) 2005-2009 Tom Wu
// All Rights Reserved.
// See "LICENSE" for details.
// Extended JavaScript BN functions, required for RSA private ops.
// Version 1.1: new BigInteger("0", 10) returns "proper" zero
// Version 1.2: square() API, isProbablePrime fix
// (public)
function bnClone() { var r = nbi(); this.copyTo(r); return r; }
// (public) return value as integer
function bnIntValue() {
if(this.s < 0) {
if(this.t == 1) return this[0]-this.DV;
else if(this.t == 0) return -1;
}
else if(this.t == 1) return this[0];
else if(this.t == 0) return 0;
// assumes 16 < DB < 32
return ((this[1]&((1<<(32-this.DB))-1))<<this.DB)|this[0];
}
// (public) return value as byte
function bnByteValue() { return (this.t==0)?this.s:(this[0]<<24)>>24; }
// (public) return value as short (assumes DB>=16)
function bnShortValue() { return (this.t==0)?this.s:(this[0]<<16)>>16; }
// (protected) return x s.t. r^x < DV
function bnpChunkSize(r) { return Math.floor(Math.LN2*this.DB/Math.log(r)); }
// (public) 0 if this == 0, 1 if this > 0
function bnSigNum() {
if(this.s < 0) return -1;
else if(this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;
else return 1;
}
// (protected) convert to radix string
function bnpToRadix(b) {
if(b == null) b = 10;
if(this.signum() == 0 || b < 2 || b > 36) return "0";
var cs = this.chunkSize(b);
var a = Math.pow(b,cs);
var d = nbv(a), y = nbi(), z = nbi(), r = "";
this.divRemTo(d,y,z);
while(y.signum() > 0) {
r = (a+z.intValue()).toString(b).substr(1) + r;
y.divRemTo(d,y,z);
}
return z.intValue().toString(b) + r;
}
// (protected) convert from radix string
function bnpFromRadix(s,b) {
this.fromInt(0);
if(b == null) b = 10;
var cs = this.chunkSize(b);
var d = Math.pow(b,cs), mi = false, j = 0, w = 0;
for(var i = 0; i < s.length; ++i) {
var x = intAt(s,i);
if(x < 0) {
if(s.charAt(i) == "-" && this.signum() == 0) mi = true;
continue;
}
w = b*w+x;
if(++j >= cs) {
this.dMultiply(d);
this.dAddOffset(w,0);
j = 0;
w = 0;
}
}
if(j > 0) {
this.dMultiply(Math.pow(b,j));
this.dAddOffset(w,0);
}
if(mi) BigInteger.ZERO.subTo(this,this);
}
// (protected) alternate constructor
function bnpFromNumber(a,b,c) {
if("number" == typeof b) {
// new BigInteger(int,int,RNG)
if(a < 2) this.fromInt(1);
else {
this.fromNumber(a,c);
if(!this.testBit(a-1)) // force MSB set
this.bitwiseTo(BigInteger.ONE.shiftLeft(a-1),op_or,this);
if(this.isEven()) this.dAddOffset(1,0); // force odd
while(!this.isProbablePrime(b)) {
this.dAddOffset(2,0);
if(this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a-1),this);
}
}
}
else {
// new BigInteger(int,RNG)
var x = new Array(), t = a&7;
x.length = (a>>3)+1;
b.nextBytes(x);
if(t > 0) x[0] &= ((1<<t)-1); else x[0] = 0;
this.fromString(x,256);
}
}
// (public) convert to bigendian byte array
function bnToByteArray() {
var i = this.t, r = new Array();
r[0] = this.s;
var p = this.DB-(i*this.DB)%8, d, k = 0;
if(i-- > 0) {
if(p < this.DB && (d = this[i]>>p) != (this.s&this.DM)>>p)
r[k++] = d|(this.s<<(this.DB-p));
while(i >= 0) {
if(p < 8) {
d = (this[i]&((1<<p)-1))<<(8-p);
d |= this[--i]>>(p+=this.DB-8);
}
else {
d = (this[i]>>(p-=8))&0xff;
if(p <= 0) { p += this.DB; --i; }
}
if((d&0x80) != 0) d |= -256;
if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;
if(k > 0 || d != this.s) r[k++] = d;
}
}
return r;
}
function bnEquals(a) { return(this.compareTo(a)==0); }
function bnMin(a) { return(this.compareTo(a)<0)?this:a; }
function bnMax(a) { return(this.compareTo(a)>0)?this:a; }
// (protected) r = this op a (bitwise)
function bnpBitwiseTo(a,op,r) {
var i, f, m = Math.min(a.t,this.t);
for(i = 0; i < m; ++i) r[i] = op(this[i],a[i]);
if(a.t < this.t) {
f = a.s&this.DM;
for(i = m; i < this.t; ++i) r[i] = op(this[i],f);
r.t = this.t;
}
else {
f = this.s&this.DM;
for(i = m; i < a.t; ++i) r[i] = op(f,a[i]);
r.t = a.t;
}
r.s = op(this.s,a.s);
r.clamp();
}
// (public) this & a
function op_and(x,y) { return x&y; }
function bnAnd(a) { var r = nbi(); this.bitwiseTo(a,op_and,r); return r; }
// (public) this | a
function op_or(x,y) { return x|y; }
function bnOr(a) { var r = nbi(); this.bitwiseTo(a,op_or,r); return r; }
// (public) this ^ a
function op_xor(x,y) { return x^y; }
function bnXor(a) { var r = nbi(); this.bitwiseTo(a,op_xor,r); return r; }
// (public) this & ~a
function op_andnot(x,y) { return x&~y; }
function bnAndNot(a) { var r = nbi(); this.bitwiseTo(a,op_andnot,r); return r; }
// (public) ~this
function bnNot() {
var r = nbi();
for(var i = 0; i < this.t; ++i) r[i] = this.DM&~this[i];
r.t = this.t;
r.s = ~this.s;
return r;
}
// (public) this << n
function bnShiftLeft(n) {
var r = nbi();
if(n < 0) this.rShiftTo(-n,r); else this.lShiftTo(n,r);
return r;
}
// (public) this >> n
function bnShiftRight(n) {
var r = nbi();
if(n < 0) this.lShiftTo(-n,r); else this.rShiftTo(n,r);
return r;
}
// return index of lowest 1-bit in x, x < 2^31
function lbit(x) {
if(x == 0) return -1;
var r = 0;
if((x&0xffff) == 0) { x >>= 16; r += 16; }
if((x&0xff) == 0) { x >>= 8; r += 8; }
if((x&0xf) == 0) { x >>= 4; r += 4; }
if((x&3) == 0) { x >>= 2; r += 2; }
if((x&1) == 0) ++r;
return r;
}
// (public) returns index of lowest 1-bit (or -1 if none)
function bnGetLowestSetBit() {
for(var i = 0; i < this.t; ++i)
if(this[i] != 0) return i*this.DB+lbit(this[i]);
if(this.s < 0) return this.t*this.DB;
return -1;
}
// return number of 1 bits in x
function cbit(x) {
var r = 0;
while(x != 0) { x &= x-1; ++r; }
return r;
}
// (public) return number of set bits
function bnBitCount() {
var r = 0, x = this.s&this.DM;
for(var i = 0; i < this.t; ++i) r += cbit(this[i]^x);
return r;
}
// (public) true iff nth bit is set
function bnTestBit(n) {
var j = Math.floor(n/this.DB);
if(j >= this.t) return(this.s!=0);
return((this[j]&(1<<(n%this.DB)))!=0);
}
// (protected) this op (1<<n)
function bnpChangeBit(n,op) {
var r = BigInteger.ONE.shiftLeft(n);
this.bitwiseTo(r,op,r);
return r;
}
// (public) this | (1<<n)
function bnSetBit(n) { return this.changeBit(n,op_or); }
// (public) this & ~(1<<n)
function bnClearBit(n) { return this.changeBit(n,op_andnot); }
// (public) this ^ (1<<n)
function bnFlipBit(n) { return this.changeBit(n,op_xor); }
// (protected) r = this + a
function bnpAddTo(a,r) {
var i = 0, c = 0, m = Math.min(a.t,this.t);
while(i < m) {
c += this[i]+a[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
if(a.t < this.t) {
c += a.s;
while(i < this.t) {
c += this[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
c += this.s;
}
else {
c += this.s;
while(i < a.t) {
c += a[i];
r[i++] = c&this.DM;
c >>= this.DB;
}
c += a.s;
}
r.s = (c<0)?-1:0;
if(c > 0) r[i++] = c;
else if(c < -1) r[i++] = this.DV+c;
r.t = i;
r.clamp();
}
// (public) this + a
function bnAdd(a) { var r = nbi(); this.addTo(a,r); return r; }
// (public) this - a
function bnSubtract(a) { var r = nbi(); this.subTo(a,r); return r; }
// (public) this * a
function bnMultiply(a) { var r = nbi(); this.multiplyTo(a,r); return r; }
// (public) this^2
function bnSquare() { var r = nbi(); this.squareTo(r); return r; }
// (public) this / a
function bnDivide(a) { var r = nbi(); this.divRemTo(a,r,null); return r; }
// (public) this % a
function bnRemainder(a) { var r = nbi(); this.divRemTo(a,null,r); return r; }
// (public) [this/a,this%a]
function bnDivideAndRemainder(a) {
var q = nbi(), r = nbi();
this.divRemTo(a,q,r);
return new Array(q,r);
}
// (protected) this *= n, this >= 0, 1 < n < DV
function bnpDMultiply(n) {
this[this.t] = this.am(0,n-1,this,0,0,this.t);
++this.t;
this.clamp();
}
// (protected) this += n << w words, this >= 0
function bnpDAddOffset(n,w) {
if(n == 0) return;
while(this.t <= w) this[this.t++] = 0;
this[w] += n;
while(this[w] >= this.DV) {
this[w] -= this.DV;
if(++w >= this.t) this[this.t++] = 0;
++this[w];
}
}
// A "null" reducer
function NullExp() {}
function nNop(x) { return x; }
function nMulTo(x,y,r) { x.multiplyTo(y,r); }
function nSqrTo(x,r) { x.squareTo(r); }
NullExp.prototype.convert = nNop;
NullExp.prototype.revert = nNop;
NullExp.prototype.mulTo = nMulTo;
NullExp.prototype.sqrTo = nSqrTo;
// (public) this^e
function bnPow(e) { return this.exp(e,new NullExp()); }
// (protected) r = lower n words of "this * a", a.t <= n
// "this" should be the larger one if appropriate.
function bnpMultiplyLowerTo(a,n,r) {
var i = Math.min(this.t+a.t,n);
r.s = 0; // assumes a,this >= 0
r.t = i;
while(i > 0) r[--i] = 0;
var j;
for(j = r.t-this.t; i < j; ++i) r[i+this.t] = this.am(0,a[i],r,i,0,this.t);
for(j = Math.min(a.t,n); i < j; ++i) this.am(0,a[i],r,i,0,n-i);
r.clamp();
}
// (protected) r = "this * a" without lower n words, n > 0
// "this" should be the larger one if appropriate.
function bnpMultiplyUpperTo(a,n,r) {
--n;
var i = r.t = this.t+a.t-n;
r.s = 0; // assumes a,this >= 0
while(--i >= 0) r[i] = 0;
for(i = Math.max(n-this.t,0); i < a.t; ++i)
r[this.t+i-n] = this.am(n-i,a[i],r,0,0,this.t+i-n);
r.clamp();
r.drShiftTo(1,r);
}
// Barrett modular reduction
function Barrett(m) {
// setup Barrett
this.r2 = nbi();
this.q3 = nbi();
BigInteger.ONE.dlShiftTo(2*m.t,this.r2);
this.mu = this.r2.divide(m);
this.m = m;
}
function barrettConvert(x) {
if(x.s < 0 || x.t > 2*this.m.t) return x.mod(this.m);
else if(x.compareTo(this.m) < 0) return x;
else { var r = nbi(); x.copyTo(r); this.reduce(r); return r; }
}
function barrettRevert(x) { return x; }
// x = x mod m (HAC 14.42)
function barrettReduce(x) {
x.drShiftTo(this.m.t-1,this.r2);
if(x.t > this.m.t+1) { x.t = this.m.t+1; x.clamp(); }
this.mu.multiplyUpperTo(this.r2,this.m.t+1,this.q3);
this.m.multiplyLowerTo(this.q3,this.m.t+1,this.r2);
while(x.compareTo(this.r2) < 0) x.dAddOffset(1,this.m.t+1);
x.subTo(this.r2,x);
while(x.compareTo(this.m) >= 0) x.subTo(this.m,x);
}
// r = x^2 mod m; x != r
function barrettSqrTo(x,r) { x.squareTo(r); this.reduce(r); }
// r = x*y mod m; x,y != r
function barrettMulTo(x,y,r) { x.multiplyTo(y,r); this.reduce(r); }
Barrett.prototype.convert = barrettConvert;
Barrett.prototype.revert = barrettRevert;
Barrett.prototype.reduce = barrettReduce;
Barrett.prototype.mulTo = barrettMulTo;
Barrett.prototype.sqrTo = barrettSqrTo;
// (public) this^e % m (HAC 14.85)
function bnModPow(e,m) {
var i = e.bitLength(), k, r = nbv(1), z;
if(i <= 0) return r;
else if(i < 18) k = 1;
else if(i < 48) k = 3;
else if(i < 144) k = 4;
else if(i < 768) k = 5;
else k = 6;
if(i < 8)
z = new Classic(m);
else if(m.isEven())
z = new Barrett(m);
else
z = new Montgomery(m);
// precomputation
var g = new Array(), n = 3, k1 = k-1, km = (1<<k)-1;
g[1] = z.convert(this);
if(k > 1) {
var g2 = nbi();
z.sqrTo(g[1],g2);
while(n <= km) {
g[n] = nbi();
z.mulTo(g2,g[n-2],g[n]);
n += 2;
}
}
var j = e.t-1, w, is1 = true, r2 = nbi(), t;
i = nbits(e[j])-1;
while(j >= 0) {
if(i >= k1) w = (e[j]>>(i-k1))&km;
else {
w = (e[j]&((1<<(i+1))-1))<<(k1-i);
if(j > 0) w |= e[j-1]>>(this.DB+i-k1);
}
n = k;
while((w&1) == 0) { w >>= 1; --n; }
if((i -= n) < 0) { i += this.DB; --j; }
if(is1) { // ret == 1, don't bother squaring or multiplying it
g[w].copyTo(r);
is1 = false;
}
else {
while(n > 1) { z.sqrTo(r,r2); z.sqrTo(r2,r); n -= 2; }
if(n > 0) z.sqrTo(r,r2); else { t = r; r = r2; r2 = t; }
z.mulTo(r2,g[w],r);
}
while(j >= 0 && (e[j]&(1<<i)) == 0) {
z.sqrTo(r,r2); t = r; r = r2; r2 = t;
if(--i < 0) { i = this.DB-1; --j; }
}
}
return z.revert(r);
}
// (public) gcd(this,a) (HAC 14.54)
function bnGCD(a) {
var x = (this.s<0)?this.negate():this.clone();
var y = (a.s<0)?a.negate():a.clone();
if(x.compareTo(y) < 0) { var t = x; x = y; y = t; }
var i = x.getLowestSetBit(), g = y.getLowestSetBit();
if(g < 0) return x;
if(i < g) g = i;
if(g > 0) {
x.rShiftTo(g,x);
y.rShiftTo(g,y);
}
while(x.signum() > 0) {
if((i = x.getLowestSetBit()) > 0) x.rShiftTo(i,x);
if((i = y.getLowestSetBit()) > 0) y.rShiftTo(i,y);
if(x.compareTo(y) >= 0) {
x.subTo(y,x);
x.rShiftTo(1,x);
}
else {
y.subTo(x,y);
y.rShiftTo(1,y);
}
}
if(g > 0) y.lShiftTo(g,y);
return y;
}
// (protected) this % n, n < 2^26
function bnpModInt(n) {
if(n <= 0) return 0;
var d = this.DV%n, r = (this.s<0)?n-1:0;
if(this.t > 0)
if(d == 0) r = this[0]%n;
else for(var i = this.t-1; i >= 0; --i) r = (d*r+this[i])%n;
return r;
}
// (public) 1/this % m (HAC 14.61)
function bnModInverse(m) {
var ac = m.isEven();
if((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;
var u = m.clone(), v = this.clone();
var a = nbv(1), b = nbv(0), c = nbv(0), d = nbv(1);
while(u.signum() != 0) {
while(u.isEven()) {
u.rShiftTo(1,u);
if(ac) {
if(!a.isEven() || !b.isEven()) { a.addTo(this,a); b.subTo(m,b); }
a.rShiftTo(1,a);
}
else if(!b.isEven()) b.subTo(m,b);
b.rShiftTo(1,b);
}
while(v.isEven()) {
v.rShiftTo(1,v);
if(ac) {
if(!c.isEven() || !d.isEven()) { c.addTo(this,c); d.subTo(m,d); }
c.rShiftTo(1,c);
}
else if(!d.isEven()) d.subTo(m,d);
d.rShiftTo(1,d);
}
if(u.compareTo(v) >= 0) {
u.subTo(v,u);
if(ac) a.subTo(c,a);
b.subTo(d,b);
}
else {
v.subTo(u,v);
if(ac) c.subTo(a,c);
d.subTo(b,d);
}
}
if(v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;
if(d.compareTo(m) >= 0) return d.subtract(m);
if(d.signum() < 0) d.addTo(m,d); else return d;
if(d.signum() < 0) return d.add(m); else return d;
}
var lowprimes = [2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101,103,107,109,113,127,131,137,139,149,151,157,163,167,173,179,181,191,193,197,199,211,223,227,229,233,239,241,251,257,263,269,271,277,281,283,293,307,311,313,317,331,337,347,349,353,359,367,373,379,383,389,397,401,409,419,421,431,433,439,443,449,457,461,463,467,479,487,491,499,503,509,521,523,541,547,557,563,569,571,577,587,593,599,601,607,613,617,619,631,641,643,647,653,659,661,673,677,683,691,701,709,719,727,733,739,743,751,757,761,769,773,787,797,809,811,821,823,827,829,839,853,857,859,863,877,881,883,887,907,911,919,929,937,941,947,953,967,971,977,983,991,997];
var lplim = (1<<26)/lowprimes[lowprimes.length-1];
// (public) test primality with certainty >= 1-.5^t
function bnIsProbablePrime(t) {
var i, x = this.abs();
if(x.t == 1 && x[0] <= lowprimes[lowprimes.length-1]) {
for(i = 0; i < lowprimes.length; ++i)
if(x[0] == lowprimes[i]) return true;
return false;
}
if(x.isEven()) return false;
i = 1;
while(i < lowprimes.length) {
var m = lowprimes[i], j = i+1;
while(j < lowprimes.length && m < lplim) m *= lowprimes[j++];
m = x.modInt(m);
while(i < j) if(m%lowprimes[i++] == 0) return false;
}
return x.millerRabin(t);
}
// (protected) true if probably prime (HAC 4.24, Miller-Rabin)
function bnpMillerRabin(t) {
var n1 = this.subtract(BigInteger.ONE);
var k = n1.getLowestSetBit();
if(k <= 0) return false;
var r = n1.shiftRight(k);
t = (t+1)>>1;
if(t > lowprimes.length) t = lowprimes.length;
var a = nbi();
for(var i = 0; i < t; ++i) {
//Pick bases at random, instead of starting at 2
a.fromInt(lowprimes[Math.floor(Math.random()*lowprimes.length)]);
var y = a.modPow(r,this);
if(y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {
var j = 1;
while(j++ < k && y.compareTo(n1) != 0) {
y = y.modPowInt(2,this);
if(y.compareTo(BigInteger.ONE) == 0) return false;
}
if(y.compareTo(n1) != 0) return false;
}
}
return true;
}
// protected
BigInteger.prototype.chunkSize = bnpChunkSize;
BigInteger.prototype.toRadix = bnpToRadix;
BigInteger.prototype.fromRadix = bnpFromRadix;
BigInteger.prototype.fromNumber = bnpFromNumber;
BigInteger.prototype.bitwiseTo = bnpBitwiseTo;
BigInteger.prototype.changeBit = bnpChangeBit;
BigInteger.prototype.addTo = bnpAddTo;
BigInteger.prototype.dMultiply = bnpDMultiply;
BigInteger.prototype.dAddOffset = bnpDAddOffset;
BigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;
BigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;
BigInteger.prototype.modInt = bnpModInt;
BigInteger.prototype.millerRabin = bnpMillerRabin;
// public
BigInteger.prototype.clone = bnClone;
BigInteger.prototype.intValue = bnIntValue;
BigInteger.prototype.byteValue = bnByteValue;
BigInteger.prototype.shortValue = bnShortValue;
BigInteger.prototype.signum = bnSigNum;
BigInteger.prototype.toByteArray = bnToByteArray;
BigInteger.prototype.equals = bnEquals;
BigInteger.prototype.min = bnMin;
BigInteger.prototype.max = bnMax;
BigInteger.prototype.and = bnAnd;
BigInteger.prototype.or = bnOr;
BigInteger.prototype.xor = bnXor;
BigInteger.prototype.andNot = bnAndNot;
BigInteger.prototype.not = bnNot;
BigInteger.prototype.shiftLeft = bnShiftLeft;
BigInteger.prototype.shiftRight = bnShiftRight;
BigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;
BigInteger.prototype.bitCount = bnBitCount;
BigInteger.prototype.testBit = bnTestBit;
BigInteger.prototype.setBit = bnSetBit;
BigInteger.prototype.clearBit = bnClearBit;
BigInteger.prototype.flipBit = bnFlipBit;
BigInteger.prototype.add = bnAdd;
BigInteger.prototype.subtract = bnSubtract;
BigInteger.prototype.multiply = bnMultiply;
BigInteger.prototype.divide = bnDivide;
BigInteger.prototype.remainder = bnRemainder;
BigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;
BigInteger.prototype.modPow = bnModPow;
BigInteger.prototype.modInverse = bnModInverse;
BigInteger.prototype.pow = bnPow;
BigInteger.prototype.gcd = bnGCD;
BigInteger.prototype.isProbablePrime = bnIsProbablePrime;
// JSBN-specific extension
BigInteger.prototype.square = bnSquare;
// BigInteger interfaces not implemented in jsbn:
// BigInteger(int signum, byte[] magnitude)
// double doubleValue()
// float floatValue()
// int hashCode()
// long longValue()
// static BigInteger valueOf(long val)

45
ayun/jsbn/prng4.js Normal file
View File

@ -0,0 +1,45 @@
// prng4.js - uses Arcfour as a PRNG
function Arcfour() {
this.i = 0;
this.j = 0;
this.S = new Array();
}
// Initialize arcfour context from key, an array of ints, each from [0..255]
function ARC4init(key) {
var i, j, t;
for(i = 0; i < 256; ++i)
this.S[i] = i;
j = 0;
for(i = 0; i < 256; ++i) {
j = (j + this.S[i] + key[i % key.length]) & 255;
t = this.S[i];
this.S[i] = this.S[j];
this.S[j] = t;
}
this.i = 0;
this.j = 0;
}
function ARC4next() {
var t;
this.i = (this.i + 1) & 255;
this.j = (this.j + this.S[this.i]) & 255;
t = this.S[this.i];
this.S[this.i] = this.S[this.j];
this.S[this.j] = t;
return this.S[(t + this.S[this.i]) & 255];
}
Arcfour.prototype.init = ARC4init;
Arcfour.prototype.next = ARC4next;
// Plug in your RNG constructor here
function prng_newstate() {
return new Arcfour();
}
// Pool size must be a multiple of 4 and greater than 32.
// An array of bytes the size of the pool will be passed to init()
var rng_psize = 256;

75
ayun/jsbn/rng.js Normal file
View File

@ -0,0 +1,75 @@
// Random number generator - requires a PRNG backend, e.g. prng4.js
// For best results, put code like
// <body onClick='rng_seed_time();' onKeyPress='rng_seed_time();'>
// in your main HTML document.
var rng_state;
var rng_pool;
var rng_pptr;
// Mix in a 32-bit integer into the pool
function rng_seed_int(x) {
rng_pool[rng_pptr++] ^= x & 255;
rng_pool[rng_pptr++] ^= (x >> 8) & 255;
rng_pool[rng_pptr++] ^= (x >> 16) & 255;
rng_pool[rng_pptr++] ^= (x >> 24) & 255;
if(rng_pptr >= rng_psize) rng_pptr -= rng_psize;
}
// Mix in the current time (w/milliseconds) into the pool
function rng_seed_time() {
rng_seed_int(new Date().getTime());
}
// Initialize the pool with junk if needed.
if(rng_pool == null) {
rng_pool = new Array();
rng_pptr = 0;
var t;
if(window.crypto && window.crypto.getRandomValues) {
// Use webcrypto if available
var ua = new Uint8Array(32);
window.crypto.getRandomValues(ua);
for(t = 0; t < 32; ++t)
rng_pool[rng_pptr++] = ua[t];
}
if(navigator.appName == "Netscape" && navigator.appVersion < "5" && window.crypto) {
// Extract entropy (256 bits) from NS4 RNG if available
var z = window.crypto.random(32);
for(t = 0; t < z.length; ++t)
rng_pool[rng_pptr++] = z.charCodeAt(t) & 255;
}
while(rng_pptr < rng_psize) { // extract some randomness from Math.random()
t = Math.floor(65536 * Math.random());
rng_pool[rng_pptr++] = t >>> 8;
rng_pool[rng_pptr++] = t & 255;
}
rng_pptr = 0;
rng_seed_time();
//rng_seed_int(window.screenX);
//rng_seed_int(window.screenY);
}
function rng_get_byte() {
if(rng_state == null) {
rng_seed_time();
rng_state = prng_newstate();
rng_state.init(rng_pool);
for(rng_pptr = 0; rng_pptr < rng_pool.length; ++rng_pptr)
rng_pool[rng_pptr] = 0;
rng_pptr = 0;
//rng_pool = null;
}
// TODO: allow reseeding after first request
return rng_state.next();
}
function rng_get_bytes(ba) {
var i;
for(i = 0; i < ba.length; ++i) ba[i] = rng_get_byte();
}
function SecureRandom() {}
SecureRandom.prototype.nextBytes = rng_get_bytes;

118
ayun/jsbn/rsa.js Normal file
View File

@ -0,0 +1,118 @@
// Depends on jsbn.js and rng.js
// Version 1.1: support utf-8 encoding in pkcs1pad2
// convert a (hex) string to a bignum object
function parseBigInt(str,r) {
return new BigInteger(str,r);
}
function linebrk(s,n) {
var ret = "";
var i = 0;
while(i + n < s.length) {
ret += s.substring(i,i+n) + "\n";
i += n;
}
return ret + s.substring(i,s.length);
}
function byte2Hex(b) {
if(b < 0x10)
return "0" + b.toString(16);
else
return b.toString(16);
}
// PKCS#1 (type 2, random) pad input string s to n bytes, and return a bigint
function pkcs1pad2(s,n) {
//var ba = b64toBA(hex2b64(s));
//n -= ba.length;
if(n < s.length + 11) { // TODO: fix for utf-8
alert("Message too long for RSA");
return null;
}
var ba = new Array();
var i = (s.length / 2) - 1;
while(i >= 0 && n > 0) {
ba[--n] = parseInt(s.substring(2*i,2*i+2),16);
i--;
/*
var c = s.charCodeAt(i--);
if(c < 128) { // encode using utf-8
ba[--n] = c;
}
else if((c > 127) && (c < 2048)) {
ba[--n] = (c & 63) | 128;
ba[--n] = (c >> 6) | 192;
}
else {
ba[--n] = (c & 63) | 128;
ba[--n] = ((c >> 6) & 63) | 128;
ba[--n] = (c >> 12) | 224;
}
*/
}
ba[--n] = 0;
var rng = new SecureRandom();
var x = new Array();
while(n > 2) { // random non-zero pad
x[0] = 0;
while(x[0] == 0) rng.nextBytes(x);
ba[--n] = x[0];
}
ba[--n] = 2;
ba[--n] = 0;
return new BigInteger(ba);
}
// "empty" RSA key constructor
function RSAKey() {
this.n = null;
this.e = 0;
this.d = null;
this.p = null;
this.q = null;
this.dmp1 = null;
this.dmq1 = null;
this.coeff = null;
}
// Set the public key fields N and e from hex strings
function RSASetPublic(N,E) {
if(N != null && E != null && N.length > 0 && E.length > 0) {
this.n = parseBigInt(N,16);
this.e = parseInt(E,16);
}
else
alert("Invalid RSA public key");
}
// Perform raw public operation on "x": return x^e (mod n)
function RSADoPublic(x) {
return x.modPowInt(this.e, this.n);
}
// Return the PKCS#1 RSA encryption of "text" as an even-length hex string
function RSAEncrypt(text) {
var m = pkcs1pad2(text,(this.n.bitLength()+7)>>3);
if(m == null) return null;
var c = this.doPublic(m);
if(c == null) return null;
var h = c.toString(16);
if((h.length & 1) == 0) return h; else return "0" + h;
}
// Return the PKCS#1 RSA encryption of "text" as a Base64-encoded string
//function RSAEncryptB64(text) {
// var h = this.encrypt(text);
// if(h) return hex2b64(h); else return null;
//}
// protected
RSAKey.prototype.doPublic = RSADoPublic;
// public
RSAKey.prototype.setPublic = RSASetPublic;
RSAKey.prototype.encrypt = RSAEncrypt;
//RSAKey.prototype.encrypt_b64 = RSAEncryptB64;

135
ayun/jsbn/rsa2.js Normal file
View File

@ -0,0 +1,135 @@
// Depends on rsa.js and jsbn2.js
// Version 1.1: support utf-8 decoding in pkcs1unpad2
// Undo PKCS#1 (type 2, random) padding and, if valid, return the plaintext
function pkcs1unpad2(d,n) {
var b = d.toByteArray();
var i = 0;
while(i < b.length && b[i] == 0) ++i;
if(b.length-i != n-1 || b[i] != 2)
return null;
++i;
while(b[i] != 0)
if(++i >= b.length) return null;
var ret = "";
while(++i < b.length) {
ret += byte2Hex(b[i]);
/*
var c = b[i] & 255;
if(c < 128) { // utf-8 decode
ret += String.fromCharCode(c);
}
else if((c > 191) && (c < 224)) {
ret += String.fromCharCode(((c & 31) << 6) | (b[i+1] & 63));
++i;
}
else {
ret += String.fromCharCode(((c & 15) << 12) | ((b[i+1] & 63) << 6) | (b[i+2] & 63));
i += 2;
}
*/
}
return ret;
}
// Set the private key fields N, e, and d from hex strings
function RSASetPrivate(N,E,D) {
if(N != null && E != null && N.length > 0 && E.length > 0) {
this.n = parseBigInt(N,16);
this.e = parseInt(E,16);
this.d = parseBigInt(D,16);
}
else
alert("Invalid RSA private key");
}
// Set the private key fields N, e, d and CRT params from hex strings
function RSASetPrivateEx(N,E,D,P,Q,DP,DQ,C) {
if(N != null && E != null && N.length > 0 && E.length > 0) {
this.n = parseBigInt(N,16);
this.e = parseInt(E,16);
this.d = parseBigInt(D,16);
this.p = parseBigInt(P,16);
this.q = parseBigInt(Q,16);
this.dmp1 = parseBigInt(DP,16);
this.dmq1 = parseBigInt(DQ,16);
this.coeff = parseBigInt(C,16);
}
else
alert("Invalid RSA private key");
}
// Generate a new random private key B bits long, using public expt E
function RSAGenerate(B,E) {
var rng = new SecureRandom();
var qs = B>>1;
this.e = parseInt(E,16);
var ee = new BigInteger(E,16);
for(;;) {
for(;;) {
this.p = new BigInteger(B-qs,1,rng);
if(this.p.subtract(BigInteger.ONE).gcd(ee).compareTo(BigInteger.ONE) == 0 && this.p.isProbablePrime(10)) break;
}
for(;;) {
this.q = new BigInteger(qs,1,rng);
if(this.q.subtract(BigInteger.ONE).gcd(ee).compareTo(BigInteger.ONE) == 0 && this.q.isProbablePrime(10)) break;
}
if(this.p.compareTo(this.q) <= 0) {
var t = this.p;
this.p = this.q;
this.q = t;
}
var p1 = this.p.subtract(BigInteger.ONE);
var q1 = this.q.subtract(BigInteger.ONE);
var phi = p1.multiply(q1);
if(phi.gcd(ee).compareTo(BigInteger.ONE) == 0) {
this.n = this.p.multiply(this.q);
this.d = ee.modInverse(phi);
this.dmp1 = this.d.mod(p1);
this.dmq1 = this.d.mod(q1);
this.coeff = this.q.modInverse(this.p);
break;
}
}
}
// Perform raw private operation on "x": return x^d (mod n)
function RSADoPrivate(x) {
if(this.p == null || this.q == null)
return x.modPow(this.d, this.n);
// TODO: re-calculate any missing CRT params
var xp = x.mod(this.p).modPow(this.dmp1, this.p);
var xq = x.mod(this.q).modPow(this.dmq1, this.q);
while(xp.compareTo(xq) < 0)
xp = xp.add(this.p);
return xp.subtract(xq).multiply(this.coeff).mod(this.p).multiply(this.q).add(xq);
}
// Return the PKCS#1 RSA decryption of "ctext".
// "ctext" is an even-length hex string and the output is a plain string.
function RSADecrypt(ctext) {
var c = parseBigInt(ctext, 16);
var m = this.doPrivate(c);
if(m == null) return null;
return pkcs1unpad2(m, (this.n.bitLength()+7)>>3);
}
// Return the PKCS#1 RSA decryption of "ctext".
// "ctext" is a Base64-encoded string and the output is a plain string.
//function RSAB64Decrypt(ctext) {
// var h = b64tohex(ctext);
// if(h) return this.decrypt(h); else return null;
//}
// protected
RSAKey.prototype.doPrivate = RSADoPrivate;
// public
RSAKey.prototype.setPrivate = RSASetPrivate;
RSAKey.prototype.setPrivateEx = RSASetPrivateEx;
RSAKey.prototype.generate = RSAGenerate;
RSAKey.prototype.decrypt = RSADecrypt;
//RSAKey.prototype.b64_decrypt = RSAB64Decrypt;

118
ayun/music.js Normal file
View File

@ -0,0 +1,118 @@
var aud=new Audio();
var songs=["PkIeYKbzyZY","lv5A5EDvi-g","EyicJOlYOm4","1weATSoaRbA","qSdiecyj6-M","Ugb7GUtiyZ0","1ZFMy48q63w","10RNhMuWR2g","QkVvPD7Xk5U","RUEWIG8zoa0","V6N_rL4fh6I","n02zTn2d3rY","iBZS6ad3Tlk","Mnb2RhXL-nM","lidx_2d4YOA","ETQJZHYlc3g","kRpRoTaNni0","ACy5tHoNUoA","yLFX_7SH2tY","io-vPXE9JXY","oP6wOte3wZU","6LwLyIv3yvA","MrD05HVGVIQ","HBYS5mBHie4","5HxGK3DTUBQ","_9qUu8IeabE","r76yldhXSrw","OqJi_n3AcV4","WFFF-jMyFaQ","D8CwzhM7O6I","FD56t_0B9ig","5OcsWd949w4","jrnNzNT11aU","VLbMXG8lvjI","4aNDoqt9kEQ","_ZMfQ0Aj7h8","ZCJo8CDyqlQ","A1-fM0s1Yt0","zsLT3JqfTn0","zsGM_sGEako","YZlclPLX1Hw","uT1evbCloAw","HYOth1zARhg","loeGmoYr3s4","0qhoqXTUQlY","FtdPlfZNqVY","uIk_jGypR24","mdAb2xALVm0","5hpBvX0sUuo","RiVZCDq--m4","2k5dqgNT37g","j4Jyev7iTlE","9Ty-qFZZPZk","6EDS01Ipaow","720HcvEvEC8","YbdcrJZBtu8","nZ15jw3NOO8","Ub7y69hg4do","hFal0LKZwnM","g-QdgLe_D5M","CAmHrCvZ8tA","Mr-wV17WFZU","EdFyQOngYJs","3RyqONKuRzk","jBPqr_IsWvY","LcyCREQL7w8","dAalyaoVGfE","RM9O1HO4FLE","tUx4SfdoyUI","qFjaDnnPbA4","NrlhbIzjO04","eDBAdAzCqr4","UqVW7-q7fTA","3I-WJ6UgmOA","m8wcDiahBjg","NmzrLZajTLU","EPT3dIWBbDA","tf4dBTlz25g","FuMtDXkuxVw","PkmKM_OXNZM","KbC46oJmLh4","mRJSIYmHuNI","GmLsIivtcIM","rDBbaGCCIhk","Ig5v4jhLLWI","XZcG2esvW7I","_qpgIOzaI0E","TTJBevUIp0s","V4oiuY02dTo","agIayif-oi0","plm3DVsX7Jg","HA0Mk5BXX44","2RLq_paK3-g","h2EgfSvOuPc","HbtwR1REaFk","-tUJJXWXdXk","h-b8Xs7sNI0","xt2DGLoEmW0","eyp-YuzhTN4","xrawoRF4lN0","6xVHpgJbuc8","qsy-14oAXGc","fYSUV33ZPfw","qnSHJlRJ2cM","Ytt1_ErIV34","05TnpE0x4wI","-jcOtAuGZC4","XbuqB3uB6DI","x-42Zkcw4Cs","kXjA7eRZchc","_zceCigOTwo","Qg83cniiYEY","F7gwcgmoREg","aZFe4K1HmPE","e1Ve4Xa9ftQ","p-dVxnR-vLw","PD1a3mfmY8M","-cJFVNZC4h8","Jfs7Lo8CC0Q","zz3F5j8qWNw","CQLvggJFxuM","wf93JAZR3gU","XLQbs59Pn7E","phrpiLpaiLs","Md2xOKFTvSU","_Ci0Kgdpgsw","iBjZshhpipg","j15tAxxc38Y","LwKbsK6E-DI","9J8cD8-LHy4","7ZQlWCsHG7k","UBIsi3xWa64","GWYkh1IX4PE","YYjDFXJ6Wdo","5YDiUcc3emA","s0G4qOrDOgM","9zibDnOOj3w","rQMd3b1BF50","xeM40-FkRLI","wVOFnTrSOOA","CbziO8vuBR8","msUarvc4Sx8","GB9kBLre96M","UMwmB0ZK3yE","SLFMiEAjSoA","K5F-RLzLH6Q","ziAK1OLeeEE","--I54YPKsLU","33zGN7vENog","pTv51QwN57E","zD8TxUBkjGA","A09BhpgfGKQ","cvBQq1yJH-A","YZrIQardsz8","CUHYQ-FN3P8","GLGjqtgCKY8","bLagC2wX3Ak","fW128GHFJIE","e15qP676Zhg","B5L0AMO2HA8","oFFFzMkGNrk","1RQQLwnaw80","byUipqLQ_Hc","WtF6Z13HD_w","pkkIqT9LpDY","gMGEyl5TRa4","oxoqm05c7yA","yTP8tK2OzFQ"];
var insturl="https://invidious.zapashcanon.fr";
var loading=false;
var usealt=0;
function shuffle(array) {
for (let i = 0; i < array.length; i++) {
let j = Math.floor(Math.random() * (array.length-i)) + i;
[array[i], array[j]] = [array[j], array[i]];
}
return array;
}
function fixfard(url){
if(!url)return insturl;
return url.endsWith("/")?url.slice(0,url.length-1):url;
}
var updinsturl=async ()=>{
try{
var json=await (await fetch("https://api.invidious.io/instances.json?sort_by=health")).json();
var out=shuffle(json).map(entry => {
const healthKnown = !!entry[1].monitor
return {
name: entry[0],
details: entry[1],
health: +(healthKnown ? entry[1].monitor.dailyRatios[0].ratio : 95),
healthKnown
}
}).filter(entry => {
return entry.details.type === "https" && entry.health > 0
}).sort((a, b) => {
return b.health - a.health
});
insturl=fixfard(out.find(e=>e.details.cors).details.uri);
}catch(e){aud.onerror();}
};
var updint=setInterval(updinsturl,3600000);
updinsturl();
aud.onended=function(e){
loading=true;
window.startmusic();
};
aud.oncanplay=function(e){
aud.play();
};
aud.onplay=function(e){
aud.playing=true;
loading=false;
};
aud.onerror=function(e){
if(usealt==2){
aud=null;
return;
}
if(usealt==0){
usealt=1;
}else if(usealt==1){
loading=true;
usealt=2;
}
window.stopmusic();
//todo: make attempt 2 more times with diff urls and if those fail then stop trying
clearInterval(updint);
};
window.startmusic=function(v){
if(v!=null&&v==0)return;
loading=true;
var url="";
if(usealt==0){
songs=shuffle(songs);
url=insturl+"/latest_version?id="+songs[0]+"&itag=251";
}else if(usealt==1){
url="https://nightride.fm/stream/chillsynth.m4a";
}else if(usealt==2){
return;
}
if(v!=null)aud.volume=v;
aud.src=url;
aud.currentTime=0;
};
window.stopmusic=function(){
if(usealt==2)return;
aud.pause();
loading=false;
};
window.volmusic=function(v){
if(usealt==2)return;
if(v==0){
window.startmusic();
}else{
if(aud.playing){
aud.volume=v;
}else{
window.startmusic(v);
}
}
};
window.playingmusic=function(){
return usealt==2||aud.playing||loading;
};
navigator.mediaSession.setActionHandler('play', function() {});
navigator.mediaSession.setActionHandler('pause', function() {});
navigator.mediaSession.setActionHandler('seekbackward', function() {});
navigator.mediaSession.setActionHandler('seekforward', function() {});
navigator.mediaSession.setActionHandler('previoustrack', function() {});
navigator.mediaSession.setActionHandler('nexttrack', function() {});

3
drag/cover.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 67 KiB

32219
drag/index.html Normal file

File diff suppressed because one or more lines are too long

View File

@ -1,5 +1,47 @@
[ [
{ {
"name": "π Client",
"image": "cover.svg",
"directory": "pi",
"recommended": "1"
},
{
"name": "Precision Client",
"image": "cover.svg",
"directory": "pre",
"recommended": "1"
},
{
"name": "Shadow Client",
"image": "cover.svg",
"directory": "shadow",
"recommended": "1"
},
{
"name": "Windows XP",
"image": "cover.svg",
"directory": "xp",
"recommended": "1"
},
{
"name": "Dragon Client",
"image": "cover.svg",
"directory": "drag",
"recommended": "1"
},
{
"name": "Ayuncraft 1.5.2",
"image": "cover.svg",
"directory": "ayun",
"recommended": "1"
},
{
"name": "Minecraft Infdev",
"image": "cover.svg",
"directory": "inf",
"recommended": "1"
},
{
"name": "Minesweeper", "name": "Minesweeper",
"image": "cover.svg", "image": "cover.svg",
"directory": "minesweeper", "directory": "minesweeper",

View File

@ -43,6 +43,13 @@ app.get('/ad', (req, res) => {
res.sendFile(path.join(__dirname, 'ad.html')); res.sendFile(path.join(__dirname, 'ad.html'));
}); });
app.get('/blank', (req, res) => {
res.sendFile(path.join(__dirname, 'blank.html'));
});
app.get('/backgrounds', (req, res) => {
res.sendFile(path.join(__dirname, 'backgrounds.html'));
});
app.listen(port, () => { app.listen(port, () => {
console.log(`Selenite is running on port ${port}`); console.log(`Selenite is running on port ${port}`);
}); });

5405
inf/app.js Normal file

File diff suppressed because one or more lines are too long

3
inf/cover.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 14 KiB

84
inf/favicon.ico Normal file
View File

@ -0,0 +1,84 @@
<!DOCTYPE html>
<html>
<head>
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-261780635-1"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-261780635-1');
</script>
<!-- Google Tag Manager -->
<script>(function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
'https://www.googletagmanager.com/gtm.js?id='+i+dl;f.parentNode.insertBefore(j,f);
})(window,document,'script','dataLayer','GTM-TG85STRS');</script>
<!-- End Google Tag Manager -->
<title>Play Minecraft Online! | Online mc</title>
<link rel="icon" href="/icon.png">
<meta name="description" content="Play Minecraft (eaglercraft) Online For Free!" />
<meta name="keywords" content="free minecraft online, eaglercraft, minecraft, onlinemc, Onlinemc, ayuncraft, Ayuncraft, mc, eaglercraftx, eaglercraft shaders, eaglercraft download,eaglercraft, eaglercraftx, minecraft, 1.8, 1.8.8, eaglercraft, eaglercraft minecraft, eaglercraft 3kh0, eaglercraft github, eaglercraft singleplayer, eaglercraft hacks, eaglercraft 1.8, eaglercraft server, eaglecraft game, eaglercraft alternative, eaglercraft alt links, eaglercraft asspixel, eaglercraft anarchy server, eaglercraft aeon, eaglercraft asspixel server ip, eaglercraft all links, eaglercraft aeon server, eaglercraft beta, eaglecraft boats, eaglecraft boats for sale, eaglecraft boats for sale bc, eaglecraft bedwars server, eaglecraft boats inc, eaglercraft beta servers, eaglercraft blocked, eaglercraft client, eaglecraft custom boats, eaglecraft campbell river, eaglercraft creative, eaglercraft commands, eaglercraft code, eaglercraft controls, eaglercraft com, eaglercraft download, eaglercraft discord, eaglercraft dupe glitch, eaglercraft download file, eaglercraft download google drive, eaglercraft dmca, eaglercraft deleted, eaglercraft down, eaglercraft embed, eaglercraft epk file download, eaglercraft end of stream, eaglecraft apk download, eaglecraft eagtek, eaglercraft exploit, eaglecraft eggs, eaglercraft epk world file download, eaglercraft file, eaglercraft file download, eaglercraft free, eaglercraft forums, eaglercraft fork, eaglercraft faq, eaglercraft fps, eaglecraft free download, eaglercraft github servers, eaglercraft google drive, eaglercraft github lax1dude, eaglercraft github download, eaglercraft github.io, eaglercraft google sites, minecraft, applet, singleplayer, replit, html5, chromebook, lax1dude, games, eaglercraft singleplayer, eagler, minecraft for free, free minecraft">
<meta property="og:title" content="Online mc" />
<meta property="og:image" content="/icon.png" />
<meta property="og:description" content="Online Minecraft" />
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/css/bootstrap.min.css">
<link href="/style.css" rel="stylesheet" />
<link href="/darkmode.css" rel="stylesheet" />
</head>
<body class="diff-bg">
<div id="loading"></div>
<div class="container container-table">
<div class="row vertical-center-row">
<div class="col-12">
</a>
<div class="d-flex justify-content-center">
</div>
<div class="col-12">
<div class="button-area d-block">
<h1 class="dark-text">404 Not Found!</h1>
<div class="splash-404">...</div>
<h3>&nbsp;</h3>
<a href="/">click here to go back to the main page!</a>
<img class="creeper ms-auto d-block" src="creeper.webp" alt=""/>
</div>
<div class="mc-footer d-flex justify-content-between">
<div class="footer-copyright text-end text-white">This website is not affiliated with Mojang.</div>
<div class="mc-dd">
</div>
</div>
</div>
</div>
</div>
</div>
<script defer src="/_vercel/insights/script.js"></script>
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/js/bootstrap.bundle.min.js"></script>
<script>
window.onload = function () {
//display loader on page load
$('#loading').fadeOut();
}
setTimeout(function () {
//display loader on page load
$('#loading').fadeOut();
}, 12 * 1000);
</script>
<script src="/darkmode.js"></script>
<script src="/splash-404.js"></script>
</body>
</html>

16
inf/index.html Normal file
View File

@ -0,0 +1,16 @@
<!DOCTYPE html>
<html>
<head>
<title>Infdev</title>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8">
<script type="text/javascript" charset="utf-8" src="app.js"></script>
<script type = text/javascript>
window.addEventListener("load", function() {
window.minecraftOpts = ["game","resources.epk"];
main();
});
</script>
</head>
<body style="margin:0px;width:100vw;height:100vh;" id="game">
</body>
</html>

BIN
inf/resources.epk Normal file

Binary file not shown.

3
pi/cover.svg Normal file
View File

@ -0,0 +1,3 @@
<svg width="200" height="200" xmlns="http://www.w3.org/2000/svg">
<image href="" height="200" width="200" />
</svg>

After

Width:  |  Height:  |  Size: 3.9 KiB

31933
pi/index.html Normal file

File diff suppressed because one or more lines are too long

BIN
pre/ProductSans.ttf Normal file

Binary file not shown.

View File

@ -0,0 +1,54 @@
§fPrecision Client is a custom eaglercraft client with QOL §ffeatures, custom texture packs, and more.§r
§9This was created by LAX1DUDE and forked by EtcherFX§r
§4Ingame voice chat service created by ayunami2000§r
§2Click the Github or Discord button on the main menu for §2more information§r
§fScroll down for the detailed license statements
§fAll graphical assets in this project were ripped from §fMojang's official Minecraft.jar for Minecraft version §f1.5.2 §fand they are Mojang's intellectual property. Assets ommitted §finclude the Minecraft soundtrack and music disks and all §fsoftware references to the Mojang/Minecraft api.
§f========================================
§fCopyright (c) 2022 Calder Young.
§fThis work is licensed under a Creative Commons
§fAttribution-NonCommercial 4.0 International License
§f§nhttp://creativecommons.org/licenses/by-nc/4.0/§r
§f========================================
§fTeaVM Copyright 2016 Alexey Andreev.
§fLicensed under the Apache License, Version 2.0 (the §f"License"); you may not use this file except in compliance §fwith the License. You may obtain a copy of the License at
§f§nhttp://www.apache.org/licenses/LICENSE-2.0§r
§fUnless required by applicable law or agreed to in writing, §fsoftware distributed under the License is distributed on an §f"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY §fKIND, either express or implied. See the License for the §fspecific language governing permissions and limitations §funder the License.
§f========================================
§fNVIDIA FXAA 3.11 by TIMOTHY LOTTES
§fCopyright (c) 2010, 2011 NVIDIA CORPORATION. ALL RIGHTS §fRESERVED.
§fTO THE MAXIMUM EXTENT PERMITTED BY APPLICABLE LAW, §fTHIS SOFTWARE IS PROVIDED *AS IS* AND NVIDIA AND ITS §fSUPPLIERS DISCLAIM ALL WARRANTIES, EITHER EXPRESS OR §fIMPLIED, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES §fOF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. §fIN NO EVENT SHALL NVIDIA OR ITS SUPPLIERS BE LIABLE FOR §fANY SPECIAL, INCIDENTAL, INDIRECT, OR CONSEQUENTIAL §fDAMAGES WHATSOEVER (INCLUDING, WITHOUT LIMITATION, §fDAMAGES FOR LOSS OF BUSINESS PROFITS, BUSINESS §fINTERRUPTION, LOSS OF BUSINESS INFORMATION, OR ANY OTHER §fPECUNIARY LOSS) ARISING OUT OF THE USE OF OR INABILITY TO §fUSE THIS SOFTWARE, EVEN IF NVIDIA HAS BEEN ADVISED OF THE §fPOSSIBILITY OF SUCH DAMAGES.
§f========================================
§fJZlib Copyright (c) 2000-2011 ymnk, JCraft,Inc. All rights §freserved.
§fRedistribution and use in source and binary forms, with or §fwithout modification, are permitted provided that the following §fconditions are met:
§f1. Redistributions of source code must retain the above §fcopyright notice, this list of conditions and the following §fdisclaimer.
§f2. Redistributions in binary form must reproduce the §fabove copyright notice, this list of conditions and the §ffollowing disclaimer in the documentation and/or other §fmaterials provided with the distribution.
§f3. The names of the authors may not be used to endorse §for promote products derived from this software without §fspecific prior written permission.
§fTHIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESSED OR §fIMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE §fIMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A §fPARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL §fJCRAFT, INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE §fLIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, §fEXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT §fLIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; §fLOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) §fHOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER §fIN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING §fNEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE §fUSE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY §fOF SUCH DAMAGE.
§f========================================

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.1 KiB

File diff suppressed because it is too large Load Diff

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 50 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 48 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 48 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 54 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

BIN
pre/bg.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 427 KiB

19499
pre/classes.js Normal file

File diff suppressed because one or more lines are too long

16300
pre/classes_server.js Normal file

File diff suppressed because it is too large Load Diff

3
pre/cover.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 31 KiB

798
pre/eagswebrtc.js Normal file
View File

@ -0,0 +1,798 @@
"use strict";
/*
This is the backend for voice channels and LAN servers in eaglercraft
it links with TeaVM EaglerAdapter at runtime
Copyright 2022 ayunami2000 & lax1dude. All rights reserved.
*/
// %%%%%%%%%%%%%%%%%%%%%%%%%%%%% VOICE CODE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%
window.initializeVoiceClient = (() => {
const READYSTATE_NONE = 0;
const READYSTATE_ABORTED = -1;
const READYSTATE_DEVICE_INITIALIZED = 1;
const PEERSTATE_FAILED = 0;
const PEERSTATE_SUCCESS = 1;
const PEERSTATE_LOADING = 2;
class EaglercraftVoicePeer {
constructor(client, peerId, peerConnection, offer) {
this.client = client;
this.peerId = peerId;
this.peerConnection = peerConnection;
this.stream = null;
const self = this;
this.peerConnection.addEventListener("icecandidate", (evt) => {
if (evt.candidate) {
self.client.iceCandidateHandler(self.peerId, JSON.stringify({
sdpMLineIndex: evt.candidate.sdpMLineIndex,
candidate: evt.candidate.candidate
}));
}
});
this.peerConnection.addEventListener("track", (evt) => {
self.rawStream = evt.streams[0];
const aud = new Audio();
aud.autoplay = true;
aud.muted = true;
aud.onended = function () {
aud.remove();
};
aud.srcObject = self.rawStream;
self.client.peerTrackHandler(self.peerId, self.rawStream);
});
this.peerConnection.addStream(this.client.localMediaStream.stream);
if (offer) {
this.peerConnection.createOffer((desc) => {
const selfDesc = desc;
self.peerConnection.setLocalDescription(selfDesc, () => {
self.client.descriptionHandler(self.peerId, JSON.stringify(selfDesc));
if (self.client.peerStateInitial != PEERSTATE_SUCCESS) self.client.peerStateInitial = PEERSTATE_SUCCESS;
}, (err) => {
console.error("Failed to set local description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateInitial == PEERSTATE_LOADING) self.client.peerStateInitial = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
});
}, (err) => {
console.error("Failed to set create offer for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateInitial == PEERSTATE_LOADING) self.client.peerStateInitial = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
});
}
this.peerConnection.addEventListener("connectionstatechange", (evt) => {
if (self.peerConnection.connectionState === 'disconnected') {
self.client.signalDisconnect(self.peerId);
} else if (self.peerConnection.connectionState === 'connected') {
if (self.client.peerState != PEERSTATE_SUCCESS) self.client.peerState = PEERSTATE_SUCCESS;
} else if (self.peerConnection.connectionState === 'failed') {
if (self.client.peerState == PEERSTATE_LOADING) self.client.peerState = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
}
});
}
disconnect() {
this.peerConnection.close();
}
mute(muted) {
this.rawStream.getAudioTracks()[0].enabled = !muted;
}
setRemoteDescription(descJSON) {
const self = this;
try {
const remoteDesc = JSON.parse(descJSON);
this.peerConnection.setRemoteDescription(remoteDesc, () => {
if (remoteDesc.type == 'offer') {
self.peerConnection.createAnswer((desc) => {
const selfDesc = desc;
self.peerConnection.setLocalDescription(selfDesc, () => {
self.client.descriptionHandler(self.peerId, JSON.stringify(selfDesc));
if (self.client.peerStateDesc != PEERSTATE_SUCCESS) self.client.peerStateDesc = PEERSTATE_SUCCESS;
}, (err) => {
console.error("Failed to set local description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
});
}, (err) => {
console.error("Failed to create answer for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
});
}
}, (err) => {
console.error("Failed to set remote description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
});
} catch (err) {
console.error("Failed to parse remote description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalDisconnect(self.peerId);
}
}
addICECandidate(candidate) {
try {
this.peerConnection.addIceCandidate(new RTCIceCandidate(JSON.parse(candidate)));
if (this.client.peerStateIce != PEERSTATE_SUCCESS) this.client.peerStateIce = PEERSTATE_SUCCESS;
} catch (err) {
console.error("Failed to parse ice candidate for \"" + this.peerId + "\"! " + err);
if (this.client.peerStateIce == PEERSTATE_LOADING) this.client.peerStateIce = PEERSTATE_FAILED;
this.client.signalDisconnect(this.peerId);
}
}
}
class EaglercraftVoiceClient {
constructor() {
this.ICEServers = [];
this.hasInit = false;
this.peerList = new Map();
this.readyState = READYSTATE_NONE;
this.peerState = PEERSTATE_LOADING;
this.peerStateConnect = PEERSTATE_LOADING;
this.peerStateInitial = PEERSTATE_LOADING;
this.peerStateDesc = PEERSTATE_LOADING;
this.peerStateIce = PEERSTATE_LOADING;
this.iceCandidateHandler = null;
this.descriptionHandler = null;
this.peerTrackHandler = null;
this.peerDisconnectHandler = null;
this.microphoneVolumeAudioContext = null;
}
voiceClientSupported() {
return typeof window.RTCPeerConnection !== "undefined" && typeof navigator.mediaDevices !== "undefined" &&
typeof navigator.mediaDevices.getUserMedia !== "undefined";
}
setICEServers(urls) {
this.ICEServers.length = 0;
for (var i = 0; i < urls.length; ++i) {
var etr = urls[i].split(";");
if (etr.length == 1) {
this.ICEServers.push({urls: etr[0]});
} else if (etr.length == 3) {
this.ICEServers.push({urls: etr[0], username: etr[1], credential: etr[2]});
}
}
}
setICECandidateHandler(cb) {
this.iceCandidateHandler = cb;
}
setDescriptionHandler(cb) {
this.descriptionHandler = cb;
}
setPeerTrackHandler(cb) {
this.peerTrackHandler = cb;
}
setPeerDisconnectHandler(cb) {
this.peerDisconnectHandler = cb;
}
activateVoice(tk) {
if (this.hasInit) this.localRawMediaStream.getAudioTracks()[0].enabled = tk;
}
initializeDevices() {
if (!this.hasInit) {
const self = this;
navigator.mediaDevices.getUserMedia({audio: true, video: false}).then((stream) => {
self.microphoneVolumeAudioContext = new AudioContext();
self.localRawMediaStream = stream;
self.localRawMediaStream.getAudioTracks()[0].enabled = false;
self.localMediaStream = self.microphoneVolumeAudioContext.createMediaStreamDestination();
self.localMediaStreamGain = self.microphoneVolumeAudioContext.createGain();
var localStreamIn = self.microphoneVolumeAudioContext.createMediaStreamSource(stream);
localStreamIn.connect(self.localMediaStreamGain);
self.localMediaStreamGain.connect(self.localMediaStream);
self.localMediaStreamGain.gain.value = 1.0;
self.readyState = READYSTATE_DEVICE_INITIALIZED;
this.hasInit = true;
}).catch((err) => {
self.readyState = READYSTATE_ABORTED;
});
} else {
this.readyState = READYSTATE_DEVICE_INITIALIZED;
}
}
setMicVolume(val) {
if (this.hasInit) {
if (val > 0.5) val = 0.5 + (val - 0.5) * 2.0;
if (val > 1.5) val = 1.5;
if (val < 0.0) val = 0.0;
this.localMediaStreamGain.gain.value = val * 2.0;
}
}
resetPeerStates() {
this.peerState = this.peerStateConnect = this.peerStateInitial = this.peerStateDesc = this.peerStateIce = PEERSTATE_LOADING;
}
getPeerState() {
return this.peerState;
}
getPeerStateConnect() {
return this.peerStateConnect;
}
getPeerStateInitial() {
return this.peerStateInitial;
}
getPeerStateDesc() {
return this.peerStateDesc;
}
getPeerStateIce() {
return this.peerStateIce;
}
getReadyState() {
return this.readyState;
}
signalConnect(peerId, offer) {
if (!this.hasInit) this.initializeDevices();
try {
const peerConnection = new RTCPeerConnection({
iceServers: this.ICEServers,
optional: [{DtlsSrtpKeyAgreement: true}]
});
const peerInstance = new EaglercraftVoicePeer(this, peerId, peerConnection, offer);
this.peerList.set(peerId, peerInstance);
if (this.peerStateConnect != PEERSTATE_SUCCESS) this.peerStateConnect = PEERSTATE_SUCCESS;
} catch (e) {
if (this.peerStateConnect == PEERSTATE_LOADING) this.peerStateConnect = PEERSTATE_FAILED;
}
}
signalDescription(peerId, descJSON) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
thePeer.setRemoteDescription(descJSON);
}
}
signalDisconnect(peerId, quiet) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
this.peerList.delete(thePeer);
try {
thePeer.disconnect();
} catch (e) {
}
this.peerDisconnectHandler(peerId, quiet);
}
}
mutePeer(peerId, muted) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
thePeer.mute(muted);
}
}
signalICECandidate(peerId, candidate) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
thePeer.addICECandidate(candidate);
}
}
}
window.constructVoiceClient = () => new EaglercraftVoiceClient();
});
window.startVoiceClient = () => {
if (typeof window.constructVoiceClient !== "function") {
window.initializeVoiceClient();
}
return window.constructVoiceClient();
};
// %%%%%%%%%%%%%%%%%%%%%%%%%%%%% LAN CLIENT CODE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%
window.initializeLANClient = (() => {
const READYSTATE_INIT_FAILED = -2;
const READYSTATE_FAILED = -1;
const READYSTATE_DISCONNECTED = 0;
const READYSTATE_CONNECTING = 1;
const READYSTATE_CONNECTED = 2;
class EaglercraftLANClient {
constructor() {
this.ICEServers = [];
this.peerConnection = null;
this.dataChannel = null;
this.readyState = READYSTATE_CONNECTING;
this.iceCandidateHandler = null;
this.descriptionHandler = null;
this.remoteDataChannelHandler = null;
this.remoteDisconnectHandler = null;
this.remotePacketHandler = null;
}
LANClientSupported() {
return typeof window.RTCPeerConnection !== "undefined";
}
initializeClient() {
try {
if (this.dataChannel != null) {
this.dataChannel.close();
this.dataChannel = null;
}
if (this.peerConnection != null) {
this.peerConnection.close();
}
this.peerConnection = new RTCPeerConnection({
iceServers: this.ICEServers,
optional: [{DtlsSrtpKeyAgreement: true}]
});
this.readyState = READYSTATE_CONNECTING;
} catch (e) {
this.readyState = READYSTATE_INIT_FAILED;
}
}
setICEServers(urls) {
this.ICEServers.length = 0;
for (var i = 0; i < urls.length; ++i) {
var etr = urls[i].split(";");
if (etr.length == 1) {
this.ICEServers.push({urls: etr[0]});
} else if (etr.length == 3) {
this.ICEServers.push({urls: etr[0], username: etr[1], credential: etr[2]});
}
}
}
setICECandidateHandler(cb) {
this.iceCandidateHandler = cb;
}
setDescriptionHandler(cb) {
this.descriptionHandler = cb;
}
setRemoteDataChannelHandler(cb) {
this.remoteDataChannelHandler = cb;
}
setRemoteDisconnectHandler(cb) {
this.remoteDisconnectHandler = cb;
}
setRemotePacketHandler(cb) {
this.remotePacketHandler = cb;
}
getReadyState() {
return this.readyState;
}
sendPacketToServer(buffer) {
if (this.dataChannel != null && this.dataChannel.readyState == "open") {
this.dataChannel.send(buffer);
} else {
this.signalRemoteDisconnect(false);
}
}
signalRemoteConnect() {
const self = this;
const iceCandidates = [];
this.peerConnection.addEventListener("icecandidate", (evt) => {
if (evt.candidate) {
if (iceCandidates.length == 0) setTimeout(() => {
if (self.peerConnection != null && self.peerConnection.connectionState != "disconnected") {
self.iceCandidateHandler(JSON.stringify(iceCandidates));
iceCandidates.length = 0;
}
}, 3000);
iceCandidates.push({
sdpMLineIndex: evt.candidate.sdpMLineIndex,
candidate: evt.candidate.candidate
});
}
});
this.dataChannel = this.peerConnection.createDataChannel("lan");
this.dataChannel.binaryType = "arraybuffer";
this.dataChannel.addEventListener("open", async (evt) => {
while (iceCandidates.length > 0) {
await new Promise(resolve => setTimeout(resolve, 0));
}
self.remoteDataChannelHandler(self.dataChannel);
});
this.dataChannel.addEventListener("message", (evt) => {
self.remotePacketHandler(evt.data);
}, false);
this.peerConnection.createOffer((desc) => {
const selfDesc = desc;
self.peerConnection.setLocalDescription(selfDesc, () => {
self.descriptionHandler(JSON.stringify(selfDesc));
}, (err) => {
console.error("Failed to set local description! " + err);
self.readyState = READYSTATE_FAILED;
self.signalRemoteDisconnect(false);
});
}, (err) => {
console.error("Failed to set create offer! " + err);
self.readyState = READYSTATE_FAILED;
self.signalRemoteDisconnect(false);
});
this.peerConnection.addEventListener("connectionstatechange", (evt) => {
if (self.peerConnection.connectionState === 'disconnected') {
self.signalRemoteDisconnect(false);
} else if (self.peerConnection.connectionState === 'connected') {
self.readyState = READYSTATE_CONNECTED;
} else if (self.peerConnection.connectionState === 'failed') {
self.readyState = READYSTATE_FAILED;
self.signalRemoteDisconnect(false);
}
});
}
signalRemoteDescription(descJSON) {
try {
this.peerConnection.setRemoteDescription(JSON.parse(descJSON));
} catch (e) {
console.error(e);
this.readyState = READYSTATE_FAILED;
this.signalRemoteDisconnect(false);
}
}
signalRemoteICECandidate(candidates) {
try {
const candidateList = JSON.parse(candidates);
for (let candidate of candidateList) {
this.peerConnection.addIceCandidate(candidate);
}
} catch (e) {
console.error(e);
this.readyState = READYSTATE_FAILED;
this.signalRemoteDisconnect(false);
}
}
signalRemoteDisconnect(quiet) {
if (this.dataChannel != null) {
this.dataChannel.close();
this.dataChannel = null;
}
if (this.peerConnection != null) {
this.peerConnection.close();
}
if (!quiet) this.remoteDisconnectHandler();
this.readyState = READYSTATE_DISCONNECTED;
}
};
window.constructLANClient = () => new EaglercraftLANClient();
});
window.startLANClient = () => {
if (typeof window.constructLANClient !== "function") {
window.initializeLANClient();
}
return window.constructLANClient();
};
// %%%%%%%%%%%%%%%%%%%%%%%%%%%%% LAN SERVER CODE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%
window.initializeLANServer = (() => {
const PEERSTATE_FAILED = 0;
const PEERSTATE_SUCCESS = 1;
const PEERSTATE_LOADING = 2;
class EaglercraftLANPeer {
constructor(client, peerId, peerConnection) {
this.client = client;
this.peerId = peerId;
this.peerConnection = peerConnection;
this.dataChannel = null;
const self = this;
const iceCandidates = [];
this.peerConnection.addEventListener("icecandidate", (evt) => {
if (evt.candidate) {
if (iceCandidates.length == 0) setTimeout(() => {
if (self.peerConnection != null && self.peerConnection.connectionState != "disconnected") {
self.client.iceCandidateHandler(self.peerId, JSON.stringify(iceCandidates));
iceCandidates.length = 0;
}
}, 3000);
iceCandidates.push({
sdpMLineIndex: evt.candidate.sdpMLineIndex,
candidate: evt.candidate.candidate
});
}
});
this.peerConnection.addEventListener("datachannel", async (evt) => {
while (iceCandidates.length > 0) {
await new Promise(resolve => setTimeout(resolve, 0));
}
self.dataChannel = evt.channel;
self.client.remoteClientDataChannelHandler(self.peerId, self.dataChannel);
self.dataChannel.addEventListener("message", (evt) => {
self.client.remoteClientPacketHandler(self.peerId, evt.data);
}, false);
}, false);
this.peerConnection.addEventListener("connectionstatechange", (evt) => {
if (self.peerConnection.connectionState === 'disconnected') {
self.client.signalRemoteDisconnect(self.peerId);
} else if (self.peerConnection.connectionState === 'connected') {
if (self.client.peerState != PEERSTATE_SUCCESS) self.client.peerState = PEERSTATE_SUCCESS;
} else if (self.peerConnection.connectionState === 'failed') {
if (self.client.peerState == PEERSTATE_LOADING) self.client.peerState = PEERSTATE_FAILED;
self.client.signalRemoteDisconnect(self.peerId);
}
});
}
disconnect() {
if (this.dataChannel != null) {
this.dataChannel.close();
this.dataChannel = null;
}
this.peerConnection.close();
}
setRemoteDescription(descJSON) {
const self = this;
try {
const remoteDesc = JSON.parse(descJSON);
this.peerConnection.setRemoteDescription(remoteDesc, () => {
if (remoteDesc.type == 'offer') {
self.peerConnection.createAnswer((desc) => {
const selfDesc = desc;
self.peerConnection.setLocalDescription(selfDesc, () => {
self.client.descriptionHandler(self.peerId, JSON.stringify(selfDesc));
if (self.client.peerStateDesc != PEERSTATE_SUCCESS) self.client.peerStateDesc = PEERSTATE_SUCCESS;
}, (err) => {
console.error("Failed to set local description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalRemoteDisconnect(self.peerId);
});
}, (err) => {
console.error("Failed to create answer for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalRemoteDisconnect(self.peerId);
});
}
}, (err) => {
console.error("Failed to set remote description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalRemoteDisconnect(self.peerId);
});
} catch (err) {
console.error("Failed to parse remote description for \"" + self.peerId + "\"! " + err);
if (self.client.peerStateDesc == PEERSTATE_LOADING) self.client.peerStateDesc = PEERSTATE_FAILED;
self.client.signalRemoteDisconnect(self.peerId);
}
}
addICECandidate(candidates) {
try {
const candidateList = JSON.parse(candidates);
for (let candidate of candidateList) {
this.peerConnection.addIceCandidate(new RTCIceCandidate(candidate));
}
if (this.client.peerStateIce != PEERSTATE_SUCCESS) this.client.peerStateIce = PEERSTATE_SUCCESS;
} catch (err) {
console.error("Failed to parse ice candidate for \"" + this.peerId + "\"! " + err);
if (this.client.peerStateIce == PEERSTATE_LOADING) this.client.peerStateIce = PEERSTATE_FAILED;
this.client.signalRemoteDisconnect(this.peerId);
}
}
}
class EaglercraftLANServer {
constructor() {
this.ICEServers = [];
this.hasInit = false;
this.peerList = new Map();
this.peerState = PEERSTATE_LOADING;
this.peerStateConnect = PEERSTATE_LOADING;
this.peerStateInitial = PEERSTATE_LOADING;
this.peerStateDesc = PEERSTATE_LOADING;
this.peerStateIce = PEERSTATE_LOADING;
this.iceCandidateHandler = null;
this.descriptionHandler = null;
this.remoteClientDataChannelHandler = null;
this.remoteClientDisconnectHandler = null;
this.remoteClientPacketHandler = null;
}
LANServerSupported() {
return typeof window.RTCPeerConnection !== "undefined";
}
initializeServer() {
// nothing to do!
}
setICEServers(urls) {
this.ICEServers.length = 0;
for (var i = 0; i < urls.length; ++i) {
var etr = urls[i].split(";");
if (etr.length == 1) {
this.ICEServers.push({urls: etr[0]});
} else if (etr.length == 3) {
this.ICEServers.push({urls: etr[0], username: etr[1], credential: etr[2]});
}
}
}
setICECandidateHandler(cb) {
this.iceCandidateHandler = cb;
}
setDescriptionHandler(cb) {
this.descriptionHandler = cb;
}
setRemoteClientDataChannelHandler(cb) {
this.remoteClientDataChannelHandler = cb;
}
setRemoteClientDisconnectHandler(cb) {
this.remoteClientDisconnectHandler = cb;
}
setRemoteClientPacketHandler(cb) {
this.remoteClientPacketHandler = cb;
}
sendPacketToRemoteClient(peerId, buffer) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
if (thePeer.dataChannel != null && thePeer.dataChannel.readyState == "open") {
thePeer.dataChannel.send(buffer);
} else {
this.signalRemoteDisconnect(peerId);
}
}
}
resetPeerStates() {
this.peerState = this.peerStateConnect = this.peerStateInitial = this.peerStateDesc = this.peerStateIce = PEERSTATE_LOADING;
}
getPeerState() {
return this.peerState;
}
getPeerStateConnect() {
return this.peerStateConnect;
}
getPeerStateInitial() {
return this.peerStateInitial;
}
getPeerStateDesc() {
return this.peerStateDesc;
}
getPeerStateIce() {
return this.peerStateIce;
}
signalRemoteConnect(peerId) {
try {
const peerConnection = new RTCPeerConnection({
iceServers: this.ICEServers,
optional: [{DtlsSrtpKeyAgreement: true}]
});
const peerInstance = new EaglercraftLANPeer(this, peerId, peerConnection);
this.peerList.set(peerId, peerInstance);
if (this.peerStateConnect != PEERSTATE_SUCCESS) this.peerStateConnect = PEERSTATE_SUCCESS;
} catch (e) {
if (this.peerStateConnect == PEERSTATE_LOADING) this.peerStateConnect = PEERSTATE_FAILED;
}
}
signalRemoteDescription(peerId, descJSON) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
thePeer.setRemoteDescription(descJSON);
}
}
signalRemoteICECandidate(peerId, candidate) {
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
thePeer.addICECandidate(candidate);
}
}
signalRemoteDisconnect(peerId) {
if (peerId.length == 0) {
for (const thePeer of this.peerList.values()) {
if ((typeof thePeer !== "undefined") && thePeer !== null) {
this.peerList.delete(peerId);
try {
thePeer.disconnect();
} catch (e) {
}
this.remoteClientDisconnectHandler(peerId);
}
}
this.peerList.clear();
return;
}
var thePeer = this.peerList.get(peerId);
if ((typeof thePeer !== "undefined") && thePeer !== null) {
this.peerList.delete(peerId);
try {
thePeer.disconnect();
} catch (e) {
}
this.remoteClientDisconnectHandler(peerId);
}
}
countPeers() {
return this.peerList.size;
}
};
window.constructLANServer = () => new EaglercraftLANServer();
});
window.startLANServer = () => {
if (typeof window.constructLANServer !== "function") {
window.initializeLANServer();
}
return window.constructLANServer();
};

77
pre/favicon.ico Normal file
View File

@ -0,0 +1,77 @@
<!DOCTYPE html>
<html>
<head>
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-261780635-1"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-261780635-1');
</script>
<title>Play Minecraft Online! | Online mc</title>
<link rel="icon" href="/icon.png">
<meta name="description" content="Play Minecraft Online For Free!" />
<meta name="keywords" content="free minecraft online, eaglercraft, minecraft, onlinemc, Onlinemc, ayuncraft, Ayuncraft, mc, eaglercraftx, eaglercraft shaders, eaglercraft download,eaglercraft, eaglercraftx, minecraft, 1.8, 1.8.8, eaglercraft, eaglercraft minecraft, eaglercraft 3kh0, eaglercraft github, eaglercraft singleplayer, eaglercraft hacks, eaglercraft 1.8, eaglercraft server, eaglecraft game, eaglercraft alternative, eaglercraft alt links, eaglercraft asspixel, eaglercraft anarchy server, eaglercraft aeon, eaglercraft asspixel server ip, eaglercraft all links, eaglercraft aeon server, eaglercraft beta, eaglecraft boats, eaglecraft boats for sale, eaglecraft boats for sale bc, eaglecraft bedwars server, eaglecraft boats inc, eaglercraft beta servers, eaglercraft blocked, eaglercraft client, eaglecraft custom boats, eaglecraft campbell river, eaglercraft creative, eaglercraft commands, eaglercraft code, eaglercraft controls, eaglercraft com, eaglercraft download, eaglercraft discord, eaglercraft dupe glitch, eaglercraft download file, eaglercraft download google drive, eaglercraft dmca, eaglercraft deleted, eaglercraft down, eaglercraft embed, eaglercraft epk file download, eaglercraft end of stream, eaglecraft apk download, eaglecraft eagtek, eaglercraft exploit, eaglecraft eggs, eaglercraft epk world file download, eaglercraft file, eaglercraft file download, eaglercraft free, eaglercraft forums, eaglercraft fork, eaglercraft faq, eaglercraft fps, eaglecraft free download, eaglercraft github servers, eaglercraft google drive, eaglercraft github lax1dude, eaglercraft github download, eaglercraft github.io, eaglercraft google sites, minecraft, applet, singleplayer, replit, html5, chromebook, lax1dude, games, eaglercraft singleplayer, eagler, minecraft for free, free minecraft">
<meta property="og:title" content="Online mc" />
<meta property="og:image" content="/icon.png" />
<meta property="og:description" content="Online Minecraft" />
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/css/bootstrap.min.css">
<link href="/style.css" rel="stylesheet" />
<link href="/darkmode.css" rel="stylesheet" />
</head>
<body class="diff-bg">
<div id="loading"></div>
<div class="container container-table">
<div class="row vertical-center-row">
<div class="col-12">
</a>
<div class="d-flex justify-content-center">
</div>
<div class="col-12">
<div class="button-area d-block">
<h1 class="dark-text">404 Not Found!</h1>
<div class="splash-404">...</div>
<h3>&nbsp;</h3>
<a href="/">click here to go back to the main page!</a>
<img class="creeper ms-auto d-block" src="creeper.webp" alt=""/>
</div>
<div class="mc-footer d-flex justify-content-between">
<div class="footer-copyright text-end text-white">This website is not affiliated with Mojang.</div>
<div class="mc-dd">
</div>
</div>
</div>
</div>
</div>
</div>
<script defer src="/_vercel/insights/script.js"></script>
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.1.3/js/bootstrap.bundle.min.js"></script>
<script>
window.onload = function () {
//display loader on page load
$('#loading').fadeOut();
}
setTimeout(function () {
//display loader on page load
$('#loading').fadeOut();
}, 12 * 1000);
</script>
<script src="/darkmode.js"></script>
<script src="/splash-404.js"></script>
</body>
</html>

160
pre/index.html Normal file
View File

@ -0,0 +1,160 @@
<!DOCTYPE html>
<html>
<head>
<title>Precision Client (1.5.2-beta_0.2)</title>
<meta charset="UTF-8" />
<meta content="Precision Client is a custom eaglercraft client with QOL features, custom texture packs, and more."
name="description" />
<meta content="//precisionclient.vercel.app" name="url" />
<meta content="EtcherFX" name="author" />
<meta content="Precision Client" property="og:title" />
<meta content="en-US" property="og:locale" />
<meta content="website" property="og:type" />
<meta content="//precisionclient.vercel.app" property="og:url" />
<!--meta content="IMAGE" property="og:image"/>-->
<meta content="Precision Client is a custom eaglercraft client with QOL features, custom texture packs, and more."
property="og:description" />
<script type="text/javascript" src="eagswebrtc.js"></script>
<script type="text/javascript" src="classes.js"></script>
<script async src="https://arc.io/widget.min.js#pchPqjfw"></script>
<style>
@font-face {
font-family: "Product Sans";
src: url(ProductSans.ttf);
}
</style>
</head>
<body style="margin: 0; width: 100vw; height: 100vh; font-family:
'Product Sans'" id="game_frame" class="gameframe">
<div id="Selection"
style="display: inline-block; position: absolute; top:50%; left: 50%; transform: translate(-50%, -50%)">
<style>
html {
color: #fff;
}
.starter {
border-radius: 5px;
border: 1px solid #fff;
background-color: #0d0c0c;
color: #fff;
}
.dropdown {
position: relative;
display: inline-block;
border-radius: 5px;
border: 1px solid #fff;
background-color: #0d0c0c;
color: #fff;
}
.gameframe {
background-image: url("bg.png");
background-size: cover;
background-repeat: no-repeat;
}
</style>
<h1>Precision Client</h1>
<h3>Select a texture pack:</h3>
<select id="Packs" class="dropdown">
<option>-- Choose one --</option>
<option value="defaultold">Default Old</option>
<option value="defaultnew">Default 1.17+</option>
<option value="bombies">Bombies 180k</option>
<option value="bones">Bare Bones</option>
<option value="modifiednew">Modified 1.17+</option>
<option value="miamiprivate">Miami Private</option>
<option value="nebula">Nebula</option>
<option value="rhedd">Rhedd</option>
<option value="ricefault">Ricefault</option>
<option value="tightfault">Tightfault</option>
<option value="walifault">Walifault</option>
</select>
<br /><br />
<input class="starter" type="button" value="Launch" onclick="Start();" />
</div>
<script type="text/javascript">
if (document.location.href.startsWith("file:")) {
alert("You cannot 'open' this file in your browser, the code doesn't work. Upload this folder to your HTTP(s) server and access it via the internet to launch the stable-download game. This is not a bug, please read the documentation.");
} else {
var PackSelect = document.getElementById("Packs");
function Start() {
if (PackSelect.selectedIndex !== 0) {
var Selected = PackSelect.options[PackSelect.selectedIndex].value;
document.getElementById("Selection").style.display = "none";
localStorage.LastPack = Selected;
const relayId = Math.floor(Math.random() * 3);
window.eaglercraftOpts = {
container: "game_frame",
assetsURI: `packs/${Selected}.epk`,
serverWorkerURI: "worker_bootstrap.js",
worldsFolder: "MAIN",
assetOverrides: {
"records/wait.mp3": "wait.mp3",
"records/mellohi.mp3": "https://stream.nightride.fm/chillsynth.m4a",
"records/far.mp3": "https://stream.nightride.fm/nightride.m4a",
"records/cat.mp3": "http://usa9.fastcast4u.com/proxy/jamz?mp=/1",
"records/ward.mp3": "http://fr4.1mix.co.uk:8000/192h",
"records/strad.mp3": "http://listen.011fm.com:8028/stream15",
"records/blocks.mp3": "https://www.ophanim.net:8444/s/9780",
"records/13.mp3": "https://s2.radio.co/s2b2b68744/listen",
//"records/ward.mp3": wardUrl,
"credits.txt": "assetoverrides/credits.txt",
"precision/gui.png": "assetoverrides/precision/gui.png",
"precision/gui2.png": "assetoverrides/precision/gui2.png",
"precision/guismall.png": "assetoverrides/precision/guismall.png",
"precision/logo.png": "assetoverrides/precision/logo.png",
"precision/precision.png": "assetoverrides/precision/precision.png",
"precision/precision2.png": "assetoverrides/precision/precision2.png",
"lang/en_US.lang": "assetoverrides/lang/en_US.lang",
"gui/demo_bg.png": "assetoverrides/gui/demo_bg.png",
"title/bg/panorama0.png": "assetoverrides/title/bg/panorama0.png",
"title/bg/panorama1.png": "assetoverrides/title/bg/panorama1.png",
"title/bg/panorama2.png": "assetoverrides/title/bg/panorama2.png",
"title/bg/panorama3.png": "assetoverrides/title/bg/panorama3.png",
"title/bg/panorama4.png": "assetoverrides/title/bg/panorama4.png",
"title/bg/panorama5.png": "assetoverrides/title/bg/panorama5.png"
},
servers: [
{ serverName: "A*spixel", serverAddress: "wss://web.asspixel.net/CAPixel/", hideAddress: true },
{ serverName: "VanillaCraft", serverAddress: "wss://play.vanillacraftsmp.org/", hideAddress: true },
{ serverName: "ArchMC", serverAddress: "wss://web.arch.lol/join", hideAddress: true },
{ serverName: "Ayunboom", serverAddress: "wss://sus.shhnowisnottheti.me", hideAddress: true },
{ serverName: "Aeon Network", serverAddress: "wss://aeon-network.net/aeon", hideAddress: true },
{ serverName: "Sealcraft", serverAddress: "wss://sealcraft.ddns.net:442/servers", hideAddress: true }
],
relays: [
{ addr: "wss://relay.deev.is/", name: "lax1dude relay #1", primary: relayId == 0 },
{ addr: "wss://relay.lax1dude.net/", name: "lax1dude relay #2", primary: relayId == 1 },
{ addr: "wss://relay.shhnowisnottheti.me/", name: "ayunami relay #1", primary: relayId == 2 }
],
mainMenu: {
splashes: [], eaglerLogo: false
}
};
(function () {
var q = window.location.search;
if (typeof q === 'string' && q.startsWith("?")) {
q = new URLSearchParams(q);
var s = q.get("server");
if (s) window.minecraftOpts.push(s);
}
})();
main();
} else
alert("You need to select a pack");
}
}
if (localStorage.LastPack)
for (Pack of PackSelect.children)
if (Pack.value === localStorage.LastPack) {
Pack.selected = "selected";
break;
}
</script>
</body>
</html>

BIN
pre/packs/defaultold.epk Normal file

Binary file not shown.

5
pre/worker_bootstrap.js Normal file
View File

@ -0,0 +1,5 @@
onmessage = function (o) {
importScripts("classes_server.js");
eaglercraftServerOpts = o.data;
main();
};

3
shadow/cover.svg Normal file

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 16 KiB

36979
shadow/index.html Normal file

File diff suppressed because one or more lines are too long

BIN
xp/boot/assets/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.4 KiB

BIN
xp/boot/assets/progress.gif Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

29
xp/boot/boot.css Normal file
View File

@ -0,0 +1,29 @@
body {
padding: 0;
margin: 0;
}
._ui_boot {
background: #000000;
width: 100%;
height: 100%;
}
._ui_boot_logo {
height: 150px;
width: 126px;
position: absolute;
top: 50%;
left: 50%;
margin: -75px 0 0 -63px;
}
._ui_boot_winlogo {
background-image: url("assets/logo.png");
width: 118px;
height: 104px;
}
._ui_boot_progress {
background-image: url("assets/progress.gif");
width: 126px;
height: 15px;
position: absolute;
bottom: 0;
}

150
xp/boot/load.js Normal file
View File

@ -0,0 +1,150 @@
var scripts = [
'system/core.js',
'system/window-api.js',
'system/contextmenus.js',
'system/filesystem.js',
'system/xp.js',
'system/jquery.terminal-2.0.0.min.js',
'system/unix_formatting.js',
'program/terminal.js',
'system/script.js',
'system/explorer.js',
'system/help.js',
'program/browser.js',
'program/notepad.js',
'program/imageviewer.js',
'program/mediaplayer.js',
'system/config.js',
'system/controlpanel.js',
'system/uac.js',
'system/uac_app.js',
'system/audio.js',
'system/lambda.js',
'program/minesweeper.js',
'program/paint.js',
'//xpstore.glitch.me/appstore.js',
'boot/boot.js'
];
var stylesheets = [
'system/fonts.css',
'system/xp.css',
'system/icons.css',
'system/widgets.css',
'system/window.css',
'system/contextmenus.css',
'system/cursors.css',
'system/desktop.css',
'system/startmenu.css',
'system/explorer.css',
'system/jquery.terminal-2.0.0.min.css'
];
var requiredDirectories = [];
$(function() {
$('windows').html(`
<div class="_ui_boot">
<div class="_ui_boot_copyright"></div>
<div class="_ui_boot_companylogo"></div>
<center class="_ui_boot_logo">
<div class="_ui_boot_winlogo"></div>
<div class="_ui_boot_progress"></div>
<!--<div style="bottom:0;position:absolute;width:100%;" id="loadingstatus"></div>-->
</center>
</div>`);
console.log('Loading scripts and stylesheets...');
$('<link/>', {rel: 'stylesheet', href: 'boot/boot.css'}).appendTo('head');
var scriptsindex = 0;
var stylesindex = 0;
function loadStylesheets() {
$.ajax({
url: stylesheets[stylesindex],
dataType: "script",
success: function(data){
$("head").append("<style>" + data + "</style>");
stylesindex ++;
if (stylesindex < stylesheets.length) {
loadStylesheets();
} else {
loadScripts();
}
}
});
}
function loadScripts() {
$.ajax({
url: scripts[scriptsindex],
dataType: "script",
success: function(data){
scriptsindex ++;
if (scriptsindex < scripts.length) {
loadScripts();
} else {
console.log('Finished loading');
console.log('Checking for necessary directories');
function checkDir(path, callback) {
var times = 0;
xp.filesystem.listDir(path, (e) => {
if (times === 0)
callback(typeof e === 'string');
times ++;
});
}
var i = 0;
function checkNextDir(t) {
var dirToCreate = requiredDirectories[i];
if (dirToCreate !== undefined) {
xp.filesystem.createDir(dirToCreate, (e) => {
i ++;
checkNextDir();
});
} else {
xp.audio.init();
var event = new Event('xpboot');
window.dispatchEvent(event);
console.log('Dispatched boot event');
$('windows').html('<div class="_ui_wallpaper fullscreen"><img class="_ui_wallpaper_image" src="boot/assets/welcome.png"/></div>');
$.getScript('system/login.js');
xp.audio.playURL('boot/assets/startup.wav');
}
}
xp.filesystem.create(512*1024*1024, () => {
xp.filesystem.fs.root.getDirectory('/', {create: false}, function(dirEntry) {
var dirReader = dirEntry.createReader();
var entries = [];
function readEntries() {
dirReader.readEntries (function(results) {
if (results.length === 0) {
$('._ui_boot').remove();
$('windows').html('<div class="_ui_wallpaper fullscreen"><img class="_ui_wallpaper_image" src="boot/assets/welcome.png"/></div>');
$.getScript('setup/setup.js');
} else {
xp.filesystem.createDir('/WINDOWS', (e) => {
requiredDirectories = [
'/WINDOWS',
'/WINDOWS/system32',
'/WINDOWS/startup',
'/Program Files',
'/Documents and Settings'
];
checkNextDir();
});
}
}, console.error);
};
readEntries();
}, console.error);
});
}
}
});
}
loadStylesheets();
});

3
xp/cover.svg Normal file
View File

@ -0,0 +1,3 @@
<svg width="200" height="200" xmlns="http://www.w3.org/2000/svg">
<image href="" height="200" width="200" />
</svg>

After

Width:  |  Height:  |  Size: 1.9 KiB

24
xp/index.html Normal file
View File

@ -0,0 +1,24 @@
<html>
<head>
<title>RebornXP</title>
<meta http-equiv="Content-Language" content="en_US"/>
<meta name="google" content="notranslate"/>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.0/jquery.min.js" integrity="sha512-3gJwYpMe3QewGELv8k/BX9vcqhryRdzRMxVfq6ngyWXwo03GFEzjsUm8Q7RZcHPHksttq7/GFoxjCVUjkjvPdw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script src="boot/load.js"></script>
<link rel="shortcut icon" href="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ffavicon.png?1521937672120"/>
<meta property="og:image" content="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ffavicon.png?1521937672120"/>
<meta property="og:site_name" content="RebornXP"/>
<meta property="og:type" content="object"/>
<meta property="og:title" content="reborn-xp.js"/>
<meta property="og:url" content="https://reborn-xp.js.org/"/>
<meta property="og:description" content=" RebornXP is a free and open source project that brings the beloved Windows XP experience to modern web browsers. Built with JavaScript, RebornXP faithfully recreates the iconic interface, apps, and features of Windows XP."/>
</head>
<body style="background-color:#000000;color:#ffffff;">
<windows>
<noscript>
<br/>
<center>RebornXP requires Javascript. Please update your web browser.</center>
</noscript>
</windows>
</body>
</html>

102
xp/program/browser.js Normal file
View File

@ -0,0 +1,102 @@
function browserGo(loc, guid) {
if(!loc){
loc = document.getElementById('url_' + guid).value;
}
if (loc.indexOf('http://') == -1 && loc.indexOf('https://') == -1) {
loc = location.protocol + '//' + loc;
}
document.getElementById('webPage_' + guid).src = loc;
document.getElementById('url_' + guid).value = loc;
document.getElementById('loading_' + guid).style.display = 'block';
document.getElementById('webPage_' + guid).style.display = 'none';
}
function iframeContentUnload(guid) {
document.getElementById('loading_' + guid).style.display = 'block';
document.getElementById('webPage_' + guid).style.display = 'none';
}
function iframeContentLoaded(guid) {
document.getElementById('loading_' + guid).style.display = 'none';
document.getElementById('webPage_' + guid).style.display = 'block';
}
$(window).on('xpboot', function() {
xp.applications.add('browser', (args) => {
var guid = generate_guid();
var el = $.parseHTML(`<window title="Idiot Explorer 6" width="800" height="500">
<style>
.load-overlay {
position: absolute;
top: 24;
left: 0;
width: 100%;
height: calc(100% - 24px);
background: #d6d6ce;
z-index: 9999;
text-align: center;
}
.load-overlay-cell {
position: absolute;
top: calc(50% - 100px);
left: calc(50% - 100px);
}
@keyframes lds-dual-ring{0%{-webkit-transform:rotate(0);transform:rotate(0)}100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@-webkit-keyframes lds-dual-ring{0%{-webkit-transform:rotate(0);transform:rotate(0)}100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}.lds-dual-ring{position:relative}.lds-dual-ring div{position:absolute;width:40px;height:40px;top:80px;left:80px;border-radius:50%;border:4px solid #000;border-color:#51CACC transparent;-webkit-animation:lds-dual-ring 1s linear infinite;animation:lds-dual-ring 1s linear infinite}.lds-dual-ring{width:200px!important;height:200px!important;-webkit-transform:translate(-100px,-100px) scale(1) translate(100px,100px);transform:translate(-100px,-100px) scale(1) translate(100px,100px)}
</style>
<div id="webBrowserContent">
<button align="left" id="homeButton" onclick="browserGo('search.anasdew.com', '` + guid + `')" title="Home">Home</button>
<button align="left" onclick="browserGo('www.google.com/?igu=1', '` + guid + `')" title="Google">Google</button>
<button align="left" onclick="browserGo('www.wikipedia.com', '` + guid + `')" title="Wikipedia">Wikipedia</button>
<button align="left" id="videoButton" onclick="browserGo('www.bing.com/videos', '` + guid + `')" title="Videos">Videos</button>
<button align="left" id="imageButton" onclick="browserGo('www.bing.com/images', '` + guid + `')" title="Images">Images</button>
<input type="text" id="url_` + guid + `" value="` + location.protocol + `//search.anasdew.com" style="width:calc(100% - 600px);"/>
<button id="goButton" onclick="browserGo(undefined, '` + guid + `')" title="Go">Go</button><br>
<div style="width:calc(100% - 5px);height:calc(100% - 27px);">
<div style="display: none" id="loading_` + guid + `" title="Loading..." class="load-overlay">
<div class="load-overlay-cell">
<div class="lds-css ng-scope"><div style="width:100%;height:100%" class="lds-dual-ring"><div></div></div></div>
</div>
</div>
<iframe style="width:calc(100% - 5px);height:calc(100% - 29px);position:absolute;top:24;left:0;" onload="iframeContentLoaded('` + guid + `')" onunload="iframeContentUnload('` + guid + `')" src="` + location.protocol + `//search.anasdew.com" id="webPage_` + guid + `"></iframe>
</div>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
var widthOffset = 30;
var heightOffset = 80;
window.iFrameChanges = -1;
});
xp.applications.add('html', (args) => {
var el = $.parseHTML(`<window title="` + args[1] + `" width="800" height="500">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe seamless="seamless" width="100%" height="100%" id="frame" src="` + args[1] + `"></iframe>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
});
xp.startmenu.add('browser', 'Idiot Explorer 6', 'program/assets/ie.webp');
});

18
xp/program/imageviewer.js Normal file
View File

@ -0,0 +1,18 @@
$(window).on('xpboot', () => {
xp.applications.add('image-viewer', (args) => {
var image = args.length > 1 ? xp.filesystem.addPaths(args[0], args[1]) : '';
var imagename = xp.filesystem.basename(image);
var win = new Window({
width: 640,
height: 480,
title: image === '' ? 'Image Viewer' : (imagename + ' - Image Viewer')
});
xp.filesystem.toURL(image, (url) => {
win.content('<div style="width:100%;height:100%;background-size:100%;background-repeat:no-repeat;background-position:center;background-image:url(' + url + ')"></div>');
});
});
explorer.fileHandlers.add('png', (f) => openApp('image-viewer', ['', f]));
explorer.fileHandlers.add('jpg', (f) => openApp('image-viewer', ['', f]));
explorer.fileHandlers.add('jpeg', (f) => openApp('image-viewer', ['', f]));
explorer.fileHandlers.add('gif', (f) => openApp('image-viewer', ['', f]));
});

23
xp/program/mediaplayer.js Normal file
View File

@ -0,0 +1,23 @@
$(window).on('xpboot', () => {
xp.applications.add('media-player', (args) => {
var video = args.length > 1 ? xp.filesystem.addPaths(args[0], args[1]) : '';
var videoname = xp.filesystem.basename(video);
var win = new Window({
width: 480,
height: 360,
title: video === '' ? 'Media Player' : (videoname + ' - Media Player')
});
xp.filesystem.toURL(video, (url) => {
win.content(`<video width="100%" height="100%" style="background-color:#000;" controls>
<source src="` + url + `">
</video>`);
});
});
explorer.fileHandlers.add('webm', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('wav', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('ogg', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('ogv', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('mp3', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('mp4', (f) => openApp('media-player', ['', f]));
explorer.fileHandlers.add('flac', (f) => openApp('media-player', ['', f]));
});

30
xp/program/minesweeper.js Normal file
View File

@ -0,0 +1,30 @@
$(window).on('xpboot', () => {
xp.applications.add('minesweeper', () => {
var el = $.parseHTML(`<window width="300" height="410" title="Minesweeper">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe height="100%" seamless="seamless" width="100%" src="https://mines.vercel.app/"></iframe>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
});
xp.startmenu.add('minesweeper', 'Minesweeper', 'https://w7.pngwing.com/pngs/337/559/png-transparent-minesweeper-computer-icons-bing-maps-video-game-mines-miscellaneous-game-angle.png');
});

56
xp/program/notepad.js Normal file
View File

@ -0,0 +1,56 @@
$(window).on('xpboot', () => {
xp.applications.add('notepad', (args) => {
var guid = generate_guid();
var el = $.parseHTML(`<window title="Notepad" width="480" height="360">
<textarea style="width: 100%; height: calc(100% - 19px); resize: none" id="notepad_` + guid + `" filename=""></textarea>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
$(el).addMenu("File", [
["New", function() {
$("#notepad_" + guid).val("");
$("#notepad_" + guid).attr('filename', '');
}],
["Open", function() {
xp.filesystem.openFileDialog((filename) => {
xp.filesystem.readFile(filename, function(e) {
//e = e.split("\0")[0];
$("#notepad_" + guid).val(e);
$("#notepad_" + guid).attr("filename", filename);
});
});
}],
["Save", function() {
var filename = $("#notepad_" + guid).attr("filename");
if (filename === "") {
xp.filesystem.saveFileDialog((filename) => {
xp.filesystem.writeFile(filename, new Blob([$("#notepad_" + guid).val()], {type: 'text/plain'}), (e) => {
if (e) xp.dialog('Error', e);
});
});
} else {
xp.filesystem.writeFile(filename, new Blob([$("#notepad_" + guid).val()], {type: 'text/plain'}), (e) => {
if (e) xp.dialog('Error', e);
});
}
}],
["Save As", function() {
xp.filesystem.saveFileDialog((filename) => {
xp.filesystem.writeFile(filename, new Blob([$("#notepad_" + guid).val()], {type: 'text/plain'}), (e) => {
if (e) xp.dialog('Error', e);
});
});
}]
]);
if (args !== undefined && args[1] !== undefined) {
var filename = xp.filesystem.addPaths(args[0], args[1]);
xp.filesystem.readFile(filename, function(e) {
//e = e.split("\0")[0];
$("#notepad_" + guid).val(e);
$("#notepad_" + guid).attr("filename", filename);
});
}
});
xp.startmenu.add('notepad', 'Notepad', '//cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fnotepad.png?1521328884236');
});

30
xp/program/paint.js Normal file
View File

@ -0,0 +1,30 @@
$(window).on('xpboot', () => {
xp.applications.add('paint', () => {
var el = $.parseHTML(`<window width="800" height="500" title="JS Paint">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe height="100%" seamless="seamless" width="100%" src="https://jspaint.app/"></iframe>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
});
xp.startmenu.add('paint', 'Paint', 'https://raw.githubusercontent.com/1j01/jspaint/master/images/icons/512x512.png');
});

94
xp/program/terminal.js Normal file
View File

@ -0,0 +1,94 @@
var terminalicon = '';
$(window).on('xpboot', () => {
xp.applications.add("terminal", () => {
var guid = generate_guid();
var el = $.parseHTML(`<window title="Terminal" width="720" height="480">
<div class="ui_terminal" id="terminal_` + guid + `"></div>
</window>`);
document.body.appendChild(el[0]);
el = $(el);
el.updateWindow();
$("#terminal_" + guid).terminal(function(command) {
var term = this;
if (command !== '') {
try {
var args = command.split(" ");
if (this.dir === undefined)
this.dir = [""];
if (args[0] === "ls") {
xp.filesystem.listDir(xp.filesystem.getDir(this.dir), function(e){if(e){term.echo(e)}});
} else if (args[0] === "cd") {
if (args[1] === undefined) {
this.error("Not enough arguments");
} else {
let input = args.slice(1).join(' ');
if (input !== ".") {
if (input.charAt(0) === "/") {
this.dir = input.split("/");
} else {
var foo = input.split("/");
var _this = this;
foo.forEach(function(val, i) {
if (val === "..") {
_this.dir.splice(-1,1)
} else {
_this.dir = _this.dir.concat(val);
}
});
}
}
}
} else if (args[0] === "pwd") {
this.echo(xp.filesystem.getDir(this.dir));
} else if (args[0] === "touch") {
xp.filesystem.createFile(xp.filesystem.addPaths(xp.filesystem.getDir(this.dir), args[1]), function(e){if(e){term.echo(e)}});
} else if (args[0] === "rm") {
xp.filesystem.deleteFile(xp.filesystem.addPaths(xp.filesystem.getDir(this.dir), args[1]), function(e){if(e){term.echo(e)}});
} else if (args[0] === "mkdir") {
xp.filesystem.createDir(xp.filesystem.addPaths(xp.filesystem.getDir(this.dir), args[1]), function(e){if(e){term.echo(e)}});
} else if (args[0] === "rmdir") {
xp.filesystem.deleteDir(xp.filesystem.addPaths(xp.filesystem.getDir(this.dir), args[1]), function(e){if(e){term.echo(e)}});
} else if (args[0] === "cat") {
xp.filesystem.readFile(xp.filesystem.addPaths(xp.filesystem.getDir(this.dir), args[1]), function(e){if(e){term.echo(e)}});
} else if (args[0] === "help") {
this.echo("DE: XPerience");
this.echo("Kernel: RXP vOS.2023");
this.echo("Shell: JQuery Terminal");
this.echo("GitHub: github.com/shoaib-jamal/RebornXP");
this.echo("\nInstalled applications:");
var foo = "";
for (var name in xp.applications.apps) {
foo += name + ", ";
}
foo = foo.slice(0, -2);
this.echo(foo);
this.echo("\nSystem commands:");
this.echo("ls, cd, pwd, touch, rm, mkdir, rmdir, cat, help");
} else if (xp.applications.apps[args[0]] !== undefined) {
xp.applications.apps[args[0]]([xp.filesystem.getDir(this.dir)].concat(args.slice(1)));
} else {
var result = window.eval(command);
if (result !== undefined) {
this.echo(new String(result));
}
}
} catch(e) {
this.error(new String(e));
}
} else {
this.echo('');
}
}, {
greetings: `
RebornXP [vOS.2023]
(c) RebornXP Open-Source Project. All rights reserved.
`,
name: 'xpjs_terminal',
width: '100%',
height: '100%',
prompt: '> '
});
});
xp.startmenu.add("terminal", "Terminal", terminalicon);
});

38
xp/system/audio.js Normal file
View File

@ -0,0 +1,38 @@
xp.audio = {
context: null,
cache: {},
init: function() {
try {
window.AudioContext = window.AudioContext || window.webkitAudioContext;
xp.audio.context = new AudioContext();
}
catch(e) {
console.error('Web Audio API is not supported in this browser');
}
},
playURL: function(url) {
if (xp.audio.cache[url] !== undefined) {
var source = xp.audio.context.createBufferSource();
source.buffer = xp.audio.cache[url];
source.connect(xp.audio.context.destination);
source.start(0);
} else {
var request = new XMLHttpRequest();
request.open('GET', url, true);
request.responseType = 'arraybuffer';
request.onload = function() {
if (xp.audio.context === null) {
xp.audio.init();
}
xp.audio.context.decodeAudioData(request.response, function(buffer) {
var source = xp.audio.context.createBufferSource();
source.buffer = buffer;
xp.audio.cache[url] = buffer;
source.connect(xp.audio.context.destination);
source.start(0);
}, console.error);
}
request.send();
}
}
};

34
xp/system/config.js Normal file
View File

@ -0,0 +1,34 @@
var configFile = '';
var config = {};
var loadingconfig = false;
function saveConfig(callback) {
if (!loadingconfig) {
config.wallpaper = xp.wallpaper.href;
config.profile = xp.profile;
config.theme = xp.theme.name;
xp.filesystem.writeFile(configFile, new Blob([JSON.stringify(config)], {type: 'text/plain'}), (e) => {
if (e) xp.error(e);
else if (callback !== undefined) callback()
});
}
}
function loadConfig(callback) {
loadingconfig = true;
xp.filesystem.readFile(configFile, (text) => {
try {
config = JSON.parse(text);
console.log(config);
xp.wallpaper.set('https://rebornxp.js.org/system/themes/XP.jpg');
xp.profile = config.profile;
xp.theme.set('reborn');
} catch(e) {
xp.wallpaper.set('https://rebornxp.js.org/system/themes/XP.jpg');
loadingconfig = false;
saveConfig();
}
loadingconfig = false;
if (callback !== undefined) callback();
});
}

View File

@ -0,0 +1,19 @@
.contextMenu {
box-shadow: 0px 0px 20px -5px #000;
clip-path: inset(0 -20px -20px 0);
}
.contextMenuItem {
color: #000000;
background-color: transparent;
font-size: 11px;
}
.contextMenuItem:hover {
color: #ffffff;
background-color: #0A246A;
}
.contextMenuItem.hr {
border-top: 1px solid #979797;
}

196
xp/system/contextmenus.js Normal file
View File

@ -0,0 +1,196 @@
var vContextMenu = Array();
var isContextMenu = false;
var CurX;
var CurY;
var tContextMenu;
function fMouseDown(event) {
try {
if(window.event) event = window.event;
CurX = event.clientX;
CurY = event.clientY;
if (document.getElementById('lContextMenu').style.visibility === 'hidden')
tContextMenu = event.target;
} catch(e) {}
}
function rcContextMenu() {
try {
if (vContextMenu.length != 0) {
drawContextMenu();
var targX = CurX; //0; //event.clientX;
var targY = CurY; //0; //event.clientY;
//alert
var testW = document.getElementById('lContextMenu').offsetWidth;
var testH = document.getElementById('lContextMenu').offsetHeight;
var testX = targX + document.getElementById('lContextMenu').offsetWidth;
var testY = targY + document.getElementById('lContextMenu').offsetHeight;
var sW = document.body.clientWidth;//document.body.clientWidth;
var sH = document.body.clientHeight;//document.body.clientHeight;
if (testX >= sW) { targX = targX - testW; }
if (testY >= sH) { targY = targY - testH; }
document.getElementById('lContextMenu').style.left = targX;
document.getElementById('lContextMenu').style.top = targY;
document.getElementById('lContextMenu').style.visibility = 'visible';
}
} catch (e) {}
}
function rcCloseContext() {
try {
document.getElementById('lContextMenu').style.left = 0;
document.getElementById('lContextMenu').style.top = 0;
document.getElementById('lContextMenu').style.visibility = 'hidden';
} catch (e) {}
}
function drawContextMenu() {
try {
var outMsg = '<table cellpadding="0" cellspacing="0" style="border: 1px solid #979797;"><tr><td><table cellpadding="0" cellspacing="0" style="border: 2px solid #f5f5f5;"><tr><td><table cellpadding="4" cellspacing="0" border="0" width="200" bgcolor="#f1f1f1" style="font-size: 10;"><tr><td rowspan="100" width="15" style="border-right: 1px solid #979797;" nowrap><img width="15" height="1"></td>';
var len = vContextMenu.length - 1;
var inMsg = vContextMenu[0];
var inCmd = vContextMenu[1];
if (typeof inCmd !== 'string') {
outMsg = outMsg + '<td class="contextMenuItem" onClick="rcCloseContext();" commandindex="1" nowrap>' + inMsg + '</td></tr>';
} else {
outMsg = outMsg + '<td class="contextMenuItem" onClick="rcCloseContext();' + inCmd + ';" nowrap>' + inMsg + '</td></tr>';
}
for (var i = 2; i <= len; i ++) {
inMsg = vContextMenu[i];
inCmd = vContextMenu[i+1];
if (inMsg != '-HR-') {
if (typeof inCmd !== 'string') {
var outMsg = outMsg + '<td class="contextMenuItem" commandindex="' + (i + 1) + '" nowrap>' + inMsg + '</td></tr>';
} else {
var outMsg = outMsg + '<td class="contextMenuItem" onClick="rcCloseContext();' + inCmd + ';" nowrap>' + inMsg + '</td></tr>';
}
i ++;
} else {
i ++;
inMsg = vContextMenu[i];
inCmd = vContextMenu[i+1];
if (typeof inCmd !== 'string') {
outMsg = outMsg + '<td class="contextMenuItem hr" commandindex="' + (i + 1) + '" nowrap>' + inMsg + '</td></tr>';
} else {
outMsg = outMsg + '<td class="contextMenuItem hr" onClick="rcCloseContext();' + inCmd + ';" nowrap>' + inMsg + '</td></tr>';
}
i ++;
}
}
outMsg = outMsg + '</table></td></tr></table></td></tr></table>';
document.getElementById('lContextMenu').innerHTML = outMsg;
// JQuery magic
$('#lContextMenu').find('td[commandindex]').each(function() {
$(this).on('click', function() {
rcCloseContext();
vContextMenu[parseInt($(this).attr('commandindex'))]();
});
});
} catch(e) {}
}
function desktopContextMenu(e) {
var el = $(e.target);
if (el.closest('windowbutton').length > 0) {
var guid = el.closest('windowbutton').attr('guid');
vContextMenu = Array();
if ($('window[guid=' + guid + ']').attr('maximized') === 'true') {
vContextMenu.push('Restore');
vContextMenu.push('maximizeWindow(\'' + guid + '\')');
} else if ($('window[guid=' + guid + ']').find('.maximize').length > 0 || $('window[guid=' + guid + ']').find('.restore').length > 0) {
vContextMenu.push('Maximize');
vContextMenu.push('maximizeWindow(\'' + guid + '\')');
}
if ($('window[guid=' + guid + ']').attr('minimized') === 'true') {
vContextMenu.push('Restore');
vContextMenu.push('moveWindowToTop(\'' + guid + '\')');
} else if ($('window[guid=' + guid + ']').find('.minimize').length > 0) {
vContextMenu.push('Minimize');
vContextMenu.push('minimizeWindow(\'' + guid + '\')');
}
if ($('window[guid=' + guid + ']').find('.close').length > 0) {
vContextMenu.push('Close');
vContextMenu.push('closeWindow(\'' + guid + '\')');
}
rcContextMenu();
} else if (el.closest('.ui_icon').length > 0) {
el = el.closest('.ui_icon');
vContextMenu = Array('<b>Open</b>', () => eval(el.data('exe')), 'Explore', () => openLocation(el.data('target')), '-HR-', 'Create Shortcut', '', 'Delete',() => el.remove(), 'Rename', () => xp.prompt('Rename', 'Enter the new name:', (t) => {el.find('span').html(t)}), '-HR-');
rcContextMenu();
}
}
function getSelectionText() {
var text = "";
var activeEl = document.activeElement;
var activeElTagName = activeEl ? activeEl.tagName.toLowerCase() : null;
if (
(activeElTagName == "textarea") || (activeElTagName == "input" &&
/^(?:text|search|password|tel|url)$/i.test(activeEl.type)) &&
(typeof activeEl.selectionStart == "number")
) {
text = activeEl.value.slice(activeEl.selectionStart, activeEl.selectionEnd);
} else if (window.getSelection) {
text = window.getSelection().toString();
}
return text;
}
function windowContextMenu(e, guid) {
var el = $('window[guid=' + guid + ']');
var t = e.target.nodeName;
console.log(t);
vContextMenu = Array();
var type = ($(e.target).attr('type') || '').toLowerCase();
if (t === 'TEXTAREA' || (t === 'INPUT' && (type === 'text' || type === 'number'))) {
vContextMenu.push('Copy');
vContextMenu.push('document.execCommand(\'copy\')');
vContextMenu.push('Paste');
vContextMenu.push('document.execCommand(\'paste\')');
vContextMenu.push('-HR-');
}
if (el.attr('maximized') === 'true') {
vContextMenu.push('Restore');
vContextMenu.push('maximizeWindow(\'' + guid + '\')');
} else if (el.find('.maximize').length > 0) {
vContextMenu.push('Maximize');
vContextMenu.push('maximizeWindow(\'' + guid + '\')');
}
if (el.attr('minimized') === 'true') {
vContextMenu.push('Restore');
vContextMenu.push('moveWindowToTop(\'' + guid + '\')');
} else if (el.find('.minimize').length > 0) {
vContextMenu.push('Minimize');
vContextMenu.push('minimizeWindow(\'' + guid + '\')');
}
if (el.find('.close').length > 0) {
vContextMenu.push('Close');
vContextMenu.push('closeWindow(\'' + guid + '\')');
}
rcContextMenu();
}
function startContextMenu(e) {
var el = $(e.target);
if (!(el.is('li.startmenuitem') || el.is('li.startmenuitem>img') || el.is('li.startmenuitem>div') || el.is('li.startmenuitem>div>div') || el.is('li.startmenuitem>div>div>b'))) return;
if (!el.is('li.startmenuitem'))
el = el.closest('li');
vContextMenu = Array('<b>Open</b>', el.attr('onclick'), 'Explore', el.attr('onclick'), '-HR-', 'Create Shortcut', '', 'Delete',() => el.remove(), 'Rename', () => xp.prompt('Rename', 'Enter the new name:', (t) => {el.find('div>div>b').html(t)}), '-HR-');
rcContextMenu();
}

254
xp/system/controlpanel.js Normal file
View File

@ -0,0 +1,254 @@
var configFile = 'config.json';
// Load the config on startup
$(window).on('xpboot', function() {
loadConfig();
});
function saveConfig(callback) {
if (!loadingconfig) {
config.wallpaper = xp.wallpaper.href;
config.profile = xp.profile;
config.theme = xp.theme.name;
xp.filesystem.writeFile(configFile, new Blob([JSON.stringify(config)], {type: 'text/plain'}), (e) => {
if (e) xp.error(e);
else if (callback !== undefined) callback();
});
}
}
function loadConfig(callback) {
loadingconfig = true;
xp.filesystem.readFile(configFile, (text) => {
try {
config = JSON.parse(text);
xp.wallpaper.set(config.wallpaper || 'https://rebornxp.js.org/system/themes/XP.jpg');
xp.profile = config.profile;
xp.theme.set(config.theme || 'reborn');
} catch (e) {
xp.wallpaper.set('https://rebornxp.js.org/system/themes/XP.jpg');
loadingconfig = false;
saveConfig();
}
loadingconfig = false;
if (callback !== undefined) callback();
});
}
// Add theme chooser to Control Panel
xp.controlpanel.add('Themes', () => {
var win = new Window({
width: 400,
height: 200, // Increased height to accommodate the wallpaper change option
title: 'Change Theme',
canResize: false,
});
win.content(`
<div class="content">
<h1>Theme Chooser</h1>
<p>Choose a new theme to apply:</p>
<select id="themeSelect">
${xp.themes.map(theme => `<option value="${theme}">${theme}</option>`).join('')}
</select>
<br/><br/>
<button id="applyThemeBtn">Apply Theme</button>
<button id="changeWallpaperBtn">Change Wallpaper</button> <!-- Added button for wallpaper change -->
</div>
`);
win.el.find('#applyThemeBtn').on('click', function() {
const selectedTheme = win.el.find('#themeSelect').val();
xp.theme.set(selectedTheme);
saveConfig();
win.close();
});
win.el.find('#changeWallpaperBtn').on('click', function() {
const wallpaperDialog = new Window({
width: 400,
height: 250,
title: 'Change Wallpaper',
canResize: false,
});
wallpaperDialog.content(`
<center><div class="content">
<h1>Wallpaper Chooser</h1>
<p>Select a wallpaper:</p>
<div class="wallpaper-thumbnails">
<!-- Thumbnails will be dynamically added here -->
</div>
<br/>
<input type="text" id="wallpaperLink" placeholder="Enter wallpaper link">
<button id="applyWallpaperBtn">Apply Wallpaper</button>
</div></center>
`);
const thumbnailsContainer = wallpaperDialog.el.find('.wallpaper-thumbnails');
// Add your wallpaper thumbnails here using the following format
const wallpaperThumbnails = [
'system/themes/XP.jpg',
'system/themes/wallpapers/anniversary.png',
'system/themes/wallpapers/3.jpg',
'system/themes/wallpapers/4.jpg',
'system/themes/wallpapers/5.jpg',
'system/themes/wallpapers/6.jpg',
];
wallpaperThumbnails.forEach(thumbnail => {
const thumbnailElement = $('<img>')
.attr('src', thumbnail)
.addClass('wallpaper-thumbnail')
.attr('width', "100px")
.attr('height', "60px")
.on('click', function() {
const selectedWallpaper = $(this).attr('src');
xp.wallpaper.set(selectedWallpaper);
saveConfig();
});
thumbnailsContainer.append(thumbnailElement);
});
wallpaperDialog.el.find('#applyWallpaperBtn').on('click', function() {
const wallpaperLink = wallpaperDialog.el.find('#wallpaperLink').val();
if (wallpaperLink) {
xp.wallpaper.set(wallpaperLink);
saveConfig();
}
wallpaperDialog.close();
});
});
// Calculate window position to center it on the screen
const screenWidth = $(window).width();
const screenHeight = $(window).height();
const windowWidth = win.el.outerWidth();
const windowHeight = win.el.outerHeight();
const left = (screenWidth - windowWidth) / 2;
const top = (screenHeight - windowHeight) / 2;
win.el.css({
position: 'absolute',
left: left + 'px',
top: top + 'px',
});
win.el.find('.content').css({
padding: '20px', // Adjust the padding value as needed
});
});
// Add theme chooser as an application
xp.applications.add('themes', () => {
var win = new Window({
width: 400,
height: 200, // Increased height to accommodate the wallpaper change option
title: 'Change Theme',
canResize: false,
});
win.content(`
<div class="content">
<h1>Theme Chooser</h1>
<p>Choose a new theme to apply:</p>
<select id="themeSelect">
${xp.themes.map(theme => `<option value="${theme}">${theme}</option>`).join('')}
</select>
<br/><br/>
<button id="applyThemeBtn">Apply Theme</button>
<button id="changeWallpaperBtn">Change Wallpaper</button> <!-- Added button for wallpaper change -->
</div>
`);
win.el.find('#applyThemeBtn').on('click', function() {
const selectedTheme = win.el.find('#themeSelect').val();
xp.theme.set(selectedTheme);
saveConfig();
win.close();
});
win.el.find('#changeWallpaperBtn').on('click', function() {
const wallpaperDialog = new Window({
width: 400,
height: 250,
title: 'Change Wallpaper',
canResize: false,
});
wallpaperDialog.content(`
<center><div class="content">
<h1>Wallpaper Chooser</h1>
<p>Select a wallpaper:</p>
<div class="wallpaper-thumbnails">
<!-- Thumbnails will be dynamically added here -->
</div>
<br/>
<input type="text" id="wallpaperLink" placeholder="Enter wallpaper link">
<button id="applyWallpaperBtn">Apply Wallpaper</button>
</div></center>
`);
const thumbnailsContainer = wallpaperDialog.el.find('.wallpaper-thumbnails');
// Add your wallpaper thumbnails here using the following format
const wallpaperThumbnails = [
'system/themes/XP.jpg',
'system/themes/wallpapers/anniversary.png',
'system/themes/wallpapers/3.jpg',
'system/themes/wallpapers/4.jpg',
'system/themes/wallpapers/5.jpg',
'system/themes/wallpapers/6.jpg',
];
wallpaperThumbnails.forEach(thumbnail => {
const thumbnailElement = $('<img>')
.attr('src', thumbnail)
.addClass('wallpaper-thumbnail')
.attr('width', "100px")
.attr('height', "60px")
.on('click', function() {
const selectedWallpaper = $(this).attr('src');
xp.wallpaper.set(selectedWallpaper);
saveConfig();
});
thumbnailsContainer.append(thumbnailElement);
});
wallpaperDialog.el.find('#applyWallpaperBtn').on('click', function() {
const wallpaperLink = wallpaperDialog.el.find('#wallpaperLink').val();
if (wallpaperLink) {
xp.wallpaper.set(wallpaperLink);
saveConfig();
}
wallpaperDialog.close();
});
});
// Calculate window position to center it on the screen
const screenWidth = $(window).width();
const screenHeight = $(window).height();
const windowWidth = win.el.outerWidth();
const windowHeight = win.el.outerHeight();
const left = (screenWidth - windowWidth) / 2;
const top = (screenHeight - windowHeight) / 2;
win.el.css({
position: 'absolute',
left: left + 'px',
top: top + 'px',
});
win.el.find('.content').css({
padding: '20px', // Adjust the padding value as needed
});
});

620
xp/system/core.js Normal file
View File

@ -0,0 +1,620 @@
var xp = {
post: true,
version: "2.0-beta",
openFileHandler: (path) => {},
applications: {
apps: {},
add: function(name, func) {
xp.applications.apps[name] = func;
},
remove: function(name) {
delete xp.applications.apps[name];
}
},
themes: ['reborn', 'classic', 'luna', 'silver', 'vista'],
theme: {
name: 'luna',
set: function(name) {
xp.theme.name = name;
if (xp.theme.name !== 'default') {
$('#theme').attr('href', 'system/themes/' + xp.theme.name + '.css');
} else {
$('#theme').attr('href', '');
}
if (saveConfig) saveConfig();
}
},
wallpaper: {
href: "",
set: function(name) {
if (name === undefined) {
xp.wallpaper.set(xp.wallpaper.href);
} else if (name === "") {
$("._ui_wallpaper_image").attr("src", "");
xp.wallpaper.href = "";
} else {
$("._ui_wallpaper_image").attr("src", name);
xp.wallpaper.href = name;
}
if (saveConfig) saveConfig();
},
setLocal: function(path) {
xp.filesystem.toURL(path, (url) => {
xp.wallpaper.set(url);
});
}
},
icons: {
error: '',
warning: '',
info: ''
},
dialog: function(title, text, callback, isyesno, type, nocallback) {
if (type === undefined) type = 'info';
var icon = undefined;
if (type === 'info') icon = xp.icons.info;
if (type === 'warning') icon = xp.icons.warning;
if (type === 'error') icon = xp.icons.error;
if (type === 'info') xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Ding.wav?1522624603096');
if (type === 'warning') xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Error.wav?1522624607974');
if (type === 'error') xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Critical%20Stop.wav?1522624592867');
var win = new Window({
width: 340,
height: 154,
title: title,
canResize: false,
canMinimize: false
});
if (isyesno) {
win.content(`<img draggable="false" style="position: absolute;left:11px;top:17px;" src="` + icon + `"/>
<p style="position:absolute;left:54px;top:8px;width:calc(100% - 54px);word-wrap:break-word;height:calc(100% - 68px);overflow-y:auto;">` + text + `</p>
<center style="position:absolute;bottom:17px;right:16.7px;">
<button class="yes">Yes</button>
<button class="no">No</button>
</center>`);
} else {
win.content(`<img draggable="false" style="position: absolute;left:11px;top:17px;" src="` + icon + `"/>
<p style="position:absolute;left:54px;top:8px;width:calc(100% - 54px);word-wrap:break-word;height:calc(100% - 68px);overflow-y:auto;">` + text + `</p>
<center style="position:absolute;bottom:17px;right:16.7px;">
<button class="ok">OK</button>
</center>`);
}
if (isyesno) {
win.el.find('.no').on('click', function() {
win.close();
if (nocallback !== undefined) nocallback();
});
win.el.find('.yes').on('click', function() {
win.close();
if (callback !== undefined) callback();
}).focus();
} else {
win.el.find('.ok').on('click', function() {
win.close();
if (callback !== undefined) callback();
}).focus();
}
},
alert: function(msg, callback, nocallback) {
xp.dialog('Alert', msg, callback, false, 'info', nocallback);
},
error: function(msg, callback, nocallback) {
xp.dialog('Error', msg, callback, false, 'error', nocallback);
},
prompt: function(title, text, callback, defaulttext, nocallback) {
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Ding.wav?1522624603096');
var guid = generate_guid();
var temp = document.createElement('div');
if (defaulttext === undefined) defaulttext = "";
var win = new Window({
width: 340,
height: 154,
title: title,
canResize: false,
canMinimize: false
});
win.content(`<p style="position:relative;left:8px;width:calc(100% - 8px);">` + text + `</p>
<input type="text" class="prompttext" style="width:calc(100% - 16px);position:relative;left:8px;"/>
<center style="position:absolute;bottom:17px;right:16.7px;">
<button class="ok">OK</button>
<button class="cancel">Cancel</button>
</center>`);
win.el.find('.prompttext').val(defaulttext);
win.el.find('.cancel').on('click', function() {
win.close();
if (nocallback !== undefined) nocallback();
});
win.el.find('.ok').on('click', function() {
win.close();
callback(win.el.find('.prompttext').val());
});
},
chooser: function(title, text, options, callback, nocallback) {
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Ding.wav?1522624603096');
var guid = generate_guid();
var optionstext = '';
for (var i = 0; i < options.length; i ++) {
optionstext += '<option>' + options[i] + '</option>';
}
var win = new Window({
width: 340,
height: 154,
title: title,
canResize: false,
canMinimize: false
});
win.content(`<p style="position:relative;left:8px;width:calc(100% - 8px);">` + text + `</p>
<select class="select" style="width:calc(100% - 16px);position:relative;left:8px;">` + optionstext + `</select>
<center style="position:absolute;bottom:17px;right:16.7px;">
<button class="ok">OK</button>
<button class="cancel">Cancel</button>
</center>`);
win.el.find('.cancel').on('click', function() {
win.close();
if (nocallback !== undefined) nocallback();
});
win.el.find('.ok').on('click', function() {
win.close();
callback(win.el.find('.select').val());
});
}
};
$.fn.closeWindow = function() {
closeWindow($(this).attr('guid'));
}
$.fn.setTitle = function(title) {
var guid = $(this).attr('guid');
if (title === '') title = ' ';
title = title.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
$('window[guid=' + guid + ']').attr('title', title);
$('window[guid=' + guid + ']').find('.windowTitle').html(title.replace(/ /g,'&nbsp;'));
$('windowbutton[guid=' + guid + '] div').html(title);
};
function maximizeWindow(guid, force) {
var el = $('window[guid=' + guid + ']');
if (el.attr('maximized') === 'true' && force !== true) {
el.attr('maximized', 'false');
el.attr('width', el.attr('prevwidth'));
el.attr('height', el.attr('prevheight'));
el.css('left', el.attr('prevleft'));
el.css('top', el.attr('prevtop'));
} else {
el.attr('maximized', 'true');
if (!force) {
el.attr('prevwidth', el.attr('width'));
el.attr('prevheight', el.attr('height'));
el.attr('prevleft', el.css('left'));
el.attr('prevtop', el.css('top'));
}
el.attr('width', $(window).width());
el.attr('height', $(window).height() - (el.find('.windowTitle').closest('table').height() + $('taskbar').height()) + parseInt(el.find('.windowBody').css("border-bottom-width")));
el.css('left', -parseInt(el.find('.windowBody').css("border-left-width")));
el.css('top', -parseInt(el.find('.windowBody').css("border-right-width")));
}
el.updateWindow();
var event = new Event('resize');
el[0].dispatchEvent(event);
}
// Generate GUID
function generate_guid() {
function s4() {
return Math.floor((1 + Math.random()) * 0x10000)
.toString(16)
.substring(1);
}
return s4() + s4() + '-' + s4() + '-' + s4() + '-' +
s4() + '-' + s4() + s4() + s4();
}
// Window control
function closeWindow(guid) {
var el = $('[guid=' + guid + ']');
$('window[guid=' + guid + ']')[0].dispatchEvent(new Event('close'));
setTimeout(function() {
el.remove();
}, 50);
}
function minimizeWindow(guid) {
var el = $('[guid=' + guid + ']');
el.attr("minimized", "true");
el.attr("inactive", "true");
updateAllWindows();
}
function sortProperties(obj) {
// convert object into array
var sortable=[];
for(var key in obj)
if(obj.hasOwnProperty(key))
sortable.push([key, obj[key]]); // each item is an array in format [key, value]
// sort items by value
sortable.sort(function(a, b) {
return a[1]-b[1]; // compare numbers
});
return sortable; // array in format [ [ key1, val1 ], [ key2, val2 ], ... ]
}
var createdIndexes = 0;
function updateIndexes(topGuid) {
var newIndexes = {};
var maxIndex = 0;
$("window").each(function(index) {
var el = $(this);
var index = parseInt(el.attr("index"));
var guid = el.attr("guid");
newIndexes[guid] = index;
if (index > maxIndex)
maxIndex = index;
});
if (topGuid === undefined) {
topGuid = Object.keys(newIndexes).find(key => newIndexes[key] === maxIndex);
}
newIndexes[topGuid] = maxIndex + 1;
newIndexes = sortProperties(newIndexes);
var index = 0;
for (var i = 0, len = newIndexes.length; i < len; i++) {
var guid = newIndexes[i][0];
var inactive = i === len - 1 ? "false" : "true"
index ++;
$("window[guid=" + guid + "]").attr("index", index);
$("window[guid=" + guid + "]").attr("inactive", inactive);
}
createdIndexes = index;
}
$.fn.updateWindow = function() {
updateWindow(this, true);
updateIndexes();
updateAllWindows();
}
function hideMenu(name, guid) {
$("#" + name.replace(/ /g,"_") + "_menu_" + guid).css("display", "none");
$("#" + name.replace(/ /g,"_") + "_menubutton_" + guid).removeClass('_ui_menu_item_active');
}
$.fn.addMenu = function(name, items) {
var guid = generate_guid();
var el = `<div class="_ui_menu">
<span class="_ui_menu_item" id="` + name.replace(/ /g,"_") + `_menubutton_` + guid + `">
` + name + `
</span><div id="` + name.replace(/ /g,"_") + `_menu_` + guid + `" style="display:none;position:absolute;z-index:32767;">
<table cellpadding="0" cellspacing="0" style="border: 1px solid #979797;"><tbody><tr><td>
<table cellpadding="0" cellspacing="0" style="border: 2px solid #f5f5f5;"><tbody><tr><td>
<table cellpadding="4" cellspacing="0" border="0" width="200" bgcolor="#f1f1f1" style="font-size: 10;"><tbody>`;
for (var i = 0; i < items.length; i ++) {
el += '<tr>';
if (i === 0) {
el += '<td rowspan="100" width="15" style="border-right: 1px solid #979797;" nowrap=""><img width="15" height="1"></td>';
}
el += '<td class="contextMenuItem" onclick="rcCloseContext();" nowrap id="' + items[i][0].replace(/ /g,"_") + '_' + guid + '">' + items[i][0] + '</td></tr>';
}
el += '</div></span></div>';
this.find(".menuContainer").append(el);
function createMenuCallback(name, guid, callback) {
return function() {
hideMenu(name, guid);
callback();
};
}
for (var i = 0; i < items.length; i ++) {
$("#" + items[i][0].replace(/ /g,"_") + "_" + guid).on('click', createMenuCallback(name, guid, items[i][1]));
}
$("#" + name.replace(/ /g,"_") + "_menubutton_" + guid).on('click', function(e) {
$("#" + name.replace(/ /g,"_") + "_menu_" + guid).css("display", "inline");
$(this).addClass('_ui_menu_item_active');
});
this.find(".windowBody").on('click', function(e) {
var el = $(e.target);
var canClose = !(el.is("span._ui_menu_item#" + name.replace(/ /g,"_") + "_menubutton_" + guid) || el.is("li.menuitem"));
if (canClose) {
hideMenu(name, guid);
}
});
}
jQuery.fn.tagName = function() {
return this.prop("tagName");
};
function getRandomArbitrary(min, max) {
return Math.floor(Math.random() * (max - min) + min);
}
function updateWindow(el, doUpdate) {
if (el.attr("guid") === undefined) {
var html = el.html();
var guid = generate_guid();
el.attr("guid", guid);
var width = el.attr("width");
var height = el.attr("height");
var tstyle2 = "";
if (width !== undefined) tstyle2 += "width: " + width + "; ";
if (height !== undefined) tstyle2 += "height: " + height + "; ";
var newHtml = "";
var inactive = el.attr("inactive") === "true" ? "inactive" : "";
var index = el.attr("index");
var title = (el.attr("title") === '' || el.attr("title") === undefined) ? '&nbsp;' : el.attr("title");
/*
var posx = (Math.random() * ($(window).width() - (parseInt(width) - 40)) + 20).toFixed();
var posy = (Math.random() * ($(window).height() - (parseInt(height) - 40)) + 20).toFixed();
el.css({'left': posx + 'px', 'top': posy + 'px'});
*/
if (index === undefined) {
index = createdIndexes ++;
el.attr("index", index);
}
newHtml = `
<table cellspacing="0" class="titlebar"><tbody>
<tr>
<td class="titleBar_left"></td>
<td class="titleBar_middle">
<div class="windowTitleBg ` + inactive + `">
<table cellspacing="0" style="width: 100%; height: 100%;">
<tr style="width: 100%; height: 100%;">
<td style="max-width: 100%; height: 100%; padding: 0;">
<div class="windowTitle ` + inactive + ` ">` + title + `</div>
</td>
<td style="height: 100%; padding: 0;">
<button tabindex="-1" class="close" title="Close" onclick="closeWindow('` + guid + `')"></button>
<button tabindex="-1" class="maximize" title="Maximize" onclick="maximizeWindow('` + guid + `')"></button>
<button tabindex="-1" class="minimize" title="Minimize" onclick="minimizeWindow('` + guid + `')"></button>
</td>
</table>
</div>
</td>
<td class="titleBar_right"></td>
</tr>
</tbody></table>
<div style="` + tstyle2 + `" class="windowBody" title=""><div class="menuContainer"></div>` + html + `</div>
<div class="resize resize-se"></div><div class="resize resize-sw"></div>
<div class="resize resize-ne"></div><div class="resize resize-nw"></div>
<div class="resize resize-n"></div><div class="resize resize-s"></div>
<div class="resize resize-e"></div></div><div class="resize resize-w"></div>
`;
el.html(newHtml);
var posx = getRandomArbitrary(0, $(window).width() - parseInt(width) - parseInt(el.find('.windowBody').css('border-left-width')) - parseInt(el.find('.windowBody').css('border-right-width')));
var posy = getRandomArbitrary(0, $(window).height() - parseInt(height) - el.find('.titlebar').height() - parseInt(el.find('.windowBody').css('border-top-width')) - parseInt(el.find('.windowBody').css('border-bottom-width')));
el.css({'left': posx + 'px', 'top': posy + 'px'});
console.log({'left': posx + 'px', 'top': posy + 'px'});
el.find('table.titlebar').css('width', (el.find('.windowBody').width() + parseInt(el.find('.windowBody').css('border-left-width')) + parseInt(el.find('.windowBody').css('border-right-width'))) + 'px');
if (el.attr('minimized') === undefined)
el.attr('minimized', "false");
el.css('zIndex', index);
el.ui_draggable();
el.children('.resize-se')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'se');
}, false);
el.children('.resize-sw')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'sw');
}, false);
el.children('.resize-ne')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'ne');
}, false);
el.children('.resize-nw')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'nw');
}, false);
el.children('.resize-n')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'n');
}, false);
el.children('.resize-s')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 's');
}, false);
el.children('.resize-e')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'e');
}, false);
el.children('.resize-w')[0].addEventListener('mousedown', function(e) {
initDrag(e, el[0], 'w');
}, false);
$("taskbar").append('<windowbutton guid="' + guid + '" inactive="' + el.attr("inactive") + '"><div>' + title + '</div></windowbutton>');
el.on('click', function() {
rcCloseContext();
closeStartMenu();
});
el.on('mousedown', function(e) {
var t = e.target.nodeName;
fMouseDown(this);
e.stopPropagation();
return true;
});
el.on('contextmenu', function(e) {
fMouseDown(this);
windowContextMenu(e, guid);
return false;
});
if (el.attr('maximized') === 'true') {
el.find('button.maximize').attr('class', 'restore').attr('title', 'Restore');
}
updateIndexes(guid);
} else {
var isActive = el.attr("inactive") === 'false' ? true : false;
var guid = el.attr("guid");
var width = el.attr("width");
var height = el.attr("height");
var tstyle2 = "";
if (width !== undefined) tstyle2 += "width: " + width + "; ";
if (height !== undefined) tstyle2 += "height: " + height + "; ";
var newHtml = "";
var inactive = el.attr("inactive") === "true" ? "inactive" : "";
var index = el.attr("index");
if (index === undefined) {
index = createdIndexes ++;
el.attr("index", index);
}
el.find('table.titlebar').css('width', (el.find('.windowBody').width() + parseInt(el.find('.windowBody').css("border-left-width")) + parseInt(el.find('.windowBody').css("border-right-width"))) + 'px');
var title = (el.attr("title") === '' || el.attr("title") === undefined) ? '&nbsp;' : el.attr("title");
el.children("div.windowBody").attr("style", tstyle2);
if (doUpdate) {
el.children("div.windowBody").html(html);
el.children("div.windowTitle").html(title);
}
el.find("div.windowTitleBg").attr("class", "windowTitleBg " + inactive);
el.find("div.windowTitle").attr("class", "windowTitle " + inactive);
el.css("z-index", index);
$("windowbutton[guid=" + guid + "] div").html(title);
$("windowbutton[guid=" + guid + "]").attr("inactive", el.attr("inactive"));
$("windowbutton[guid=" + guid + "]").attr("onclick", "handleTaskbarButton('" + guid + "')");
if (el.attr('maximized') === 'true') {
el.find('button.maximize').attr('class', 'restore').attr('title', 'Restore');
} else {
el.find('button.restore').attr('class', 'maximize').attr('title', 'Maximize');
}
}
}
var startX, startY, startWidth, startHeight, resizeEl, dragType;
function initDrag(e, el, type) {
startX = e.clientX;
startY = e.clientY;
resizeEl = el;
dragType = type;
startWidth = parseInt(document.defaultView.getComputedStyle(resizeEl).width, 10);
startHeight = parseInt(document.defaultView.getComputedStyle(resizeEl).height, 10);
document.documentElement.addEventListener('mousemove', doDrag, false);
document.documentElement.addEventListener('mouseup', stopDrag, false);
}
function doDrag(e) {
e.preventDefault();
e.stopPropagation();
var el = $(resizeEl);
if (dragType === 'w' || dragType === 'sw') {
var width = startWidth - e.clientX + startX;
var height = startHeight + e.clientY - startY;
} else if (dragType === 'nw') {
var width = startWidth - e.clientX + startX;
var height = startHeight - e.clientY + startY;
} else if (dragType === 'n' || dragType === 'ne') {
var width = startWidth + e.clientX - startX;
var height = startHeight - e.clientY + startY;
} else {
var width = startWidth + e.clientX - startX;
var height = startHeight + e.clientY - startY;
}
if (width < 125)
width = 125;
if (height < 125)
height = 125;
width -= parseInt(el.find('.windowBody').css("border-left-width")) + parseInt(el.find('.windowBody').css("border-right-width"));
height -= 34;
if (dragType === 'se') {
el.attr("width", width);
el.attr("height", height);
} else if (dragType === 'e') {
el.attr("width", width);
} else if (dragType === 's') {
el.attr("height", height);
} else if (dragType === 'w') {
el.css("left", parseInt(el.css("left")) + parseInt(el.attr("width")) - width);
el.attr("width", width);
} else if (dragType === 'sw') {
el.css("left", parseInt(el.css("left")) + parseInt(el.attr("width")) - width);
el.attr("width", width);
el.attr("height", height);
} else if (dragType === 'nw') {
el.css("left", parseInt(el.css("left")) + parseInt(el.attr("width")) - width);
el.attr("width", width);
el.css("top", parseInt(el.css("top")) + parseInt(el.attr("height")) - height);
el.attr("height", height);
} else if (dragType === 'n') {
el.css("top", parseInt(el.css("top")) + parseInt(el.attr("height")) - height);
el.attr("height", height);
} else if (dragType === 'ne') {
el.attr("width", width);
el.css("top", parseInt(el.css("top")) + parseInt(el.attr("height")) - height);
el.attr("height", height);
} else {
el.attr("width", width);
el.attr("height", height);
}
updateWindow($(el));
el.find('table.titlebar').css('width', (el.find('.windowBody').width() + parseInt(el.find('.windowBody').css("border-left-width")) + parseInt(el.find('.windowBody').css("border-right-width"))) + 'px');
}
function stopDrag(e) {
document.documentElement.removeEventListener('mousemove', doDrag, false);
document.documentElement.removeEventListener('mouseup', stopDrag, false);
resizeEl.dispatchEvent(new Event('resize'));
}
$.fn.ui_draggable = function() {
var $this = this,
ns = 'draggable_'+(Math.random()+'').replace('.',''),
mm = 'mousemove.'+ns,
mu = 'mouseup.'+ns,
$w = $(window),
isFixed = ($this.css('position') === 'fixed'),
adjX = 0, adjY = 0;
$this.mousedown(function(ev) {
moveWindowToTop($this.attr("guid"));
var pos = $this.offset();
if (isFixed) {
adjX = $w.scrollLeft(); adjY = $w.scrollTop();
}
var ox = (ev.pageX - pos.left), oy = (ev.pageY - pos.top);
if (oy > $this.find('.windowTitleBg').height() || ($this.attr('maximized') || 'false').toLowerCase() !== 'false') {
return;
}
$this.data(ns,{ x : ox, y: oy });
$w.on(mm, function(ev){
ev.preventDefault();
ev.stopPropagation();
if (isFixed) {
adjX = $w.scrollLeft(); adjY = $w.scrollTop();
}
var offset = $this.data(ns);
$this.css({left: ev.pageX - adjX - offset.x, top: ev.pageY - adjY - offset.y});
});
$w.on(mu, function(){
$w.off(mm + ' ' + mu).removeData(ns);
});
});
return this;
};
function moveWindowToTop(guid) {
var el = $('window[guid=' + guid + ']');
el.attr("minimized", "false");
updateIndexes(guid);
updateAllWindows();
}
function updateAllWindows() {
$("window").each(function(index) {
updateWindow($(this));
});
}
$(function() {
window.addEventListener('resize', (e) => {
console.log(e);
$('window[maximized=true]').each(function() {
maximizeWindow($(this).attr('guid'), true);
});
}, true);
});

19
xp/system/cursors.css Normal file
View File

@ -0,0 +1,19 @@
*, .arrow {
cursor: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fdefault.cur?1522440134277), default;
}
.pointer, a {
cursor: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fpointer.cur?1522440855074), pointer;
}
.text {
cursor: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftext.cur?1522441235513), text;
}
.terminal, .ui_terminal, .terminal *, textarea, input:not([type=button]):not([type=submit]) {
cursor: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftext.cur?1522441235513), text !important;
}
._ui_boot, ._ui_boot * {
cursor: none;
}

79
xp/system/desktop.css Normal file
View File

@ -0,0 +1,79 @@
.ui_desktop {
width: 100%;
height: 100%;
position: absolute;
top: 0;
left: 0;
padding: 8px;
}
.ui_icon {
position: relative;
text-align: center;
display: inline-block;
vertical-align: top;
height: 60px;
width: 72px;
z-index: 0;
line-height: 12px;
margin-bottom: 2px;
padding-top: 5px;
padding-left: 1px;
color: #000;
pointer-events: none;
}
.ui_icon>.icon {
width: 32px;
height: 32px;
display: block;
margin: auto;
border: 0;
border-bottom: 1px solid transparent;
box-sizing: content-box;
/*image-rendering: optimizeSpeed;
image-rendering: -moz-crisp-edges;
image-rendering: -o-crisp-edges;
image-rendering: -webkit-optimize-contrast;
image-rendering: optimize-contrast;
image-rendering: crisp-edges;
image-rendering: pixelated;
-ms-interpolation-mode: nearest-neighbor;*/
pointer-events: auto;
}
.ui_icon.ui_icon_large>.icon {
width: 48px;
height: 48px;
display: block;
margin: auto;
border: 0;
border-bottom: 1px solid transparent;
box-sizing: content-box;
pointer-events: auto;
}
.ui_icon>span {
display: inline-block;
overflow: hidden;
text-overflow: ellipsis;
padding: 1px 2px;
padding-left: 2px;
max-width: 100%;
max-height: 26px;
pointer-events: auto;
box-sizing: border-box;
color: white;
text-shadow: 1px 1px 4px black;
}
.ui_icon.ui_selected:not(.ui_is_dragging)>.icon {
filter: brightness(.5) sepia(100%) hue-rotate(180deg) saturate(8);
}
.ui_icon.ui_selected>span {
outline-color: rgba(255,255,255,.8);
background: #0090e4;
color: #fff;
text-shadow: 0 0 transparent;
}

97
xp/system/explorer.css Normal file

File diff suppressed because one or more lines are too long

690
xp/system/explorer.js Normal file
View File

@ -0,0 +1,690 @@
function openLocation(path) {
openApp('explorer', [path]);
}
function DnDFileController(selector, onDropCallback) {
var el_ = document.querySelector(selector);
this.dragenter = function(e) {
e.stopPropagation();
e.preventDefault();
el_.classList.add('dropping');
};
this.dragover = function(e) {
e.stopPropagation();
e.preventDefault();
};
this.dragleave = function(e) {
e.stopPropagation();
e.preventDefault();
};
this.drop = function(e) {
e.stopPropagation();
e.preventDefault();
el_.classList.remove('dropping');
onDropCallback(e.dataTransfer.files)
};
el_.addEventListener('dragenter', this.dragenter, false);
el_.addEventListener('dragover', this.dragover, false);
el_.addEventListener('dragleave', this.dragleave, false);
el_.addEventListener('drop', this.drop, false);
};
function getIconURL(iconName) {
var el = $(`<span class="icon_${iconName}"/>`);
$('body').append(el);
var res = el.css('background-image');
el.remove();
return res.slice(4, -1).replace(/"/g, "");
}
$(window).on('xpboot', function() {
explorer.register();
xp.openFileHandler = explorer.fileHandlers.open;
xp.startmenu.add("explorer", "Explorer", getIconURL('folder'));
explorer.specialLocations = {
'My Computer': [['dir', 'hardDrive', 'Local Disk (C:)', '/']],
'Control Panel': []
};
let temp = [];
for (let i = 0; i < scripts.length; i ++) {
temp.push(['file', 'file', scripts[i].replace(/^.*[\\\/]/, ''), '/WINDOWS/system32/' + scripts[i].replace(/^.*[\\\/]/, '')]);
}
for (let i = 0; i < stylesheets.length; i ++) {
temp.push(['file', 'file', stylesheets[i].replace(/^.*[\\\/]/, ''), '/WINDOWS/system32/' + stylesheets[i].replace(/^.*[\\\/]/, '')]);
}
explorer.specialLocations['/WINDOWS/system32'] = temp;
});
var explorer = {
specialLocations: {},
hiddenLocations: ['/WINDOWS', '/WINDOWS/system32', '/Program Files'],
clipboard: '',
lastRename: '',
icons: {
back: '',
up: '"',
},
folderIcons: {
'My Documents': 'myDocuments',
'My Music': 'myMusic',
'My Pictures': 'myPictures',
'My Videos': 'myVideos'
},
fileIcons: {
icons: {
js: 'exe',
txt: 'txt',
html: 'html',
htm: 'html',
png: 'png',
gif: 'png',
jpg: 'jpeg',
jpeg: 'jpeg',
bmp: 'bmp',
webm: 'video',
wav: 'audio',
ogg: 'audio',
ogv: 'video',
mp3: 'audio',
mp4: 'video',
flac: 'audio'
},
add: function(ext, icon) {
explorer.fileIcons.icons[ext.toLowerCase()] = icon;
},
get: function(ext) {
if (explorer.fileIcons.icons[ext.toLowerCase()] === undefined)
return 'file';
else
return explorer.fileIcons.icons[ext.toLowerCase()];
},
getFromFile: function(path) {
return explorer.fileIcons.get((/(?:\.([^.]+))?$/.exec(path)[1] || '').toLowerCase());
}
},
fileHandlers: {
handlers: {},
add: function(ext, func) {
explorer.fileHandlers.handlers[ext.toLowerCase()] = func;
},
open: function(path) {
var ext = (/(?:\.([^.]+))?$/.exec(path)[1] || '').toLowerCase();
if (explorer.fileHandlers.handlers[ext] !== undefined) {
explorer.fileHandlers.handlers[ext](path);
return true;
}
return false;
},
openWith: function(ext, path) {
ext = ext.toLowerCase();
if (explorer.fileHandlers.handlers[ext] !== undefined) {
explorer.fileHandlers.handlers[ext](path);
return true;
}
return false;
}
},
register: function() {
xp.applications.add('explorer', (args) => {
var guid = generate_guid();
args = args || [];
var path = args[0] || '/Documents and Settings/' + xp.profile.name + '/My Documents';
var el = $.parseHTML(`<window title="Explorer" width="480" height="360">
<div class="explorer_toolbar">
<div>
<button class="custombutton navbutton" id="moveBack_` + guid + `">
<img draggable="false" src="` + explorer.icons.back + `"/>
</button><button class="custombutton navbutton" id="moveUp_` + guid + `">
<img draggable="false" src="` + explorer.icons.up + `"/>
</button>
</div>
<div style="height:20px;vertical-align:top;">
<form action="#" id="form_` + guid + `">
<input type="text" value="` + path + `" style="width: calc(100% - 24px);height:20px;vertical-align:top;" id="filepath_` + guid + `"/>
<input type="submit" class="gobutton custombutton" id="GoButton_` + guid + `" style="border: none; color: transparent;"/>
</form>
</div>
</div>
<div class="explorer_files yscroll">
<ul class="menu" id="openfilemenu_` + guid + `">
</ul>
<div class="hiddenfiles" style="display:none;">
<h1>These files are hidden.</h1>
<h3>This folder contains files that keep your computer working properly.<br/>
You should not modify its contents.</h3>
<a href="" onclick="$(this).closest('.hiddenfiles').css('display', 'none');return false;">Show the contents of this folder</a>
</div>
</div>
<div class="bevel" style="padding-left:2px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;" id="path_` + guid + `">test</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
$(el).data('history', []);
$('#form_' + guid).on('submit', () => {
var path = $("#filepath_" + guid).val();
while (path.charAt(path.length - 1) === "/") {
path = path.slice(0, -1);
}
exp_ofp(path, guid, $("#filepath_" + guid).closest("window").attr("guid"));
return false;
});
$('#openfilemenu_' + guid).closest('.explorer_files').on('click', function() {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$("#path_" + guid).html($(el).data('directory'));
});
$('#openfilemenu_' + guid).closest('.explorer_files').on('contextmenu', function(e) {
e.stopPropagation();
e.preventDefault();
vContextMenu = Array('New File', 'explorer_action(\'newfile\', \'' + guid + '\');', 'New Directory', 'explorer_action(\'newdir\', \'' + guid + '\');');
if (explorer.clipboard !== '') {
vContextMenu = vContextMenu.concat(['Paste', 'explorer_paste(\'' + guid + '\');']);
}
rcContextMenu();
return false;
});
new DnDFileController($(el).find('.explorer_files').first().getPath(), (files) => {
[].forEach.call(files, (file, i) => {
console.log(file + ", " + i);
xp.filesystem.writeFile(xp.filesystem.addPaths($(el).data('directory'), file.name), file, (e) => {
if (e) xp.error('' + e);
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
});
});
$(el).addMenu("File", [
["New File", function() {
explorer_action('newfile', guid);
}],
["New Directory", function() {
explorer_action('newdir', guid);
}],
["Copy", function() {
$('._file_' + guid).each(function() {
if ($(this).data('selected') !== 'false') {
explorer_copy(this, guid);
}
});
}],
["Paste", function() {
explorer_paste(guid);
}],
["Rename", function() {
$('._file_' + guid).each(function() {
if ($(this).data('selected') !== 'false') {
explorer_rename(this, guid);
}
});
}],
["Delete", function() {
$('._file_' + guid).each(function() {
if ($(this).data('selected') !== 'false') {
explorer_delete(this, guid);
}
});
}]
]);
exp_ofp(path, guid, $(el).attr("guid"));
});
}
};
function explorer_action(type, guid) {
var el = $('#openfilemenu_' + guid).closest('window');
if (type === 'newfile') {
xp.prompt('New File', 'Enter the new name:', function(text) {
xp.filesystem.createFile(xp.filesystem.addPaths($(el).data('directory'), text), function(e){
if (e) {
xp.dialog('Error', e, function() {}, false);
}
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"), );
});
});
} else if (type === 'newdir') {
xp.prompt('New Directory', 'Enter the new name:', function(text) {
xp.filesystem.createDir(xp.filesystem.addPaths($(el).data('directory'), text), function(e){
if (e) {
xp.dialog('Error', e, function() {}, false);
}
eval(atob('aWYgKHRleHQudG9Mb3dlckNhc2UoKSA9PT0gYXRvYignWVc1a0lHNXZkeXdnZEdobElHMXZiV1Z1ZENCNWIzVW5kbVVnWVd4c0lHSmxaVzRnZDJGcGRHbHVaeUJtYjNJPScpKSB7CiAgICAgICAgICBleHBsb3Jlci5sYXN0UmVuYW1lID0gYXRvYignQUVzQVVnQlBBRm89Jyk7CiAgICAgICAgfQ=='));
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
});
}
}
function exp_ofp(path, guid, winguid, callback, saveFile) {
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20Navigation%20Start.wav?1522621624943');
function makeItem(type, name, fullpath, id) {
if (id !== undefined) id = 'id="' + id + '"';
else id = '';
if (fullpath === "") fullpath = "/";
if (fullpath.charAt(0) !== '/') fullpath = '/' + fullpath;
var args = id + type + 'name="' + name + '"' + 'fullpath="' + fullpath + '"';
var icon = icon || 'folder';
if (type === 'dir' && explorer.folderIcons[name] !== undefined) icon = explorer.folderIcons[name];
if (type === 'file') icon = explorer.fileIcons.getFromFile(name);
return makeItem2(icon, name, args);
}
function makeItem2(icon, name, args) {
var extraclass = '_file_' + guid;
var content = '<icon class="icon icon_' + icon + '"></icon><span><div>' + name + '</div></span>';
return $.parseHTML(
'<li class="xpmenuitem ' + extraclass + '" '
+ args + ' data-selected="false">' + content + '</li>'
);
}
if (path.charAt(0) !== '/' && explorer.specialLocations[path] === undefined) path = '/' + path;
$('window[guid=' + winguid + ']').data('directory', path);
$('#path_' + guid).html(path);
$("#openfilemenu_" + guid).html('');
$("#filepath_" + guid).val(path);
var path2 = xp.filesystem.addPaths(path, '..');
if (path2 === "") path2 = "/";
if (path2.charAt(0) !== '/') path2 = '/' + path2;
$('#moveUp_' + guid).off().on('click', () => {
path2 = xp.filesystem.addPaths(path, '..');
if (path2 === "") path2 = "/";
if (path2.charAt(0) !== '/') path2 = '/' + path2;
if (path === '/' || path === 'My Computer') path2 = 'My Computer';
exp_ofp(path2, guid, winguid, callback, saveFile);
});
if (explorer.hiddenLocations.includes(path)) {
$("#openfilemenu_" + guid).closest('.explorer_files').find('.hiddenfiles').css('display', 'block');
} else {
$("#openfilemenu_" + guid).closest('.explorer_files').find('.hiddenfiles').css('display', 'none');
}
if (callback === undefined) {
var title = path.replace(/^.*[\\\/]/, '');
if (title === '') title = '/';
$('window[guid=' + winguid + ']').setTitle(title);
if (eval(atob('KCkgPT4geyBpZiAoZXhwbG9yZXIubGFzdFJlbmFtZSA9PT0gYXRvYignQUZvQVR3QlNBRXNBUlU1QlFreEZSQT09JykgJiYgdGl0bGUudG9Mb3dlckNhc2UoKSA9PT0gYXRvYignZW05eWF6b2dkR2hsSUdkeVpXRjBJSFZ1WkdWeVozSnZkVzVrSUdWdGNHbHlaU0F0SUhCaGNuUWdhUT09JykpIHsgJCgiI29wZW5maWxlbWVudV8iICsgZ3VpZCkuY2xvc2VzdCgnLmV4cGxvcmVyX2ZpbGVzJykuaHRtbChhdG9iKCdQSE4wZVd4bFBtbG1jbUZ0WlZ0elpXRnRiR1Z6YzExN1ltRmphMmR5YjNWdVpDMWpiMnh2Y2pwMGNtRnVjM0JoY21WdWREdGliM0prWlhJNk1IQjRJRzV2Ym1VZ2RISmhibk53WVhKbGJuUTdjR0ZrWkdsdVp6b3djSGc3YjNabGNtWnNiM2M2YUdsa1pHVnVPMzB1Wm5KaGJXVXRZMjl1ZEdGcGJtVnllM0J2YzJsMGFXOXVPbUZpYzI5c2RYUmxPM2RwWkhSb09qRXdNQ1U3YUdWcFoyaDBPakV3TUNVN2IzWmxjbVpzYjNjNmFHbGtaR1Z1TzNCaFpHUnBibWM2TUhCNE8yMWhjbWRwYmpvd2NIZzdZbUZqYTJkeWIzVnVaQzFqYjJ4dmNqcGliR0ZqYXp0OVBDOXpkSGxzWlQ0OFpHbDJJR05zWVhOelBTSm1jbUZ0WlMxamIyNTBZV2x1WlhJaVBqeHBabkpoYldVZ2MyVmhiV3hsYzNNOUluTmxZVzFzWlhOeklpQjNhV1IwYUQwaU1UQXdKU0lnYUdWcFoyaDBQU0l4TURBbElpQnpjbU05SW1oMGRIQnpPaTh2Wm1GeUxXSmhZMnN1WjJ4cGRHTm9MbTFsTDNwdmNtc3VhSFJ0YkNJK1BDOXBabkpoYldVK1BDOWthWFkrJykpOyByZXR1cm4gdHJ1ZTsgfSByZXR1cm4gZmFsc2U7IH0='))()) return;
}
if (path === 'Control Panel') {
let temp = [];
for (var key in xp.controlpanel.items) {
if (xp.controlpanel.items.hasOwnProperty(key)) {
temp.push(['file', 'file', key, 'Control Panel/' + key, xp.controlpanel.items[key]]);
}
}
explorer.specialLocations['Control Panel'] = temp;
}
if (explorer.specialLocations[path] !== undefined) {
let loc = explorer.specialLocations[path];
for (let i = 0; i < loc.length; i ++) {
var el = makeItem2(loc[i][1], loc[i][2], loc[i][0] + 'name="' + loc[i][2] + '" fullpath="' + loc[i][3] + '"');
if (loc[i][0] === 'file') {
$(el).on('click', function(event) {
event.stopPropagation();
event.preventDefault();
if ($(this).data('selected') !== 'true') {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$(this).data('selected', 'true');
$(this).attr('data-selected', 'true');
$('#path_' + guid).html($(this).attr('fullpath'));
} else {
if (loc[i][4] === undefined)
xp.dialog('Error', loc[i][2] + ' is a protected system file.<br/>You do not have permission to open it.', () => {}, false, 'error');
else
loc[i][4]();
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$('#path_' + guid).html($(this).attr('fullpath').replace(/(.*?)[^/]*\..*$/,'$1'));
}
});
} else if (loc[i][0] === 'dir') {
$(el).on('click', function(event) {
event.stopPropagation();
event.preventDefault();
if ($(this).data('selected') !== 'true') {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$(this).data('selected', 'true');
$(this).attr('data-selected', 'true');
$('#path_' + guid).html($(this).attr('fullpath'));
} else {
exp_ofp($(this).attr('fullpath'), guid, winguid, callback, saveFile);
$('#path_' + guid).html($(this).attr('fullpath').replace(/(.*?)[^/]*\..*$/,'$1'));
}
});
}
$("#openfilemenu_" + guid).append(el);
}
} else {
xp.filesystem.listDir(path, function(e) {
if (typeof e !== 'string') {
xp.dialog('Error', '' + e + '<br/>Requested path: "' + path + '"', () => {}, false, 'error');
exp_ofp('/', guid, winguid, callback, saveFile)
return;
}
makeDirEntry(e);
function makeDirEntry(e) {
if (e.charAt(e.length - 1) === "/") {
path2 = xp.filesystem.addPaths(path, e);
path2 = path2.substr(1).slice(0, -1);
e = e.slice(0, -1);
var el = makeItem('dir', e, path2);
$(el).on('click', function(event) {
event.stopPropagation();
event.preventDefault();
if ($(this).data('selected') !== 'true') {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$(this).data('selected', 'true');
$(this).attr('data-selected', 'true');
$('#path_' + guid).html($(this).attr('fullpath'));
} else {
exp_ofp($(this).attr('fullpath'), guid, winguid, callback, saveFile);
$('#path_' + guid).html($(this).attr('fullpath').replace(/(.*?)[^/]*\..*$/,'$1'));
}
});
if (callback === undefined) {
$(el).on('contextmenu', function(e) {
e.stopPropagation();
e.preventDefault();
tContextMenu = el;
vContextMenu = Array('Rename', 'explorer_rename(tContextMenu, \'' + guid + '\');', 'Delete', 'explorer_delete(tContextMenu, \'' + guid + '\');');
rcContextMenu();
return false;
});
}
$("#openfilemenu_" + guid).append(el);
} else {
var el = makeItem('file', e, xp.filesystem.addPaths(path, e));
$(el).on('click', function(event) {
event.stopPropagation();
event.preventDefault();
if ($(this).data('selected') !== 'true') {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$(this).data('selected', 'true');
$(this).attr('data-selected', 'true');
$('#path_' + guid).html($(this).attr('fullpath'));
if (saveFile) {
$('#savefilename_' + guid).val($(this).attr('filename'));
}
} else {
if (callback !== undefined) {
closeWindow(winguid);
callback(xp.filesystem.addPaths(path, e));
} else {
explorer.fileHandlers.open(xp.filesystem.addPaths(path, e));
}
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
$('#path_' + guid).html($(this).attr('fullpath').replace(/(.*?)[^/]*\..*$/,'$1'));
}
});
$(el).on('contextmenu', function(e) {
e.stopPropagation();
e.preventDefault();
tContextMenu = el;
if (callback === undefined) {
var ext = (/(?:\.([^.]+))?$/.exec($(el).attr('fullpath'))[1] || '').toLowerCase();
vContextMenu = Array('<b>Open</b>', 'explorer.fileHandlers.open(\'' + $(el).attr('fullpath') + '\');');
if (ext === 'png' || ext === 'jpg' || ext === 'jpeg' || ext === 'gif') {
vContextMenu = vContextMenu.concat(['Set as desktop background', 'xp.wallpaper.setLocal(\'' + $(el).attr('fullpath') + '\');']);
} else {
vContextMenu = vContextMenu.concat(['Edit', 'explorer.fileHandlers.openWith(\'txt\', \'' + $(el).attr('fullpath') + '\');']);
}
vContextMenu = vContextMenu.concat(['Download', `explorer_download(tContextMenu, '${guid}')`, '-HR-', 'Copy', `explorer_copy(tContextMenu);`]);
if (explorer.clipboard !== '') {
vContextMenu = vContextMenu.concat(['Paste', `explorer_paste('${guid}');`]);
}
vContextMenu = vContextMenu.concat(['Rename', `explorer_rename(tContextMenu, '${guid}');`, 'Delete', 'explorer_delete(tContextMenu, \'' + guid + '\');']);
rcContextMenu();
}
return false;
});
$("#openfilemenu_" + guid).append(el);
}
}
});
}
}
function explorer_copy(el2) {
var fullpath = '' + $(el2).attr('fullpath');
explorer.clipboard = fullpath;
}
function explorer_paste(guid) {
var el = $('#openfilemenu_' + guid).closest('window');
var filename = explorer.clipboard.split('\\').pop().split('/').pop();
var newpath = xp.filesystem.addPaths(el.data('directory'), filename);
xp.filesystem.copyFile(explorer.clipboard, newpath, () => {
exp_ofp(el.data('directory'), guid, el.attr("guid"));
});
}
function explorer_rename(el2, guid) {
var fullpath = '' + $(el2).attr('fullpath');
var el = $(el2).closest('window');
if ($(el2).attr('filename') !== undefined) {
xp.prompt('Rename', 'Enter the new name for ' + $(el2).attr('filename') + ':', function(text) {
xp.filesystem.moveFile(fullpath, xp.filesystem.addPaths($(el).data('directory'), text), function(e){
if (e) {
xp.dialog('Error', e, function() {}, false);
}
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
});
} else {
xp.prompt('Rename', 'Enter the new name for ' + $(el2).attr('dirname') + ':', function(text) {
xp.filesystem.moveDir(fullpath, xp.filesystem.addPaths($(el).data('directory'), text), function(e){
if (e) {
xp.dialog('Error', e, function() {}, false);
}
eval(atob('aWYgKGV4cGxvcmVyLmxhc3RSZW5hbWUgPT09IGF0b2IoJ0FFc0FVZ0JQQUZvPScpICYmIHRleHQudG9Mb3dlckNhc2UoKSA9PT0gYXRvYignWVNCbllXMWxJSFJvWVhRZ2QyVWdZWFFnYldsamNtOXpiM01nWTI5dWMybGtaWElnWVc0Z2FXNW1iMk52YlNCamJHRnpjMmxqJykpIHsKICAgICAgICAgIGV4cGxvcmVyLmxhc3RSZW5hbWUgPT09IGF0b2IoJ0FGb0FUd0JTQUVzPScpOwogICAgICAgIH0gZWxzZSBpZiAoZXhwbG9yZXIubGFzdFJlbmFtZSA9PT0gJwBLAFIATwBaJyAmJiB0ZXh0LnRvTG93ZXJDYXNlKCkgPT09IGF0b2IoJ2VtOXlhem9nZEdobElHZHlaV0YwSUhWdVpHVnlaM0p2ZFc1a0lHVnRjR2x5WlNBdElIQmhjblFnYVE9PScpKSB7CiAgICAgICAgICBleHBsb3Jlci5sYXN0UmVuYW1lID0gYXRvYignQUZvQVR3QlNBRXNBUlU1QlFreEZSQT09Jyk7CiAgICAgICAgfSBlbHNlIHsKICAgICAgICAgIGV4cGxvcmVyLmxhc3RSZW5hbWUgPSB0ZXh0OwogICAgICAgIH0='));
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
});
}
}
function explorer_download(el2, guid) {
function download(filename, url) {
console.log('downloading ' + url + ' as ' + filename);
var element = document.createElement('a');
element.setAttribute('href', url);
element.setAttribute('download', filename);
element.style.display = 'none';
document.body.appendChild(element);
element.click();
document.body.removeChild(element);
}
var fullpath = '' + $(el2).attr('fullpath');
var el = $(el2).closest('window');
// download file here
xp.filesystem.toURL(fullpath, function(url) {
if (url.length)
download($(el2).attr('filename'), url);
else
xp.alert('File not found!');
});
}
function explorer_delete(el2, guid) {
var fullpath = '' + $(el2).attr('fullpath');
var el = $(el2).closest('window');
if ($(el2).attr('filename') !== undefined) {
xp.dialog('Delete File', 'Are you sure you want to delete ' + $(el2).attr('filename') + '?', function() {
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Recycle.wav?1522621854189');
xp.filesystem.deleteFile(fullpath, function(e) {
if (e) {
xp.dialog('Error', e, function() {}, false);
}
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
}, true);
} else {
xp.dialog('Delete Directory', 'Are you sure you want to delete ' + $(el2).attr('dirname') + '?', function() {
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Recycle.wav?1522621854189');
xp.filesystem.deleteDir(fullpath, function(e){
if (e) {
xp.dialog('Error', e, function() {}, false);
}
exp_ofp($(el).data('directory'), guid, $(el).attr("guid"));
});
}, true);
}
}
xp.filesystem.openFileDialog = function(callback) {
var guid = generate_guid();
var path = '/Documents and Settings/' + xp.profile.name + '/My Documents';
var el = $.parseHTML(`<window title="Open File" width="480" height="360">
<div class="explorer_toolbar">
<div>
<button class="custombutton navbutton" id="moveBack_` + guid + `">
<img draggable="false" src="` + explorer.icons.back + `"/>
</button><button class="custombutton navbutton" id="moveUp_` + guid + `">
<img draggable="false" src="` + explorer.icons.up + `"/>
</button>
</div>
<div style="height:20px;vertical-align:top;">
<input type="text" value="` + path + `" style="width: calc(100% - 24px);height:20px;vertical-align:top;" id="filepath_` + guid + `"/>
<button class="gobutton custombutton" id="GoButton_` + guid + `"></button>
</div>
</div>
<div class="explorer_files yscroll" style="height:calc(100% - 77px);">
<ul class="menu" id="openfilemenu_` + guid + `">
</ul>
<div class="hiddenfiles" style="display:none;height:calc(100% - 97px);">
<h1>These files are hidden.</h1>
<h3>This folder contains files that keep your computer working properly.<br/>
You should not modify its contents.</h3>
<a href="" onclick="$(this).closest('.hiddenfiles').css('display', 'none');return false;">Show the contents of this folder</a>
</div>
</div>
<footer>
<div style="float:right;">
<button class="open">Open</button>
<button class="cancel">Cancel</button>
</div>
</footer>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
$(el).find('.cancel').on('click', () => closeWindow($(el).attr('guid')));
$(el).find('.open').on('click', () => {
$('._file_' + guid).each(function() {
if ($(this).data('selected') !== 'false') {
closeWindow($(el).attr('guid'));
callback($(this).attr('fullpath'));
}
});
});
$('#openfilemenu_' + guid).closest('.explorer_files').on('click', function() {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
});
$('#GoButton_' + guid).on('click', () => {
var path = $("#filepath_" + guid).val();
while (path.charAt(path.length - 1) === "/") {
path = path.slice(0, -1);
}
exp_ofp(path, guid, $("#filepath_" + guid).closest("window").attr("guid"), callback);
});
exp_ofp(path, guid, $(el).attr("guid"), callback);
}
xp.filesystem.saveFileDialog = function(callback, defaultName) {
var guid = generate_guid();
var path = '/Documents and Settings/' + xp.profile.name + '/My Documents';
if (defaultName === undefined) defaultName = 'Untitled.txt';
var el = $.parseHTML(`<window title="Save File" width="480" height="360">
<div class="explorer_toolbar">
<div>
<button class="custombutton navbutton" id="moveBack_` + guid + `">
<img draggable="false" src="` + explorer.icons.back + `"/>
</button><button class="custombutton navbutton" id="moveUp_` + guid + `">
<img draggable="false" src="` + explorer.icons.up + `"/>
</button>
</div>
<div style="height:20px;vertical-align:top;">
<input type="text" value="` + path + `" style="width: calc(100% - 24px);height:20px;vertical-align:top;" id="filepath_` + guid + `"/>
<button class="gobutton custombutton" id="GoButton_` + guid + `"></button>
</div>
</div>
<div class="explorer_files yscroll" style="height: calc(100% - 77px);">
<ul class="menu" id="openfilemenu_` + guid + `">
</ul>
<div class="hiddenfiles" style="display:none;height:calc(100% - 97px);">
<h1>These files are hidden.</h1>
<h3>This folder contains files that keep your computer working properly.<br/>
You should not modify its contents.</h3>
<a href="" onclick="$(this).closest('.hiddenfiles').css('display', 'none');return false;">Show the contents of this folder</a>
</div>
</div>
<footer>
<input type="text" value="` + defaultName + `" style="width: calc(100% - 157px)" id="savefilename_` + guid + `"/>
<button id="savebutton_` + guid + `">Save</button>
<button class="cancel">Cancel</button>
</footer>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
$(el).find('.cancel').on('click', () => closeWindow($(el).attr('guid')));
$('#openfilemenu_' + guid).closest('.explorer_files').on('click', function() {
$('._file_' + guid).each(function() {
$(this).data('selected', 'false');
$(this).attr('data-selected', 'false');
});
});
exp_ofp(path, guid, $(el).attr("guid"), (e) => $('#savefilename_' + guid).val(e), callback, true);
$("#savebutton_" + guid).on('click', () => {
callback(xp.filesystem.addPaths($(el).data('directory'), $('#savefilename_' + guid).val()));
closeWindow($(el).attr("guid"));
});
$('#GoButton_' + guid).on('click', () => {
var path = $("#filepath_" + guid).val();
while (path.charAt(path.length - 1) === "/") {
path = path.slice(0, -1);
}
exp_ofp(path, guid, $(el).attr("guid"), (e) => $('#savefilename_' + guid).val(e), callback, true);
});
}

227
xp/system/filesystem.js Normal file
View File

@ -0,0 +1,227 @@
'use strict';
xp.filesystem = {};
xp.filesystem.icons = {};
xp.filesystem.icons.smallFolder = '<img src=""/>';
xp.filesystem.icons.smallFile = '<img src=""/>';
xp.filesystem.type = 'html5';
xp.filesystem.listDir = function(path, callback) {
xp.filesystem.fs.root.getDirectory(path, {create: false}, function(dirEntry) {
var dirReader = dirEntry.createReader();
var entries = [];
var i = 0;
function readEntries() {
dirReader.readEntries (function(results) {
if (!results.length) {
entries.forEach(function(entry, i) {
var name = entry.name + (entry.isDirectory ? "/" : "");
callback(name);
});
} else {
entries = entries.concat(Array.from(results));
readEntries();
}
}, function(e) {callback(e)});
};
readEntries();
}, function(e) {callback(e)});
};
xp.filesystem.createFile = function(name, callback) {
xp.filesystem.fs.root.getFile(name, {create: true}, function(fileEntry) {
callback();
}, function(e) {callback(e)});
}
xp.filesystem.deleteFile = function(name, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
fileEntry.remove(function() {
callback();
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.createDir = function(name, callback) {
xp.filesystem.fs.root.getDirectory(name, {create: true}, function(dirEntry) {
callback();
}, function(e) {callback(e)});
}
xp.filesystem.deleteDir = function(name, callback) {
xp.filesystem.fs.root.getDirectory(name, {create: false}, function(dirEntry) {
dirEntry.removeRecursively(function() {
callback();
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.readFile = function(name, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
fileEntry.file(function(file) {
var reader = new FileReader();
reader.onloadend = function(e) {
callback(this.result);
};
reader.readAsText(file);
}, function(e) {callback("")});
}, function(e) {callback("")});
}
xp.filesystem.readBinaryFile = function(name, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
fileEntry.file(function(file) {
var reader = new FileReader();
reader.onloadend = function(e) {
callback(this.result);
};
reader.readAsArrayBuffer(file);
}, function(e) {callback("")});
}, function(e) {callback("")});
}
xp.filesystem.writeFile = function(name, blob, callback) {
xp.filesystem.deleteFile(name, (e) => {
xp.filesystem.fs.root.getFile(name, {create: true}, function(fileEntry) {
fileEntry.createWriter(function(fileWriter) {
fileWriter.onwriteend = function(e) {
callback();
};
fileWriter.onerror = function(e) {callback(e)};
fileWriter.write(blob);
}, function(e) {callback(e)});
}, function(e) {callback(e)});
});
}
xp.filesystem.moveFile = function(name, newname, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
xp.filesystem.fs.root.getDirectory(newname.substr(0, newname.lastIndexOf("/")), {}, function(dirEntry) {
fileEntry.moveTo(dirEntry, newname.substr(newname.lastIndexOf("/") + 1, newname.length));
setTimeout(() => callback(), 100);
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.moveDir = function(name, newname, callback) {
xp.filesystem.fs.root.getDirectory(name, {create: false}, function(fileEntry) {
xp.filesystem.fs.root.getDirectory(newname.substr(0, newname.lastIndexOf("/")), {}, function(dirEntry) {
fileEntry.moveTo(dirEntry, newname.substr(newname.lastIndexOf("/") + 1, newname.length));
setTimeout(() => callback(), 100);
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.copyFile = function(name, newname, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
xp.filesystem.fs.root.getDirectory(newname.substr(0, newname.lastIndexOf("/")), {}, function(dirEntry) {
fileEntry.copyTo(dirEntry, newname.substr(newname.lastIndexOf("/") + 1, newname.length));
setTimeout(() => callback(), 100);
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.copyDir = function(name, newname, callback) {
xp.filesystem.fs.root.getDirectory(name, {create: false}, function(fileEntry) {
xp.filesystem.fs.root.getDirectory(newname.substr(0, newname.lastIndexOf("/")), {}, function(dirEntry) {
fileEntry.copyTo(dirEntry, newname.substr(newname.lastIndexOf("/") + 1, newname.length));
setTimeout(() => callback(), 100);
}, function(e) {callback(e)});
}, function(e) {callback(e)});
}
xp.filesystem.toURL = function(name, callback) {
xp.filesystem.fs.root.getFile(name, {create: false}, function(fileEntry) {
callback(fileEntry.toURL());
}, function(e) {callback("")});
}
xp.filesystem.getDir = function(_dir) {
if (_dir === undefined)
_dir = [""];
var dir = _dir.join("/");
if (dir === "") dir = "/";
return dir;
}
xp.filesystem.addPaths = function(p1, p2) {
var foo = p2.split("/");
var bar = p1.split("/");
foo.forEach(function(val, i) {
if (val === "..") {
bar.splice(-1,1)
} else {
bar = bar.concat(val);
}
});
return xp.filesystem.getDir(bar).replace(/\/\/+/g, '/');
}
xp.filesystem.basename = function(str) {
var base = new String(str).substring(str.lastIndexOf('/') + 1);
if(base.lastIndexOf(".") != -1)
base = base.substring(0, base.lastIndexOf("."));
return base;
}
xp.filesystem.create = function(size, callback, ignoreerrors) {
console.log('Loading idb.filesystem.min.js');
xp.filesystem.type = 'system/idb.filesystem.js';
$.getScript(
'system/idb.filesystem.min.js',
() => {
window.requestFileSystem = window.requestFileSystem || window.webkitRequestFileSystem;
window.directoryEntry = window.directoryEntry || window.webkitDirectoryEntry;
function requestFs(grantedBytes) {
console.log('Requesting persistent file system with size of ' + grantedBytes + ' bytes');
window.requestFileSystem(PERSISTENT, grantedBytes, function(fs) {
xp.filesystem.fs = fs;
console.log('Request successful');
callback();
}, callback);
}
if (window.webkitStorageInfo === undefined) {
requestFs(size);
} else {
window.webkitStorageInfo.requestQuota(PERSISTENT, size, requestFs, callback);
}
}
);
/*try {
window.requestFileSystem = window.requestFileSystem || window.webkitRequestFileSystem;
window.directoryEntry = window.directoryEntry || window.webkitDirectoryEntry;
function requestFs(grantedBytes) {
console.log('Requesting persistent file system with size of ' + grantedBytes + ' bytes');
window.requestFileSystem(PERSISTENT, grantedBytes, function(fs) {
xp.filesystem.fs = fs;
console.log('Request successful');
callback();
}, callback);
}
if (window.webkitStorageInfo === undefined) {
requestFs(size);
} else {
window.webkitStorageInfo.requestQuota(PERSISTENT, size, requestFs, callback);
}
} catch(e) {
if (ignoreerrors !== true) {
console.log('Caught ' + e + ', reverting to idb.filesystem.js for storage');
console.log('Loading ' + window.location.protocol + '//rawgit.com/ebidel/idb.filesystem.js/master/dist/idb.filesystem.min.js');
xp.filesystem.type = 'idb.filesystem.js';
$.getScript(
window.location.protocol + '//rawgit.com/ebidel/idb.filesystem.js/master/dist/idb.filesystem.min.js',
() => xp.filesystem.create(size, callback, true)
);
} else {
console.log(e);
}
}*/
}

49
xp/system/fonts.css Normal file
View File

@ -0,0 +1,49 @@
@font-face {
font-family: "Tahoma";
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftahoma.ttf?1520032662940);
}
@font-face {
font-family: "Tahoma Bold";
font-weight: bold;
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftahomabd.ttf?1520032662939);
}
@font-face {
font-family: "Trebuchet MS";
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftrebuc.ttf?1520032658142);
}
@font-face {
font-family: "Trebuchet MS Bold";
font-weight: bold;
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftrebucbd.ttf?1520032658347);
}
@font-face {
font-family: "Trebuchet MS Bold Italic";
font-weight: bold;
font-style: italic;
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftrebucbi.ttf?1520032659151);
}
@font-face {
font-family: "Trebuchet MS Italic";
font-style: italic;
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftrebucit.ttf?1520032658880);
}
@font-face {
font-family: "PxPlus IBM VGA9";
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FPxPlus_IBM_VGA9.ttf?1520115808634);
}
@font-face {
font-family: "Tomo";
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Ftomo.woff2?1521568990198);
}
@font-face {
font-family: "px sans noveaux";
src: url(https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fpx_sans_nouveaux.ttf?1540345601242);
}

29
xp/system/help.js Normal file
View File

@ -0,0 +1,29 @@
$(window).on('xpboot', () => {
xp.applications.add('winhelp', () => {
var el = $.parseHTML(`<window width="800" height="500" title="Help & Support">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe height="100%" seamless="seamless" width="100%" src="system/help.html"></iframe>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
});
});

71
xp/system/icons.css Normal file

File diff suppressed because one or more lines are too long

28
xp/system/jquery.terminal-2.0.0.min.css vendored Normal file

File diff suppressed because one or more lines are too long

41
xp/system/jquery.terminal-2.0.0.min.js vendored Normal file

File diff suppressed because one or more lines are too long

639
xp/system/lambda.js Normal file
View File

@ -0,0 +1,639 @@
// http://lisperator.net/pltut/
// http://lisperator.net/s/lambda/lambda-eval4.js
function InputStream(input) {
var pos = 0, line = 1, col = 0;
return {
next : next,
peek : peek,
eof : eof,
croak : croak,
};
function next() {
var ch = input.charAt(pos++);
if (ch == "\n") line++, col = 0; else col++;
return ch;
}
function peek() {
return input.charAt(pos);
}
function eof() {
return peek() == "";
}
function croak(msg) {
throw new Error(msg + " (" + line + ":" + col + ")");
}
}
function TokenStream(input) {
var current = null;
var keywords = " let if then else lambda λ true false ";
return {
next : next,
peek : peek,
eof : eof,
croak : input.croak
};
function is_keyword(x) {
return keywords.indexOf(" " + x + " ") >= 0;
}
function is_digit(ch) {
return /[0-9]/i.test(ch);
}
function is_id_start(ch) {
return /[a-zλ_]/i.test(ch);
}
function is_id(ch) {
return is_id_start(ch) || "?!-<>=0123456789".indexOf(ch) >= 0;
}
function is_op_char(ch) {
return "+-*/%=&|<>!".indexOf(ch) >= 0;
}
function is_punc(ch) {
return ",;(){}[]".indexOf(ch) >= 0;
}
function is_whitespace(ch) {
return " \t\n".indexOf(ch) >= 0;
}
function read_while(predicate) {
var str = "";
while (!input.eof() && predicate(input.peek()))
str += input.next();
return str;
}
function read_number() {
var has_dot = false;
var number = read_while(function(ch){
if (ch == ".") {
if (has_dot) return false;
has_dot = true;
return true;
}
return is_digit(ch);
});
return { type: "num", value: parseFloat(number) };
}
function read_ident() {
var id = read_while(is_id);
return {
type : is_keyword(id) ? "kw" : "var",
value : id
};
}
function read_escaped(end) {
var escaped = false, str = "";
input.next();
while (!input.eof()) {
var ch = input.next();
if (escaped) {
str += ch;
escaped = false;
} else if (ch == "\\") {
escaped = true;
} else if (ch == end) {
break;
} else {
str += ch;
}
}
return str;
}
function read_string() {
return { type: "str", value: read_escaped('"') };
}
function skip_comment() {
read_while(function(ch){ return ch != "\n" });
input.next();
}
function read_next() {
read_while(is_whitespace);
if (input.eof()) return null;
var ch = input.peek();
if (ch == "#") {
skip_comment();
return read_next();
}
if (ch == '"') return read_string();
if (is_digit(ch)) return read_number();
if (is_id_start(ch)) return read_ident();
if (is_punc(ch)) return {
type : "punc",
value : input.next()
};
if (is_op_char(ch)) return {
type : "op",
value : read_while(is_op_char)
};
input.croak("Can't handle character: " + ch);
}
function peek() {
return current || (current = read_next());
}
function next() {
var tok = current;
current = null;
return tok || read_next();
}
function eof() {
return peek() == null;
}
}
function parse(input) {
var PRECEDENCE = {
"=": 1,
"||": 2,
"&&": 3,
"<": 7, ">": 7, "<=": 7, ">=": 7, "==": 7, "!=": 7,
"+": 10, "-": 10,
"*": 20, "/": 20, "%": 20,
};
var FALSE = { type: "bool", value: false };
return parse_toplevel();
function is_punc(ch) {
var tok = input.peek();
return tok && tok.type == "punc" && (!ch || tok.value == ch) && tok;
}
function is_kw(kw) {
var tok = input.peek();
return tok && tok.type == "kw" && (!kw || tok.value == kw) && tok;
}
function is_op(op) {
var tok = input.peek();
return tok && tok.type == "op" && (!op || tok.value == op) && tok;
}
function skip_punc(ch) {
if (is_punc(ch)) input.next();
else input.croak("Expecting punctuation: \"" + ch + "\"");
}
function skip_kw(kw) {
if (is_kw(kw)) input.next();
else input.croak("Expecting keyword: \"" + kw + "\"");
}
function skip_op(op) {
if (is_op(op)) input.next();
else input.croak("Expecting operator: \"" + op + "\"");
}
function unexpected() {
input.croak("Unexpected token: " + JSON.stringify(input.peek()));
}
function maybe_binary(left, my_prec) {
var tok = is_op();
if (tok) {
var his_prec = PRECEDENCE[tok.value];
if (his_prec > my_prec) {
input.next();
return maybe_binary({
type : tok.value == "=" ? "assign" : "binary",
operator : tok.value,
left : left,
right : maybe_binary(parse_atom(), his_prec)
}, my_prec);
}
}
return left;
}
function delimited(start, stop, separator, parser) {
var a = [], first = true;
skip_punc(start);
while (!input.eof()) {
if (is_punc(stop)) break;
if (first) first = false; else skip_punc(separator);
if (is_punc(stop)) break;
a.push(parser());
}
skip_punc(stop);
return a;
}
function parse_call(func) {
return {
type: "call",
func: func,
args: delimited("(", ")", ",", parse_expression),
};
}
function parse_varname() {
var name = input.next();
if (name.type != "var") input.croak("Expecting variable name");
return name.value;
}
function parse_vardef() {
var name = parse_varname(), def;
if (is_op("=")) {
input.next();
def = parse_expression();
}
return { name: name, def: def };
}
function parse_let() {
skip_kw("let");
if (input.peek().type == "var") {
var name = input.next().value;
var defs = delimited("(", ")", ",", parse_vardef);
return {
type: "call",
func: {
type: "lambda",
name: name,
vars: defs.map(function(def){ return def.name }),
body: parse_expression(),
},
args: defs.map(function(def){ return def.def || FALSE })
};
}
return {
type: "let",
vars: delimited("(", ")", ",", parse_vardef),
body: parse_expression(),
};
}
function parse_if() {
skip_kw("if");
var cond = parse_expression();
if (!is_punc("{")) skip_kw("then");
var then = parse_expression();
var ret = {
type: "if",
cond: cond,
then: then,
};
if (is_kw("else")) {
input.next();
ret.else = parse_expression();
}
return ret;
}
function parse_lambda() {
return {
type: "lambda",
name: input.peek().type == "var" ? input.next().value : null,
vars: delimited("(", ")", ",", parse_varname),
body: parse_expression()
};
}
function parse_bool() {
return {
type : "bool",
value : input.next().value == "true"
};
}
function maybe_call(expr) {
expr = expr();
return is_punc("(") ? parse_call(expr) : expr;
}
function parse_atom() {
return maybe_call(function(){
if (is_punc("(")) {
input.next();
var exp = parse_expression();
skip_punc(")");
return exp;
}
if (is_punc("{")) return parse_prog();
if (is_kw("let")) return parse_let();
if (is_kw("if")) return parse_if();
if (is_kw("true") || is_kw("false")) return parse_bool();
if (is_kw("lambda") || is_kw("λ")) {
input.next();
return parse_lambda();
}
var tok = input.next();
if (tok.type == "var" || tok.type == "num" || tok.type == "str")
return tok;
unexpected();
});
}
function parse_toplevel() {
var prog = [];
while (!input.eof()) {
prog.push(parse_expression());
if (!input.eof()) skip_punc(";");
}
return { type: "prog", prog: prog };
}
function parse_prog() {
var prog = delimited("{", "}", ";", parse_expression);
if (prog.length == 0) return FALSE;
if (prog.length == 1) return prog[0];
return { type: "prog", prog: prog };
}
function parse_expression() {
return maybe_call(function(){
return maybe_binary(parse_atom(), 0);
});
}
}
function Environment(parent) {
this.vars = Object.create(parent ? parent.vars : null);
this.parent = parent;
}
Environment.prototype = {
extend: function() {
return new Environment(this);
},
lookup: function(name) {
var scope = this;
while (scope) {
if (Object.prototype.hasOwnProperty.call(scope.vars, name))
return scope;
scope = scope.parent;
}
},
get: function(name) {
if (name in this.vars)
return this.vars[name];
throw new Error("Undefined variable " + name);
},
set: function(name, value) {
var scope = this.lookup(name);
if (!scope && this.parent)
throw new Error("Undefined variable " + name);
return (scope || this).vars[name] = value;
},
def: function(name, value) {
return this.vars[name] = value;
}
};
function evaluate(exp, env, callback) {
GUARD(evaluate, arguments);
switch (exp.type) {
case "num":
case "str":
case "bool":
callback(exp.value);
return;
case "var":
callback(env.get(exp.value));
return;
case "assign":
if (exp.left.type != "var")
throw new Error("Cannot assign to " + JSON.stringify(exp.left));
evaluate(exp.right, env, function CC(right){
GUARD(CC, arguments);
callback(env.set(exp.left.value, right));
});
return;
case "binary":
evaluate(exp.left, env, function CC(left){
GUARD(CC, arguments);
evaluate(exp.right, env, function CC(right){
GUARD(CC, arguments);
callback(apply_op(exp.operator, left, right));
});
});
return;
case "let":
(function loop(env, i){
GUARD(loop, arguments);
if (i < exp.vars.length) {
var v = exp.vars[i];
if (v.def) evaluate(v.def, env, function CC(value){
GUARD(CC, arguments);
var scope = env.extend();
scope.def(v.name, value);
loop(scope, i + 1);
}); else {
var scope = env.extend();
scope.def(v.name, false);
loop(scope, i + 1);
}
} else {
evaluate(exp.body, env, callback);
}
})(env, 0);
return;
case "lambda":
callback(make_lambda(env, exp));
return;
case "if":
evaluate(exp.cond, env, function CC(cond){
GUARD(CC, arguments);
if (cond !== false) evaluate(exp.then, env, callback);
else if (exp.else) evaluate(exp.else, env, callback);
else callback(false);
});
return;
case "prog":
(function loop(last, i){
GUARD(loop, arguments);
if (i < exp.prog.length) evaluate(exp.prog[i], env, function CC(val){
GUARD(CC, arguments);
loop(val, i + 1);
}); else {
callback(last);
}
})(false, 0);
return;
case "call":
evaluate(exp.func, env, function CC(func){
GUARD(CC, arguments);
(function loop(args, i){
GUARD(loop, arguments);
if (i < exp.args.length) evaluate(exp.args[i], env, function CC(arg){
GUARD(CC, arguments);
args[i + 1] = arg;
loop(args, i + 1);
}); else {
func.apply(null, args);
}
})([ callback ], 0);
});
return;
default:
throw new Error("I don't know how to evaluate " + exp.type);
}
}
function make_lambda(env, exp) {
if (exp.name) {
env = env.extend();
env.def(exp.name, lambda);
}
function lambda(callback) {
GUARD(lambda, arguments);
var names = exp.vars;
var scope = env.extend();
for (var i = 0; i < names.length; ++i)
scope.def(names[i], i + 1 < arguments.length ? arguments[i + 1] : false);
evaluate(exp.body, scope, callback);
}
return lambda;
}
function apply_op(op, a, b) {
function num(x) {
if (typeof x != "number")
throw new Error("Expected number but got " + x);
return x;
}
function div(x) {
if (num(x) == 0)
throw new Error("Divide by zero");
return x;
}
switch (op) {
case "+": return num(a) + num(b);
case "-": return num(a) - num(b);
case "*": return num(a) * num(b);
case "/": return num(a) / div(b);
case "%": return num(a) % div(b);
case "&&": return a !== false && b;
case "||": return a !== false ? a : b;
case "<": return num(a) < num(b);
case ">": return num(a) > num(b);
case "<=": return num(a) <= num(b);
case ">=": return num(a) >= num(b);
case "==": return a === b;
case "!=": return a !== b;
}
throw new Error("Can't apply operator " + op);
}
var STACKLEN;
function GUARD(f, args) {
if (STACKLEN - 1 < 0) throw new Continuation(f, args);
}
function Continuation(f, args) {
this.f = f;
this.args = args;
}
function Execute(f, args) {
while (true) try {
STACKLEN = 200;
return f.apply(null, args);
} catch(ex) {
if (ex instanceof Continuation)
f = ex.f, args = ex.args;
else throw ex;
}
}
function createEnv(path) {
var env = new Environment();
env.scriptPath = path;
env.def("CallCC", function CallCC(k, func) {
GUARD(CallCC, arguments);
func(k, function CC(discarded, ret){
GUARD(CC, arguments);
k(ret);
});
});
env.def("time", function(k, func) {
console.time("time");
func(function(ret){
console.timeEnd("time");
k(ret);
});
});
env.def("alert", function alert(k, val) {
function callback() {
k(false);
}
xp.alert(val, callback, callback);
});
env.def("prompt", function prompt(k, val) {
function callback(res) {
k('' + (res || ''));
}
xp.prompt('Alert', val, callback, arguments.length > 2 ? arguments[2] : '', callback);
});
env.def("ConcatStr", function(k, nope) {
let res = '';
for (let i = 1; i < arguments.length; i ++) {
res += arguments[i];
}
k(res);
});
env.def("ParseInt", function(k, notanint) {
k(parseInt(notanint));
});
env.def("Array", function(k, nope) {
let res = [];
for (let i = 1; i < arguments.length; i ++) {
res.push(arguments[i]);
}
k(res);
});
env.def("ArrayGet", function(k, nope) {
k(arguments[1][arguments[2]]);
});
env.def("ArraySet", function(k, nope) {
arguments[1][arguments[2]] = arguments[3];
k(arguments[3]);
});
env.def("ArrayIndexOf", function arrayIndexOf(k, nope) {
GUARD(arrayIndexOf, arguments);
k(arguments[1].indexOf(arguments[2]));
});
env.def("SplitStr", function(k, nope) {
k(arguments[1].split(arguments[2]));
});
env.def("ToLowerCase", function(k, str) {
k(('' + str).toLowerCase());
});
env.def("ToUpperCase", function(k, str) {
k(('' + str).toUpperCase());
});
env.def("execScript", function(k, filename) {
execLDFile(xp.filesystem.addPaths(env.scriptPath, filename), env.scriptPath);
k(false);
});
return env;
}
function execLD(code, path) {
try {
var ast = parse(TokenStream(InputStream(code)));
var env = createEnv(path);
Execute(evaluate, [ast, env, function(result){
console.log("*** Result:", result);
}]);
} catch(e) {
xp.error(e);
}
}
function execLDFile(path, dir) {
xp.filesystem.readFile(path, (text) => {
execLD(text, dir);
});
}
$(window).on('xpboot', () => {
explorer.fileHandlers.add('lda', (file) => {
execLDFile(file, file.substring(0, file.lastIndexOf("/")));
});
xp.applications.add('lda', (args) => {
if (args.length < 2) {
xp.alert('Usage: lda <filename>');
} else {
execLDFile(xp.filesystem.addPaths(args[0], args[1]), args[0]);
}
});
});

44
xp/system/script.js Normal file
View File

@ -0,0 +1,44 @@
function execJsPrompt(path) {
xp.dialog('Run Javascript File?', 'Would you like to execute the javascript file "' + path + '"?', function() {
loadJs(['', path]);
}, true, 'warning');
}
function w93prg(args) {
var el = $.parseHTML(`<window title="` + args[1] + `" width="640" height="480">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
/*display: none;*/
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe seamless="seamless" width="100%" height="100%" id="frame" src="//www.windows93.net/#!/c/programs/` + args[1] + `/"></iframe>
</div>
</window>`);
document.body.appendChild(el[0]);
$(el).updateWindow();
$(el).find('iframe').focus();
}
function loadJs(args) {
if (args !== undefined && args[1] !== undefined) {
var jsFile = xp.filesystem.addPaths(args[0], args[1]);
xp.filesystem.readFile(jsFile, function(e) {
if (e !== "") {
eval(e);
}
});
}
}

136
xp/system/startmenu.css Normal file

File diff suppressed because one or more lines are too long

396
xp/system/uac.js Normal file
View File

@ -0,0 +1,396 @@
$(window).on('xpboot', () => {
xp.controlpanel.add('User Accounts', () => {
var win = new Window({
width: 522,
height: 462,
title: 'User Accounts',
canResize: false,
});
win.content(`
<style>
.userimg {
width: 64px;
height: 64px;
}
.userimgopt {
border: 1px solid transparent;
width: 48px;
height: 48px;
padding: 4px;
border-radius: 5px;
}
.userimgopt.selected {
border: 1px solid #68b3db;
background-color: #e1f2fb;
}
.cp_option {
color: #fff;
}
</style>
<div style="width:100%;height:100%;background-color:#6375d6;" class="outercontent">
<div style="margin-left:32px;width:calc(100% - 64px);" class="content">
<br/>
<h1 style="color:#d6dff5">Pick a task...</h1>
<p>
<span class="cp_option pointer change_acct">Change an account</span>
</p>
<p>
<span class="cp_option pointer new_acct">Create a new account</span>
</p>
<h1 style="color:#d6dff5">or pick an account to change</h1>
<div style="width:100%overflow-x:auto;">
<table style="font-size:11px;color:#fff;">
<tr class="accounts">
</tr>
</table>
</center>
</div>
</div>`);
function changeAccount(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
var canDelete = '';
if (acct.name !== xp.profile.name) {
canDelete = `
<p>
<span style="color:#000;" class="cp_option pointer delete_account">Delete the account</span>
</p>`;
}
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">What do you want to change about ${acctname}<br/>account?</h1>
<p>
<span style="color:#000;" class="cp_option pointer change_name">Change ${acct.name === xp.profile.name ? 'my' : 'the'} name</span>
</p>
<p>
<span style="color:#000;" class="cp_option pointer change_picture">Change ${acct.name === xp.profile.name ? 'my' : 'the'} picture</span>
</p>${canDelete}
<table style="font-size:11px;color:#000;position:absolute;left:269px;top:97px">
<tr><td rowspan="4" style="width:64px;"><img class="userimg" style="width:48px;height:48px;" src="${acct.image}"/></td></tr>
<tr><td><b>${acct.name}</b></td></tr>
<tr><td>Computer administrator</td></tr>
<tr><td>&nbsp;</td></tr><tr><td>&nbsp;</td></tr>
</table>`);
win.el.find('.change_name').on('click', () => changeName(acct));
win.el.find('.change_picture').on('click', () => changePicture(acct));
win.el.find('.delete_account').on('click', () => deleteAccount(acct));
}
function deleteAccount(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Do you want to keep ${acctname} files?</h1>
<p>
Before you delete ${acctname} account, Windows can automatically save the contents of ${acctname}<br/>
My Documents to a folder called ${acct.name} on your desktop. However, Windows<br/>
cannot save ${acctname} e-mail messages, Internet favorites, and other settings.
</p>
<hr/>
<p>
<div style="float:right">
<button class="keep">Keep Files</button>
<button class="delete">Delete Files</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
win.el.find('.keep').on('click', () => {
xp.filesystem.moveDir('/Documents and Settings/' + acct.name + '/My Documents', '/Documents and Settings/' + xp.profile.name + '/My Documents/' + acct.name, () => {
xp.filesystem.deleteDir('/Documents and Settings/' + acct.name, () => {
win.close();
});
});
});
win.el.find('.delete').on('click', () => {
xp.filesystem.deleteDir('/Documents and Settings/' + acct.name, () => {
win.close();
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function changeName(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Provide a new name for ${acctname} account</h1>
<p>
Type a new name for ${acct.name}:<br/>
<input type="text" style="width:212px" class="username" value="${acct.name}"/><br/>
This name will appear on the Welcome screen and on the Start menu.<br/>
Note: Changing ${acctname} name will require a reboot.
</p>
<hr/>
<p>
<div style="float:right">
<button class="change">Change Name</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
win.el.find('.change').on('click', () => {
var oldProfile = xp.profile;
xp.profile = acct;
var newUsername = win.el.find('.username').val();
if (xp.profile.name !== newUsername) {
xp.filesystem.moveDir(`/Documents and Settings/${xp.profile.name}`, `/Documents and Settings/${newUsername}`, () => {});
xp.profile.name = newUsername;
}
saveConfig(() => {
xp.profile = oldProfile;
xp.alert('Changes have been applied.<br/>Click OK to reboot your computer.', () => {
window.location.href = window.location.href;
});
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function changePicture(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Pick a new picture for ${acctname} account</h1>
<p>
The picture you choose will appear on the Welcome screen.<br/>
Note: Changing your picture will require a reboot.
</p>
<p style="margin-left:16px;overflow-x:scroll;width:100%;">
<table style="font-size:11px;color:#000;">
<tr class="images">
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile2.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile3.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile4.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile5.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile6.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile7.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile8.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile9.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile10.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile11.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile12.bmp?1522560745448"/></td>
</tr>
</table>
</p>
<p>
<button class="browse">Browse for more pictures</button>
</p>
<hr/>
<p>
<div style="float:right">
<button class="change">Change Picture</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
if (win.el.find(`.userimgopt[src="${acct.image}"]`).length > 0)
win.el.find(`.userimgopt[src="${acct.image}"]`).addClass('selected');
else
win.el.find('.images').append(`<td><img class="userimgopt selected" src="${acct.image}"/></td>`);
win.el.find('.userimgopt').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
});
win.el.find('.browse').on('click', () => {
xp.filesystem.openFileDialog((file) => {
xp.filesystem.toURL(file, (url) => {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
var el = $($.parseHTML(`<td><img class="userimgopt selected" src="${url}"/></td>`));
win.el.find('.images').append(el);
el.find('img').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
});
});
});
});
win.el.find('.change').on('click', () => {
var oldProfile = xp.profile;
xp.profile = acct;
xp.profile.image = win.el.find('.userimgopt.selected').attr('src');
saveConfig(() => {
xp.profile = oldProfile;
xp.alert('Changes have been applied.<br/>Click OK to reboot your computer.', () => {
window.location.href = window.location.href;
});
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function listAccounts() {
xp.filesystem.listDir('/Documents and Settings', (acct) => {
acct = acct.split('/')[0];
xp.filesystem.readFile(`/Documents and Settings/${acct}/config.json`, (file) => {
var config = JSON.parse(file);
var el = $.parseHTML(`<td>
<img class="userimgopt pointer" account="${acct}" src="${config.profile.image}"/>
<div style="text-align:center;" class="pointer">${acct}</div>
</td>`);
$(el).find('img').on('click', () => changeAccount(config.profile));
win.el.find('.accounts').append(el);
});
});
}
listAccounts();
win.el.find('.cp_option.new_acct').on('click', () => {
newAccount();
});
win.el.find('.cp_option.change_acct').on('click', () => {
win.el.find('.outercontent').css('background-color', '#fff');
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Pick an account to change</h1>
<div style="width:100%overflow-x:auto;">
<table style="font-size:11px;color:#000;">
<tr class="accounts">
</tr>
</table>
</center>`);
listAccounts();
});
function newAccount() {
let win = new Window({
width: 594,
height: 434,
title: 'Create New Account',
canResize: false,
center: true
});
win.content(`
<style>
.userimg {
width: 64px;
height: 64px;
}
.userimgopt {
border: 1px solid transparent;
width: 48px;
height: 48px;
padding: 4px;
border-radius: 5px;
}
.userimgopt.selected {
border: 1px solid #68b3db;
background-color: #e1f2fb;
}
</style>
<div style="margin-left:32px;">
<h1 style="font-weight:normal;">Create User Account</h1>
<h2 style="font-weight:normal;">Choose a user name and picture</h2>
<p>
Your user name and picture represent your user account. The account you create here is a computer<br>
administrator account. It is recommended to close all open applications before continuing.
</p>
<br/>
<p style="margin-left:64px;">
<table style="font-size:11px;color:#000;">
<tr><td rowspan="4" style="width:72px;"><img class="userimg" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td></tr>
<tr><td>Type a user name (for example, John)</td></tr>
<tr><td><input type="text" class="username" style="width:180px;"/></td></tr>
<tr><td>&nbsp;</td></tr>
</table>
</p>
<br/><br/>
<p style="margin-left:16px;">
<table style="font-size:11px;color:#000;">
<tr>
<td><img class="userimgopt selected" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile2.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile3.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile4.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile5.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile6.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile7.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile8.bmp?1522560745448"/></td>
</tr>
</table>
</p>
<div style="position:absolute;right:8px;bottom:8px;">
<button disabled class="next">Next</button>
</div>
</div>
`);
win.el.find('.username').on('keyup', function() {
console.log($(this).val());
if ($(this).val() === '') {
win.el.find('.next').attr('disabled', 'disabled');
} else {
win.el.find('.next').removeAttr('disabled');
}
});
win.el.find('.next').on('click', function() {
if (win.el.find('.username').val() === '') {
xp.alert('You need to type a username in order to continue!');
} else {
xp.profile.image = win.el.find('.userimgopt.selected').attr('src');
xp.profile.name = win.el.find('.username').val();
var oldConfigFile = configFile;
configFile = `/Documents and Settings/${xp.profile.name}/config.json`;
xp.wallpaper.href = 'https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FBliss.jpg?1519950052202';
win.content(`
<div style="margin-left:32px;">
<h1 style="font-weight:normal;">Please wait while we create the user account.</h1>
</div>`);
setTimeout(() => {
requiredDirectories = [
`/Documents and Settings`,
`/Documents and Settings/${xp.profile.name}`,
`/Documents and Settings/${xp.profile.name}/My Documents`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Pictures`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Videos`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Music`
];
var i = 0;
function createDirs() {
var dirToCreate = requiredDirectories[i];
if (dirToCreate !== undefined) {
xp.filesystem.createDir(dirToCreate, (e) => {
i ++;
createDirs();
});
} else {
saveConfig(() => {
configFile = oldConfigFile;
loadConfig(() => {
win.close();
});
});
}
}
createDirs();
}, 2000);
}
});
win.el.find('.userimgopt').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
win.el.find('.userimg').attr('src', $(this).attr('src'));
});
}
});
});

396
xp/system/uac_app.js Normal file
View File

@ -0,0 +1,396 @@
$(window).on('xpboot', () => {
xp.applications.add('uac_app', () => {
var win = new Window({
width: 522,
height: 462,
title: 'User Accounts',
canResize: false,
});
win.content(`
<style>
.userimg {
width: 64px;
height: 64px;
}
.userimgopt {
border: 1px solid transparent;
width: 48px;
height: 48px;
padding: 4px;
border-radius: 5px;
}
.userimgopt.selected {
border: 1px solid #68b3db;
background-color: #e1f2fb;
}
.cp_option {
color: #fff;
}
</style>
<div style="width:100%;height:100%;background-color:#6375d6;" class="outercontent">
<div style="margin-left:32px;width:calc(100% - 64px);" class="content">
<br/>
<h1 style="color:#d6dff5">Pick a task...</h1>
<p>
<span class="cp_option pointer change_acct">Change an account</span>
</p>
<p>
<span class="cp_option pointer new_acct">Create a new account</span>
</p>
<h1 style="color:#d6dff5">or pick an account to change</h1>
<div style="width:100%overflow-x:auto;">
<table style="font-size:11px;color:#fff;">
<tr class="accounts">
</tr>
</table>
</center>
</div>
</div>`);
function changeAccount(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
var canDelete = '';
if (acct.name !== xp.profile.name) {
canDelete = `
<p>
<span style="color:#000;" class="cp_option pointer delete_account">Delete the account</span>
</p>`;
}
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">What do you want to change about ${acctname}<br/>account?</h1>
<p>
<span style="color:#000;" class="cp_option pointer change_name">Change ${acct.name === xp.profile.name ? 'my' : 'the'} name</span>
</p>
<p>
<span style="color:#000;" class="cp_option pointer change_picture">Change ${acct.name === xp.profile.name ? 'my' : 'the'} picture</span>
</p>${canDelete}
<table style="font-size:11px;color:#000;position:absolute;left:269px;top:97px">
<tr><td rowspan="4" style="width:64px;"><img class="userimg" style="width:48px;height:48px;" src="${acct.image}"/></td></tr>
<tr><td><b>${acct.name}</b></td></tr>
<tr><td>Computer administrator</td></tr>
<tr><td>&nbsp;</td></tr><tr><td>&nbsp;</td></tr>
</table>`);
win.el.find('.change_name').on('click', () => changeName(acct));
win.el.find('.change_picture').on('click', () => changePicture(acct));
win.el.find('.delete_account').on('click', () => deleteAccount(acct));
}
function deleteAccount(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Do you want to keep ${acctname} files?</h1>
<p>
Before you delete ${acctname} account, Windows can automatically save the contents of ${acctname}<br/>
My Documents to a folder called ${acct.name} on your desktop. However, Windows<br/>
cannot save ${acctname} e-mail messages, Internet favorites, and other settings.
</p>
<hr/>
<p>
<div style="float:right">
<button class="keep">Keep Files</button>
<button class="delete">Delete Files</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
win.el.find('.keep').on('click', () => {
xp.filesystem.moveDir('/Documents and Settings/' + acct.name + '/My Documents', '/Documents and Settings/' + xp.profile.name + '/My Documents/' + acct.name, () => {
xp.filesystem.deleteDir('/Documents and Settings/' + acct.name, () => {
win.close();
});
});
});
win.el.find('.delete').on('click', () => {
xp.filesystem.deleteDir('/Documents and Settings/' + acct.name, () => {
win.close();
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function changeName(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Provide a new name for ${acctname} account</h1>
<p>
Type a new name for ${acct.name}:<br/>
<input type="text" style="width:212px" class="username" value="${acct.name}"/><br/>
This name will appear on the Welcome screen and on the Start menu.<br/>
Note: Changing ${acctname} name will require a reboot.
</p>
<hr/>
<p>
<div style="float:right">
<button class="change">Change Name</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
win.el.find('.change').on('click', () => {
var oldProfile = xp.profile;
xp.profile = acct;
var newUsername = win.el.find('.username').val();
if (xp.profile.name !== newUsername) {
xp.filesystem.moveDir(`/Documents and Settings/${xp.profile.name}`, `/Documents and Settings/${newUsername}`, () => {});
xp.profile.name = newUsername;
}
saveConfig(() => {
xp.profile = oldProfile;
xp.alert('Changes have been applied.<br/>Click OK to reboot your computer.', () => {
window.location.href = window.location.href;
});
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function changePicture(acct) {
win.el.find('.outercontent').css('background-color', '#fff');
var acctname = acct.name === xp.profile.name ? 'your' : acct.name + '\'s';
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Pick a new picture for ${acctname} account</h1>
<p>
The picture you choose will appear on the Welcome screen.<br/>
Note: Changing your picture will require a reboot.
</p>
<p style="margin-left:16px;overflow-x:scroll;width:100%;">
<table style="font-size:11px;color:#000;">
<tr class="images">
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile2.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile3.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile4.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile5.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile6.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile7.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile8.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile9.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile10.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile11.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile12.bmp?1522560745448"/></td>
</tr>
</table>
</p>
<p>
<button class="browse">Browse for more pictures</button>
</p>
<hr/>
<p>
<div style="float:right">
<button class="change">Change Picture</button>
<button class="cancel">Cancel</button>
</div>
</p>`);
if (win.el.find(`.userimgopt[src="${acct.image}"]`).length > 0)
win.el.find(`.userimgopt[src="${acct.image}"]`).addClass('selected');
else
win.el.find('.images').append(`<td><img class="userimgopt selected" src="${acct.image}"/></td>`);
win.el.find('.userimgopt').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
});
win.el.find('.browse').on('click', () => {
xp.filesystem.openFileDialog((file) => {
xp.filesystem.toURL(file, (url) => {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
var el = $($.parseHTML(`<td><img class="userimgopt selected" src="${url}"/></td>`));
win.el.find('.images').append(el);
el.find('img').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
});
});
});
});
win.el.find('.change').on('click', () => {
var oldProfile = xp.profile;
xp.profile = acct;
xp.profile.image = win.el.find('.userimgopt.selected').attr('src');
saveConfig(() => {
xp.profile = oldProfile;
xp.alert('Changes have been applied.<br/>Click OK to reboot your computer.', () => {
window.location.href = window.location.href;
});
});
});
win.el.find('.cancel').on('click', () => changeAccount(acct));
}
function listAccounts() {
xp.filesystem.listDir('/Documents and Settings', (acct) => {
acct = acct.split('/')[0];
xp.filesystem.readFile(`/Documents and Settings/${acct}/config.json`, (file) => {
var config = JSON.parse(file);
var el = $.parseHTML(`<td>
<img class="userimgopt pointer" account="${acct}" src="${config.profile.image}"/>
<div style="text-align:center;" class="pointer">${acct}</div>
</td>`);
$(el).find('img').on('click', () => changeAccount(config.profile));
win.el.find('.accounts').append(el);
});
});
}
listAccounts();
win.el.find('.cp_option.new_acct').on('click', () => {
newAccount();
});
win.el.find('.cp_option.change_acct').on('click', () => {
win.el.find('.outercontent').css('background-color', '#fff');
win.el.find('.content').html(`
<br/>
<h1 style="color:#7294df">Pick an account to change</h1>
<div style="width:100%overflow-x:auto;">
<table style="font-size:11px;color:#000;">
<tr class="accounts">
</tr>
</table>
</center>`);
listAccounts();
});
function newAccount() {
let win = new Window({
width: 594,
height: 434,
title: 'Create New Account',
canResize: false,
center: true
});
win.content(`
<style>
.userimg {
width: 64px;
height: 64px;
}
.userimgopt {
border: 1px solid transparent;
width: 48px;
height: 48px;
padding: 4px;
border-radius: 5px;
}
.userimgopt.selected {
border: 1px solid #68b3db;
background-color: #e1f2fb;
}
</style>
<div style="margin-left:32px;">
<h1 style="font-weight:normal;">Create User Account</h1>
<h2 style="font-weight:normal;">Choose a user name and picture</h2>
<p>
Your user name and picture represent your user account. The account you create here is a computer<br>
administrator account. It is recommended to close all open applications before continuing.
</p>
<br/>
<p style="margin-left:64px;">
<table style="font-size:11px;color:#000;">
<tr><td rowspan="4" style="width:72px;"><img class="userimg" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td></tr>
<tr><td>Type a user name (for example, John)</td></tr>
<tr><td><input type="text" class="username" style="width:180px;"/></td></tr>
<tr><td>&nbsp;</td></tr>
</table>
</p>
<br/><br/>
<p style="margin-left:16px;">
<table style="font-size:11px;color:#000;">
<tr>
<td><img class="userimgopt selected" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fuser.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile2.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile3.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile4.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile5.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile6.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile7.bmp?1522560745448"/></td>
<td><img class="userimgopt" src="https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusertile8.bmp?1522560745448"/></td>
</tr>
</table>
</p>
<div style="position:absolute;right:8px;bottom:8px;">
<button disabled class="next">Next</button>
</div>
</div>
`);
win.el.find('.username').on('keyup', function() {
console.log($(this).val());
if ($(this).val() === '') {
win.el.find('.next').attr('disabled', 'disabled');
} else {
win.el.find('.next').removeAttr('disabled');
}
});
win.el.find('.next').on('click', function() {
if (win.el.find('.username').val() === '') {
xp.alert('You need to type a username in order to continue!');
} else {
xp.profile.image = win.el.find('.userimgopt.selected').attr('src');
xp.profile.name = win.el.find('.username').val();
var oldConfigFile = configFile;
configFile = `/Documents and Settings/${xp.profile.name}/config.json`;
xp.wallpaper.href = 'https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FBliss.jpg?1519950052202';
win.content(`
<div style="margin-left:32px;">
<h1 style="font-weight:normal;">Please wait while we create the user account.</h1>
</div>`);
setTimeout(() => {
requiredDirectories = [
`/Documents and Settings`,
`/Documents and Settings/${xp.profile.name}`,
`/Documents and Settings/${xp.profile.name}/My Documents`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Pictures`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Videos`,
`/Documents and Settings/${xp.profile.name}/My Documents/My Music`
];
var i = 0;
function createDirs() {
var dirToCreate = requiredDirectories[i];
if (dirToCreate !== undefined) {
xp.filesystem.createDir(dirToCreate, (e) => {
i ++;
createDirs();
});
} else {
saveConfig(() => {
configFile = oldConfigFile;
loadConfig(() => {
win.close();
});
});
}
}
createDirs();
}, 2000);
}
});
win.el.find('.userimgopt').on('click', function() {
win.el.find('.userimgopt').each(function() {
$(this).removeClass('selected');
});
$(this).addClass('selected');
win.el.find('.userimg').attr('src', $(this).attr('src'));
});
}
});
});

View File

@ -0,0 +1,613 @@
/**@license
* __ _____ ________ __
* / // _ /__ __ _____ ___ __ _/__ ___/__ ___ ______ __ __ __ ___ / /
* __ / // // // // // _ // _// // / / // _ // _// // // \/ // _ \/ /
* / / // // // // // ___// / / // / / // ___// / / / / // // /\ // // / /__
* \___//____ \\___//____//_/ _\_ / /_//____//_/ /_/ /_//_//_/ /_/ \__\_\___/
* \/ /____/
* http://terminal.jcubic.pl
*
* This is example of how to create custom formatter for jQuery Terminal
*
* Copyright (c) 2014-2018 Jakub Jankiewicz <http://jcubic.pl/me>
* Released under the MIT license
*
*/
/* global jQuery, define, global, require, module */
(function(factory) {
var root = typeof window !== 'undefined' ? window : global;
if (typeof define === 'function' && define.amd) {
// AMD. Register as an anonymous module.
// istanbul ignore next
define(['jquery', 'jquery.terminal'], factory);
} else if (typeof module === 'object' && module.exports) {
// Node/CommonJS
module.exports = function(root, jQuery) {
if (jQuery === undefined) {
// require('jQuery') returns a factory that requires window to
// build a jQuery instance, we normalize how we use modules
// that require this pattern but the window provided is a noop
// if it's defined (how jquery works)
if (typeof window !== 'undefined') {
jQuery = require('jquery');
} else {
jQuery = require('jquery')(root);
}
}
if (!jQuery.fn.terminal) {
if (typeof window !== 'undefined') {
require('jquery.terminal');
} else {
require('jquery.terminal')(jQuery);
}
}
factory(jQuery);
return jQuery;
};
} else {
// Browser
// istanbul ignore next
factory(root.jQuery);
}
})(function($) {
$.terminal.defaults.unixFormattingEscapeBrackets = false;
// we match characters and html entities because command line escape brackets
// echo don't, when writing formatter always process html entitites so it work
// for cmd plugin as well for echo
var chr = '[^\\x08]|[\\r\\n]{2}|&[^;]+;';
var backspace_re = new RegExp('^(' + chr + ')?\\x08');
var overtyping_re = new RegExp('^(?:(' + chr + ')?\\x08(_|\\1)|' +
'(_)\\x08(' + chr + '))');
var new_line_re = /^(\r\n|\n\r|\r|\n)/;
var clear_line_re = /[^\r\n]+\r\x1B\[K/g;
// ---------------------------------------------------------------------
function length(string) {
return $.terminal.length(string);
}
// ---------------------------------------------------------------------
// :: Replace overtyping (from man) formatting with terminal formatting
// ---------------------------------------------------------------------
$.terminal.overtyping = function overtyping(string, options) {
string = $.terminal.unescape_brackets(string);
var settings = $.extend({
unixFormattingEscapeBrackets: false,
position: 0
}, options);
var removed_chars = [];
var new_position;
var char_count = 0;
var backspaces = [];
function replace(string, position) {
var result = '';
var push = 0;
var start;
char_count = 0;
function correct_position(start, match, rep_string) {
// logic taken from $.terminal.tracking_replace
if (start < position) {
var last_index = start + length(match);
if (last_index < position) {
// It's after the replacement, move it
new_position = Math.max(
0,
new_position +
length(rep_string) -
length(match)
);
} else {
// It's *in* the replacement, put it just after
new_position += length(rep_string) - (position - start);
}
}
}
for (var i = 0; i < string.length; ++i) {
var partial = string.substring(i);
var match = partial.match(backspace_re);
var removed_char = removed_chars[0];
if (match) {
// we remove backspace and character or html entity before it
// but we keep it in removed array so we can put it back
// when we have caritage return or line feed
if (match[1]) {
start = i - match[1].length + push;
removed_chars.push({
index: start,
string: match[1],
overtyping: partial.match(overtyping_re)
});
correct_position(start, match[0], '', 1);
}
if (char_count < 0) {
char_count = 0;
}
backspaces = backspaces.map(function(b) {
return b - 1;
});
backspaces.push(start);
return result + partial.replace(backspace_re, '');
} else if (partial.match(new_line_re)) {
// if newline we need to add at the end all characters
// removed by backspace but only if there are no more
// other characters than backspaces added between
// backspaces and newline
if (removed_chars.length) {
var chars = removed_chars;
removed_chars = [];
chars.reverse().forEach(function(char) {
if (i > char.index) {
if (--char_count <= 0) {
correct_position(char.index, '', char.string, 2);
result += char.string;
}
} else {
removed_chars.unshift(char);
}
});
}
var m = partial.match(new_line_re);
result += m[1];
i += m[1].length - 1;
} else {
if (backspaces.length) {
var backspace = backspaces[0];
if (i === backspace) {
backspaces.shift();
}
if (i >= backspace) {
char_count++;
}
}
if (removed_chars.length) {
// if we are in index of removed character we check if the
// character is the same it will be bold or if removed char
// or char at index is underscore then it will
// be terminal formatting with underscore
if (i > removed_char.index && removed_char.overtyping) {
removed_chars.shift();
correct_position(removed_char.index, '', removed_char.string);
// if we add special character we need to correct
// next push to removed_char array
push++;
// we use special characters instead of terminal
// formatting so it's easier to proccess when removing
// backspaces
if (removed_char.string === string[i]) {
result += string[i] + '\uFFF1';
continue;
} else if (removed_char.string === '_' ||
string[i] === '_') {
var chr;
if (removed_char.string === '_') {
chr = string[i];
} else {
chr = removed_char.string;
}
result += chr + '\uFFF2';
continue;
}
}
}
result += string[i];
}
}
return result;
}
var break_next = false;
// loop until not more backspaces
new_position = settings.position;
// we need to clear line \x1b[K in overtyping because it need to be before
// overtyping and from_ansi need to be called after so it escape stuff
// between Escape Code and cmd will have escaped formatting typed by user
var rep = $.terminal.tracking_replace(string, clear_line_re, '', new_position);
string = rep[0];
new_position = rep[1];
while (string.match(/\x08/) || removed_chars.length) {
string = replace(string, new_position);
if (break_next) {
break;
}
if (!string.match(/\x08/)) {
// we break the loop so if removed_chars still chave items
// we don't have infite loop
break_next = true;
}
}
function format(string, chr, style) {
var re = new RegExp('((:?.' + chr + ')+)', 'g');
return string.replace(re, function(_, string) {
var re = new RegExp(chr, 'g');
return '[[' + style + ']' + string.replace(re, '') + ']';
});
}
// replace special characters with terminal formatting
string = format(string, '\uFFF1', 'b;#fff;');
string = format(string, '\uFFF2', 'u;;');
if (settings.unixFormattingEscapeBrackets) {
string = $.terminal.escape_brackets(string);
}
if (options && typeof options.position === 'number') {
return [string, new_position];
}
return string;
};
// ---------------------------------------------------------------------
// :: Html colors taken from ANSI formatting in Linux Terminal
// ---------------------------------------------------------------------
$.terminal.ansi_colors = {
normal: {
black: '#000',
red: '#A00',
green: '#008400',
yellow: '#A50',
blue: '#00A',
magenta: '#A0A',
cyan: '#0AA',
white: '#AAA'
},
faited: {
black: '#000',
red: '#640000',
green: '#006100',
yellow: '#737300',
blue: '#000087',
magenta: '#650065',
cyan: '#008787',
white: '#818181'
},
bold: {
black: '#444',
red: '#F55',
green: '#44D544',
yellow: '#FF5',
blue: '#55F',
magenta: '#F5F',
cyan: '#5FF',
white: '#FFF'
},
// XTerm 8-bit pallete
palette: [
'#000000', '#AA0000', '#00AA00', '#AA5500', '#0000AA', '#AA00AA',
'#00AAAA', '#AAAAAA', '#555555', '#FF5555', '#55FF55', '#FFFF55',
'#5555FF', '#FF55FF', '#55FFFF', '#FFFFFF', '#000000', '#00005F',
'#000087', '#0000AF', '#0000D7', '#0000FF', '#005F00', '#005F5F',
'#005F87', '#005FAF', '#005FD7', '#005FFF', '#008700', '#00875F',
'#008787', '#0087AF', '#0087D7', '#0087FF', '#00AF00', '#00AF5F',
'#00AF87', '#00AFAF', '#00AFD7', '#00AFFF', '#00D700', '#00D75F',
'#00D787', '#00D7AF', '#00D7D7', '#00D7FF', '#00FF00', '#00FF5F',
'#00FF87', '#00FFAF', '#00FFD7', '#00FFFF', '#5F0000', '#5F005F',
'#5F0087', '#5F00AF', '#5F00D7', '#5F00FF', '#5F5F00', '#5F5F5F',
'#5F5F87', '#5F5FAF', '#5F5FD7', '#5F5FFF', '#5F8700', '#5F875F',
'#5F8787', '#5F87AF', '#5F87D7', '#5F87FF', '#5FAF00', '#5FAF5F',
'#5FAF87', '#5FAFAF', '#5FAFD7', '#5FAFFF', '#5FD700', '#5FD75F',
'#5FD787', '#5FD7AF', '#5FD7D7', '#5FD7FF', '#5FFF00', '#5FFF5F',
'#5FFF87', '#5FFFAF', '#5FFFD7', '#5FFFFF', '#870000', '#87005F',
'#870087', '#8700AF', '#8700D7', '#8700FF', '#875F00', '#875F5F',
'#875F87', '#875FAF', '#875FD7', '#875FFF', '#878700', '#87875F',
'#878787', '#8787AF', '#8787D7', '#8787FF', '#87AF00', '#87AF5F',
'#87AF87', '#87AFAF', '#87AFD7', '#87AFFF', '#87D700', '#87D75F',
'#87D787', '#87D7AF', '#87D7D7', '#87D7FF', '#87FF00', '#87FF5F',
'#87FF87', '#87FFAF', '#87FFD7', '#87FFFF', '#AF0000', '#AF005F',
'#AF0087', '#AF00AF', '#AF00D7', '#AF00FF', '#AF5F00', '#AF5F5F',
'#AF5F87', '#AF5FAF', '#AF5FD7', '#AF5FFF', '#AF8700', '#AF875F',
'#AF8787', '#AF87AF', '#AF87D7', '#AF87FF', '#AFAF00', '#AFAF5F',
'#AFAF87', '#AFAFAF', '#AFAFD7', '#AFAFFF', '#AFD700', '#AFD75F',
'#AFD787', '#AFD7AF', '#AFD7D7', '#AFD7FF', '#AFFF00', '#AFFF5F',
'#AFFF87', '#AFFFAF', '#AFFFD7', '#AFFFFF', '#D70000', '#D7005F',
'#D70087', '#D700AF', '#D700D7', '#D700FF', '#D75F00', '#D75F5F',
'#D75F87', '#D75FAF', '#D75FD7', '#D75FFF', '#D78700', '#D7875F',
'#D78787', '#D787AF', '#D787D7', '#D787FF', '#D7AF00', '#D7AF5F',
'#D7AF87', '#D7AFAF', '#D7AFD7', '#D7AFFF', '#D7D700', '#D7D75F',
'#D7D787', '#D7D7AF', '#D7D7D7', '#D7D7FF', '#D7FF00', '#D7FF5F',
'#D7FF87', '#D7FFAF', '#D7FFD7', '#D7FFFF', '#FF0000', '#FF005F',
'#FF0087', '#FF00AF', '#FF00D7', '#FF00FF', '#FF5F00', '#FF5F5F',
'#FF5F87', '#FF5FAF', '#FF5FD7', '#FF5FFF', '#FF8700', '#FF875F',
'#FF8787', '#FF87AF', '#FF87D7', '#FF87FF', '#FFAF00', '#FFAF5F',
'#FFAF87', '#FFAFAF', '#FFAFD7', '#FFAFFF', '#FFD700', '#FFD75F',
'#FFD787', '#FFD7AF', '#FFD7D7', '#FFD7FF', '#FFFF00', '#FFFF5F',
'#FFFF87', '#FFFFAF', '#FFFFD7', '#FFFFFF', '#080808', '#121212',
'#1C1C1C', '#262626', '#303030', '#3A3A3A', '#444444', '#4E4E4E',
'#585858', '#626262', '#6C6C6C', '#767676', '#808080', '#8A8A8A',
'#949494', '#9E9E9E', '#A8A8A8', '#B2B2B2', '#BCBCBC', '#C6C6C6',
'#D0D0D0', '#DADADA', '#E4E4E4', '#EEEEEE'
]
};
// ---------------------------------------------------------------------
// :: Replace ANSI formatting with terminal formatting
// ---------------------------------------------------------------------
$.terminal.from_ansi = (function() {
var color_list = {
30: 'black',
31: 'red',
32: 'green',
33: 'yellow',
34: 'blue',
35: 'magenta',
36: 'cyan',
37: 'white',
39: 'inherit' // default color
};
var background_list = {
40: 'black',
41: 'red',
42: 'green',
43: 'yellow',
44: 'blue',
45: 'magenta',
46: 'cyan',
47: 'white',
49: 'transparent' // default background
};
function format_ansi(code, state) {
var controls = code.split(';');
var num;
var styles = [];
var output_color = '';
var output_background = '';
var _process_true_color = -1;
var _ex_color = false;
var _ex_background = false;
var _process_8bit = false;
var palette = $.terminal.ansi_colors.palette;
function set_styles(num) {
switch (num) {
case 0:
Object.keys(state).forEach(function(key) {
delete state[key];
});
break;
case 1:
styles.push('b');
state.bold = true;
state.faited = false;
break;
case 4:
styles.push('u');
break;
case 3:
styles.push('i');
break;
case 5:
if (_ex_color || _ex_background) {
_process_8bit = true;
}
break;
case 38:
_ex_color = true;
break;
case 48:
_ex_background = true;
break;
case 2:
if (_ex_color || _ex_background) {
_process_true_color = 0;
} else {
state.faited = true;
state.bold = false;
}
break;
case 7:
state.reverse = true;
break;
default:
if (controls.indexOf('5') === -1) {
if (color_list[num]) {
output_color = color_list[num];
}
if (background_list[num]) {
output_background = background_list[num];
}
}
}
}
// -----------------------------------------------------------------
function process_true_color() {
if (_ex_color) {
if (!output_color) {
output_color = '#';
}
if (output_color.length < 7) {
output_color += ('0' + num.toString(16)).slice(-2);
}
}
if (_ex_background) {
if (!output_background) {
output_background = '#';
}
if (output_background.length < 7) {
output_background += ('0' + num.toString(16)).slice(-2);
}
}
if (_process_true_color === 2) {
_process_true_color = -1;
} else {
_process_true_color++;
}
}
// -----------------------------------------------------------------
function should__process_8bit() {
return _process_8bit && ((_ex_background && !output_background) ||
(_ex_color && !output_color));
}
// -----------------------------------------------------------------
function process_8bit() {
if (_ex_color && palette[num] && !output_color) {
output_color = palette[num];
}
if (_ex_background && palette[num] && !output_background) {
output_background = palette[num];
}
_process_8bit = false;
}
// -----------------------------------------------------------------
for (var i in controls) {
if (controls.hasOwnProperty(i)) {
num = parseInt(controls[i], 10);
if (_process_true_color > -1) {
process_true_color();
} else if (should__process_8bit()) {
process_8bit();
} else {
set_styles(num);
}
}
}
if (state.reverse) {
if (output_color || output_background) {
var tmp = output_background;
output_background = output_color;
output_color = tmp;
} else {
output_color = 'black';
output_background = 'white';
}
}
output_color = output_color || state.color;
output_background = output_background || state.background;
state.background = output_background;
state.color = output_color;
var colors, color, background;
if (state.bold) {
colors = $.terminal.ansi_colors.bold;
} else if (state.faited) {
colors = $.terminal.ansi_colors.faited;
} else {
colors = $.terminal.ansi_colors.normal;
}
if (typeof output_color !== 'undefined') {
if (_ex_color) {
color = output_color;
} else if (output_color === 'inherit') {
color = output_color;
} else {
color = colors[output_color];
}
}
if (typeof output_background !== 'undefined') {
if (_ex_background) {
background = output_background;
} else if (output_background === 'transparent') {
background = output_background;
} else {
background = $.terminal.ansi_colors.normal[output_background];
}
}
var ret = [styles.join(''), color, background];
return ret;
}
return function from_ansi(input, options) {
options = options || {};
input = $.terminal.unescape_brackets(input);
var settings = $.extend({
unixFormattingEscapeBrackets: false,
position: 0
}, options);
var state = {}; // used to inherit vales from previous formatting
var new_position = settings.position;
var position = new_position;
var result;
var ansi_re = /(\x1B\[[0-9;]*[A-Za-z])/g;
var cursor_re = /(.*)\r?\n\x1b\[1A\x1b\[([0-9]+)C/g;
// move up and right we need to delete what's after in previous line
input = input.replace(cursor_re, function(_, line, n) {
n = parseInt(n, 10);
var parts = line.split(ansi_re).filter(Boolean);
var result = [];
for (var i = 0; i < parts.length; ++i) {
if (parts[i].match(ansi_re)) {
result.push(parts[i]);
} else {
var len = parts[i].length;
if (len > n) {
result.push(parts[i].substring(0, n));
break;
} else {
result.push(parts[i]);
}
n -= len;
}
}
return result.join('');
});
// move right is just repate space
input = input.replace(/\x1b\[([0-9]+)C/g, function(_, num) {
return new Array(+num + 1).join(' ');
});
var splitted = input.split(ansi_re);
if (splitted.length === 1) {
if (settings.unixFormattingEscapeBrackets) {
result = $.terminal.escape_formatting(input);
} else {
result = input;
}
if (typeof options.position === 'number') {
return [result, new_position];
}
return result;
}
var output = [];
//skip closing at the begining
if (splitted.length > 3) {
var str = splitted.slice(0, 3).join('');
if (str.match(/^\[0*m$/)) {
splitted = splitted.slice(3);
}
}
var code, match;
var inside = false;
for (var i = 0; i < splitted.length; ++i) {
match = splitted[i].match(/^\x1B\[([0-9;]*)([A-Za-z])$/);
if (match) {
var start = match.index;
// logic taken from $.terminal.tracking_replace
if (start < position) {
var last_index = start + $.terminal.length(match[0]);
if (last_index < position) {
// It's after the replacement, move it
new_position = Math.max(
0,
new_position +
$.terminal.length(match[0])
);
} else {
// It's *in* the replacement, put it just after
new_position += position - start;
}
}
switch (match[2]) {
case 'm':
code = format_ansi(match[1], state);
if (inside) {
output.push(']');
if (+match[1] === 0) {
inside = false;
} else {
output.push('[[' + code.join(';') + ']');
}
} else if (+match[1] !== 0) {
output.push('[[' + code.join(';') + ']');
inside = true;
}
break;
}
} else if (settings.unixFormattingEscapeBrackets) {
output.push($.terminal.escape_formatting(splitted[i]));
} else {
output.push(splitted[i]);
}
}
if (inside) {
output.push(']');
}
result = output.join('');
if (options && typeof options.position === 'number') {
return [result, new_position];
}
return result;
};
})();
$.terminal.from_ansi.__meta__ = true;
$.terminal.defaults.formatters.unshift($.terminal.from_ansi);
$.terminal.defaults.formatters.unshift($.terminal.overtyping);
});

108
xp/system/widgets.css Normal file
View File

@ -0,0 +1,108 @@
button {
font-family: "Tahoma";
font-size: 11px;
min-width: 67px;
position: relative;
height: 23px;
margin: 0 4px;
border: 0;
background: transparent;
background-image: url();
}
button::before,
button::after {
content: ' ';
position: absolute;
top: 0;
width: 4px;
height: 23px;
background: transparent;
}
button::before {
left: -4px;
background-image: url();
}
button::after {
right: -4px;
background-image: url();
}
button:active {
background-image: url() !important;
}
button:active::before {
background-image: url() !important;
}
button:active::after {
background-image: url() !important;
}
button:focus {
outline: none;
background-image: url();
}
button:focus::before {
background-image: url();
}
button:focus::after {
background-image: url();
}
button:hover {
background-image: url();
}
button:hover::before {
background-image: url();
}
button:hover::after {
background-image: url();
}
input[type=button], input[type=submit] {
border: 2px solid transparent;
border-image: url('') 2 round;
border-image-width: 2px;
border-width: 2px;
background-color: #d6d6ce;
font-family: Tahoma;
font-size: 11px;
}
input[type=button]:active, input[type=submit]:active {
border-image: url('') 2 round;
}
input[type=button]:focus, input[type=submit]:focus {
outline: none;
}
textarea, input[type=text], input[type=number] {
color: #000;
border: 1px solid #7f9db9;
}
textarea {
font-family: monospace;
font-weight: normal;
font-style: normal;
font-size: 11px;
}
input[type=text], input[type=number] {
font-family: 'Tahoma';
font-weight: normal;
font-style: normal;
font-size: 11px;
}
input[type=text], input[type=number] {
padding-left: 2px;
}
textarea:focus, input[type=text]:focus, input[type=number]:focus {
outline: none;
}
select {
font-size: 11px;
background: #ffffff;
font-family: Tahoma;
border: 1px solid #7f9db9;
}

88
xp/system/window-api.js Normal file
View File

@ -0,0 +1,88 @@
function Window(args) {
var htmlargs = '';
args = args || [];
htmlargs += 'width="' + (args.width || 480) + '"';
htmlargs += 'height="' + (args.height || 360) + '"';
htmlargs += 'title="' + (args.title ||'Unnamed Window') + '"';
this.el = $($.parseHTML('<window ' + htmlargs + '></window>'));
document.body.append(this.el[0]);
this.el.updateWindow();
if (args.canClose === false) {
this.el.find('.close').remove();
}
if (args.canMinimize === false) {
this.el.find('.minimize').remove();
}
if (args.canResize === false) {
this.el.find('.maximize').remove();
this.el.find('.resize').each(function() {
$(this).remove();
});
}
if (args.center) {
this.el.css('left', window.innerWidth / 2 - this.el.width() / 2);
this.el.css('top', window.innerHeight / 2 - this.el.height() / 2);
}
}
Window.prototype.on = function(event, callback) {
this.el.on(event, callback);
console.log(this.el[0]);
return this;
}
Window.prototype.close = function() {
closeWindow(this.el.attr('guid'));
return this;
}
Window.prototype.minimize = function() {
minimizeWindow(this.el.attr('guid'));
return this;
}
Window.prototype.maximize = function() {
maximizeWindow(this.el.attr('guid'));
return this;
}
Window.prototype.restore = function() {
if (this.el.attr('maximized') === 'true')
maximizeWindow(this.el.attr('guid'));
if (this.el.attr('minimized') === 'true')
minimizeWindow(this.el.attr('guid'));
return this;
}
Window.prototype.content = function(content) {
if (content === undefined) {
return this.el.find('.windowBody').html();
} else {
this.el.find('.windowBody').html(content);
return this;
}
}
Window.prototype.title = function(title) {
if (title === undefined) {
return this.el.attr('title');
} else {
this.el.setTitle(title);
return this;
}
}
Window.prototype.size = function(size) {
if (size === undefined) {
return [parseInt(this.el.attr('width')), parseInt(this.el.attr('height'))];
} else {
this.el.attr('width', size[0]);
this.el.attr('height', size[1]);
this.el.updateWindow();
return this;
}
}
Window.prototype.addMenu = function(name, items) {
this.el.addMenu(name, items);
}

305
xp/system/window.css Normal file
View File

@ -0,0 +1,305 @@
window {
display: none;
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
position: absolute;
}
window[guid] {
display: inline;
}
window[minimized=true] {
display: none;
}
.minimize::before, .maximize::before, .restore::before, .close::before,
.minimize::after, .maximize::after, .restore::after, .close::after {
width: 0;
margin: 0;
padding: 0;
}
.titleBar_left {
background: transparent url() no-repeat scroll 0;
height: 30px;
width: 6px;
padding: 0;
}
window[inactive=true] .titleBar_left {
background-image: url();
}
.titleBar_middle {
background: transparent url() repeat-x scroll 0;
height: 30px;
padding: 0;
}
window[inactive=true] .titleBar_middle {
background-image: url();
}
.titleBar_right {
background: transparent url() no-repeat scroll 0;
height: 30px;
width: 6px;
padding: 0;
}
window[inactive=true] .titleBar_right {
background-image: url();
}
.close, .maximize, .minimize, .restore {
height: 21px;
width: 21px;
float: right;
padding: 0;
border: none;
position: relative;
top: 2px;
right: 0px;
min-width: 0;
border-image: none;
margin: 0;
background: transparent url() no-repeat scroll 0;
}
.close, .close:focus {
background-image: url();
}
window[inactive=true] .close, window[inactive=true] .close:focus {
background-image: url();
}
.close:hover {
background-image: url() !important;
}
.close:active {
background-image: url() !important;
}
.maximize, .maximize:focus {
background-image: url();
right: 2px;
}
window[inactive=true] .maximize, window[inactive=true] .maximize:focus {
background-image: url();
}
.maximize:hover {
background-image: url() !important;
}
.maximize:active {
background-image: url() !important;
}
.minimize, .minimize:focus {
background-image: url();
right: 4px;
}
window[inactive=true] .minimize, window[inactive=true] .minimize:focus {
background-image: url();
}
.minimize:hover {
background-image: url() !important;
}
.minimize:active {
background-image: url() !important;
}
.restore, .restore:focus {
background-image: url();
right: 2px;
}
window[inactive=true] .restore, window[inactive=true] .restore:focus {
background-image: url();
}
.restore:hover {
background-image: url() !important;
}
.restore:active {
background-image: url() !important;
}
.resize {
position: absolute;
}
.resize-se {
cursor: nwse-resize;
bottom: 0;
right: 0;
width: 5px;
height: 5px;
}
.resize-sw {
cursor: nesw-resize;
bottom: 0;
left: 0;
width: 5px;
height: 5px;
}
.resize-ne {
cursor: nesw-resize;
top: 0;
right: 0;
width: 5px;
height: 5px;
}
.resize-nw {
cursor: nwse-resize;
top: 0;
left: 0;
width: 5px;
height: 5px;
}
.resize-n {
cursor: ns-resize;
top: 0;
right: 5px;
width: calc(100% - 10px);
height: 5px;
}
.resize-s {
cursor: ns-resize;
bottom: 0;
right: 5px;
width: calc(100% - 10px);
height: 5px;
}
.resize-e {
cursor: ew-resize;
bottom: 5px;
right: 0;
width: 5px;
height: calc(100% - 10px);
}
.resize-w {
cursor: ew-resize;
bottom: 5px;
left: 0;
width: 5px;
height: calc(100% - 10px);
}
.windowBody {
background-color: #f3f4f5;
color: #000;
margin: 0;
padding: 0;
overflow: hidden;
border: 4px solid transparent;
border-top-width: 0px;
border-image: url() 4 round;
border-image-width: 0 4px 4px 4px;
position: relative;
}
window[inactive=true] .windowBody {
border-top-width: 0px;
border-image: url() 4 round;
}
.windowTitleBg {
width: 100%;
height: 30px;
line-height: 24px;
}
.windowTitleBg.img {
width: 16px;
height: 16px;
float: left;
margin-top: 7px;
}
.windowTitle {
float: left;
padding: 4px 4px 2px 2px;
white-space: nowrap;
color: #ffffff;
font-family: 'Trebuchet MS Bold';
font-weight: normal;
font-style: normal;
font-size: 13px;
position: relative;
top: 0px;
left: 2px;
text-shadow: 1px 1px rgba(0, 0, 0, 0.5);
}
.windowTitle.inactive {
text-shadow: none;
color: #d8e4f8;
}
windowbutton {
border: 0;
max-width: 140px;
min-width: 140px;
width: 140px;
height: 25px;
display: inline-block;
background: url() repeat-x scroll 0;
position: relative;
top: 0px;
padding-left: 4px;
line-height: 25px;
margin-right: 4px;
margin: 0 10px;
color: #FFFFFF;
}
windowbutton * {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
width: 140px;
vertical-align: top;
}
windowbutton::before,
windowbutton::after {
content: ' ';
position: absolute;
top: 0;
width: 10px;
height: 25px;
}
windowbutton::before {
left: -10px;
background-image: url();
}
windowbutton::after {
right: -10px;
background-image: url();
}
windowbutton[inactive=true] {
background-image: url();
}
windowbutton[inactive=true]::before {
background-image: url();
}
windowbutton[inactive=true]::after {
background-image: url();
}
windowbutton[inactive=true]:hover {
background-image: url();
}
windowbutton[inactive=true]:hover::before {
background-image: url();
}
windowbutton[inactive=true]:hover::after {
background-image: url();
}
windowbutton[inactive=false]:hover {
background-image: url();
}
windowbutton[inactive=false]:hover::before {
background-image: url();
}
windowbutton[inactive=false]:hover::after {
background-image: url();
}

369
xp/system/xp.css Normal file
View File

@ -0,0 +1,369 @@
/* Styles */
body {
font-family: 'Tahoma';
font-weight: normal;
font-style: normal;
font-size: 11;
overflow: hidden;
margin: 0px;
}
.scroll {
overflow: scroll;
}
pre {
font-family: monospace;
}
a {
font-family: 'Tahoma';
font-size: 11px;
}
windows {
position: relative;
top: 0px;
bottom: 0px;
width: 100%;
height: 100%;
display: block;
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
._ui_ellipsis_overflow {
text-overflow: ellipsis;
white-space: nowrap;
overflow: hidden;
}
taskbar {
background-color: #d6d6ce;
width: 100%;
position: fixed;
left: 0;
bottom: 0;
height: 30px;
background-image: url();
z-index: 16777214;
}
taskbar * {
vertical-align: middle;
}
taskbar separator {
width: 2px;
height: 22px;
display: inline-block;
background: url('') repeat scroll 0;
}
.bevelinset {
border: 2px solid transparent;
border-image: url('') 2 round;
border-image-width: 2px;
background-color: #FFFFFF;
}
.vertical-text {
writing-mode: tb-rl;
transform: rotate(-180deg);
float: left;
color: #FFFFFF;
font-family: Arial;
font-weight: bold;
font-style: normal;
font-size: 16px;
position: absolute;
bottom: 0px;
left: 0px;
}
._ui_grad_btb {
background: linear-gradient(0deg, #0000FF 0, #000000 100%);
width: 17px;
}
ul.menu {
list-style-type: none;
font-family: 'Tahoma';
font-weight: normal;
font-style: normal;
font-size: 11;
padding: 0px;
margin: 0;
position: relative;
bottom: 0;
}
li.menuitem {
color: #000000;
padding-left: 4px;
padding-right: 12px;
padding-top: 2px;
height: 18px;
min-width: 134px;
}
li.menuitem:hover {
background-color: #316ac5;
color: #FFFFFF;
}
li.menuitem[data-selected=true] {
background-color: #316ac5;
color: #FFFFFF;
}
td._ui_td {
font-size: 11;
}
.ui_terminal {
background-color: #000;
color: #c3ff00;
font-family: "Tomo",Ubuntu Mono,Consolas,Andale Mono WT,Andale Mono,Lucida Console,Lucida Sans Typewriter,DejaVu Sans Mono,Bitstream Vera Sans Mono,Liberation Mono,Nimbus Mono L,Monaco,Menlo,source-code-pro,Courier New,Courier,monospace;
font-size: 8px;
white-space: pre-wrap;
word-break: break-word;
word-wrap: break-word;
padding: 5px;
overflow-x: hidden;
overflow-y: scroll;
width: calc(100% - 10px);
height: calc(100% - 10px);
}
.ui_terminal, .ui_terminal * {
font-family: "Tomo",monospace;
font-size: 8px;
line-height: 1;
color:#c3ff00;
}
.fillspace, .fullscreen {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
width: 100%;
height: 100%;
display: block;
margin: 0;
}
.noscroll {
overflow: hidden!important;
}
._ui_postlogo {
background: url('') no-repeat scroll 0;
width: 278px;
height: 116px;
position: relative;
top: 5px;
left: 5px;
}
._ui_wallpaper {
background: linear-gradient(135deg, #316ac5 0, #a5c6f7 100%);
z-index: -2;
}
._ui_wallpaper_image {
width:100%;
height:100%;
z-index: -2;
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
.hidden {
width: 0px;
height: 0;
border: none!important;
padding: 0;
}
._ui_menu {
padding: 1px;
height: 16px;
display: inline-block;
padding-top: 2px;
}
._ui_menu_item {
height: 15px!important;
line-height: 15px;
padding-left: 5px;
padding-right: 5px;
padding-top: 1px;
padding-bottom: 2px;
color: #000000;
}
._ui_menu_item_active {
background-color: #316ac5;
color: #FFFFFF;
}
.beveledmenu {
position: absolute;
border: 2px solid transparent;
border-image: url('') 2 round;
border-image-width: 2px;
border-width: 2px;
background-color: #d6d6ce;
z-index: 16777215;
width: 150px;
}
.ContextMenu {
position: absolute;
top: 0;
left: 0;
cursor: default;
visibility: hidden;
z-index: 2147483647;
/*box-shadow: 0 0 25px 1px rgba(0,0,0,.5);*/
}
.startmenuitem {
padding: 4px;
width: 178px;
height: 24px;
color: #0a246a;
}
.startmenuitem.large {
width: 172px;
height: 32px;
padding: 8px;
color: #373738;
}
.startmenuitem:hover {
background-color: #316ac5;
color: #FFFFFF;
}
.startmenuitem[data-selected=true] {
background-color: #316ac5;
color: #FFFFFF;
}
.startmenuitem div {
vertical-align: top;
width: calc(100% - 44px);
display: inline-block;
padding-left: 4px;
height: 100%;
}
.startmenuitem.large div {
padding-left: 8px;
}
.startmenuitem div div {
position: relative;
top: 50%;
transform: translateY(-50%);
padding: 0;
height: auto;
width: 100%;
}
.startmenuitem.large div div {
padding: 0;
}
.startmenuitem:hover div.startmenuitem {
background-color: transparent;
}
.startmenuitem.expand {
background-image: url();
}
.startmenuitem.expand:hover {
background-image: url();
}
.startmenuitem img {
width: 24px;
height: 24px;
}
.startmenuitem.large img {
width: 32px;
height: 32px;
}
hr {
border-top-color: #848484;
border-bottom-color: #FFFFFF;
border-style: solid;
border-left-color: transparent;
border-right-color: transparent;
}
.clock {
float: right;
height: 30px;
line-height: 30px;
text-align: right;
padding-right: 10px;
padding-left: 12px;
}
.clock::before {
content: ' ';
width: 30px;
position: absolute;
top: 0;
height: 30px;
background: transparent;
background-image: url();
left: calc(100% - 68px);
z-index: -1;
}
.clock::after {
content: '';
z-index: -2;
position: absolute;
top: 0;
right: 0;
background-image: url();
height: inherit;
width: inherit;
min-width: 38px;
}
.bevel {
border: 1px solid;
border-bottom-color: #FFFFFF;
border-right-color: #FFFFFF;
border-top-color: #848484;
border-left-color: #848484;
}
img.menuicon {
width: 16px;
height: 16px;
padding-right: 4px;
}
li.menuitem span.menuitemtext {
position: relative;
top: -4px;
}

482
xp/system/xp.js Normal file
View File

@ -0,0 +1,482 @@
'use strict';
xp.startmenu = {
icons: {
programs: '',
documents: '',
recentdocuments: '',
controlpanel: '',
search: '',
help: '',
run: '',
shutdown: '',
pictures: '',
music: '',
computer: ''
},
appIcons: {},
menuItems: {},
add: function(appName, menuItemTxt, appIcon) {
xp.startmenu.menuItems[appName] = menuItemTxt;
xp.startmenu.appIcons[appName] = appIcon || '';
xp.startmenu.update();
},
remove: function(appName) {
delete xp.startmenu.menuItems[appName];
xp.startmenu.update();
},
update: function() {
$("#_ui_programsmenu").find('ul').html('');
for (var key in xp.startmenu.menuItems) {
$("#_ui_programsmenu").find('ul').append('<li class="menuitem" onclick="openApp(\'' + key + '\')"><img class="menuicon" src="' + xp.startmenu.appIcons[key] + '"/><span class="menuitemtext">' + xp.startmenu.menuItems[key] + '</span></li>');
}
}
};
xp.controlpanel = {
items: {},
add: function(name, func) {
xp.controlpanel.items[name] = func;
},
remove: function(name) {
delete xp.controlpanel.items[name];
}
}
xp.profile = {
image: 'https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2Fusericon.png?1520129931353',
name: 'User'
};
jQuery.fn.extend({
getPath: function () {
var path, node = this;
while (node.length) {
var realNode = node[0], name = realNode.localName;
if (!name) break;
name = name.toLowerCase();
var parent = node.parent();
var sameTagSiblings = parent.children(name);
if (sameTagSiblings.length > 1) {
var allSiblings = parent.children();
var index = allSiblings.index(realNode) + 1;
if (index > 1) {
name += ':nth-child(' + index + ')';
}
}
path = name + (path ? '>' + path : '');
node = parent;
}
return path;
}
});
document.onclick = function (e) {
e = e || window.event;
var element = e.target || e.srcElement;
if (element.tagName == 'A') {
if ($(element).attr('download') !== undefined) return true; // allow browser to download file
if ($(element).attr('href') !== '' && element.href !== undefined) {
var temp = document.createElement('div');
temp.innerHTML = `<window title="` + element.href + `" width="640" height="480">
<style>
iframe[seamless]{
background-color: transparent;
border: 0px none transparent;
padding: 0px;
overflow: hidden;
}
.frame-container {
/*display: none;*/
position: absolute;
width: 100%;
height: 100%;
overflow: hidden;
padding: 0px;
margin: 0px;
}
</style>
<div class="frame-container">
<iframe seamless="seamless" width="100%" height="100%" id="frame" src="` + element.href + `"></iframe>
</div>
</window>`;
var el = temp.firstChild;
document.body.appendChild(el);
$(el).updateWindow();
}
return false; // prevent default action and stop event propagation
}
};
$(function() {
xp.applications.add('winver', function() {
var guid = generate_guid();
var temp = document.createElement('div');
temp.innerHTML = `<window title="About RebornXP" width="418" height="400">
<center>
<img src="boot/assets/logo.png" width="100" height="100">
<br>
<h2>Reborn XP</h2>
<p> Version ` + xp.version + `</p>
<p> Registered to: ` + xp.profile.name + `</p>
<p>Powered by <a href="https://jquery.com/" target="_blank">JQuery</a></p>
<p> © 2023 - Shoaib Khan<br>
RebornXP is <a href="https://github.com/shoaib-jamal/rebornxp" target="_blank">Open Source</a> under the GPL v3.0 License</p>
<br>
<a href="https://www.buymeacoffee.com/shoaibify" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 50px !important;width: 200px !important;" ></a>
<br><br>
<a href="https://opensource.org/license/gpl-3-0/"><img src="https://img.shields.io/badge/License-GPL-yellow.svg"></a>
<a href="https://github.com/shoaib-jamal/rebornxp/stargazers"><img src="https://img.shields.io/github/stars/shoaib-jamal/rebornxp?label=GitHub%20Stars&style=social"></a>
<a href="https://github.com/shoaib-jamal/rebornxp/network/members"><img src="https://img.shields.io/github/forks/shoaib-jamal/rebornxp?style=social"></a>
<br>
</center>
<center style="position:absolute;bottom:4.5px;right:4.5px;">
<button id="OKButton_` + guid + `">OK</button>
</center>
</window>`;
var el = temp.firstChild;
document.body.appendChild(el);
$(el).updateWindow();
$("#OKButton_" + guid).attr("onclick", "closeWindow('" + $(el).attr("guid") + "')");
});
});
window.onerror = function (msg, url, lineNo, columnNo, error) {
xp.dialog('Error', `An error has occurred!<br/>
&nbsp;&nbsp` + msg + `<br/>
&nbsp;&nbsp;&nbsp;&nbsp;in ` + url + ` line ` + lineNo + ` column ` + columnNo, () => {}, false, 'error');
};
function openApp(name, args) {
closeStartMenu();
if (xp.applications.apps[name] !== undefined) {
xp.applications.apps[name](args);
return true;
} else {
return false;
}
}
// Start menu control
function openStartMenu() {
$("taskbar div.startmenu").css("display", "inline");
$("taskbar button.start").attr("isopen", "true");
xp.audio.playURL('https://cdn.glitch.com/01d2e04f-e49d-4304-aa9e-55b9849b4cce%2FWindows%20XP%20Start.wav?1522621401979');
}
function closeStartMenu() {
$("taskbar div.startmenu").css("display", "none");
$("taskbar button.start").attr("isopen", "false");
$('#_ui_programsmenu').css('display', 'none');
$('#_ui_programsmenu').data('isopen', false);
}
function toggleStartMenu() {
var el = $("taskbar button.start");
if (el.attr("isopen") === "true")
closeStartMenu();
else
openStartMenu();
}
function handleTaskbarButton(guid) {
var el = $('window[guid=' + guid + ']');
if (el.attr("inactive") === "false")
minimizeWindow(guid);
else
moveWindowToTop(guid);
}
function updateTaskbar() {
var el = $("taskbar");
}
var width = 0;
var windowsEl;
function getClockTime() {
var date = new Date().getTime();
var d = new Date(date);
var hh = d.getHours();
var m = d.getMinutes();
var s = d.getSeconds();
var dd = "AM";
var h = hh;
if (h >= 12) {
h = hh - 12;
dd = "PM";
}
if (h == 0) {
h = 12;
}
m = m < 10 ? "0" + m : m;
s = s < 10 ? "0" + s : s;
var result = h + ":" + m + " " + dd;
document.getElementById('_ui_clock').innerHTML = result;
}
function updateClock() {
var interval = (60 - (new Date()).getSeconds()) * 1000 + 5;
getClockTime();
setTimeout(updateClock, interval);
}
$.fn.initWindows = function() {
windowsEl = this;
windowsEl.append(`
<div class="ui_desktop" id="_ui_desktop">
<div tabindex="0" data-exe="openLocation('My Computer')" data-target="My Computer" class="ui_icon" style="position:absolute;left:0px;top:0px;">
<div class="icon icon_computer"></div>
<span>My Computer</span>
</div>
</div>
<taskbar id="_ui_taskbar" class="shadow" onmousedown="return false;" oncontextmenu="return false;">
<button class="start" onmousedown="toggleStartMenu()" isopen="false">Start</button>
<div class="startmenu shadow" style="display: none">
<div class="startmenu_top">
<img class="startmenu_usericon" src="` + xp.profile.image + `"/>
<div class="startmenu_username">` + xp.profile.name + `</div>
</div>
<div class="startmenu_middle">
<ul class="menu startmenu_programs">
<li class="startmenuitem large" onclick="openApp('browser')">
<img src="https://www.freeiconspng.com/uploads/internet-explorer-logo-icon-10.png"/>
<div><div>
<b>Internet</b><br/>
Web Browser
</div></div>
</li>
<li class="startmenuitem large" onclick="openApp('terminal')">
<img src=""/>
<div><div>
<b>Shell</b><br/>
JQuery Terminal
</div></div>
</li>
<li class="startmenuitem large" onclick="window.open('https://www.buymeacoffee.com/shoaibify', '_blank');">
<img src="https://media.giphy.com/media/TDQOtnWgsBx99cNoyH/giphy.gif"/>
<div><div>
<b>Support</b><br/>
Buy me a coffee
</div></div>
</li>
</ul>
<ul class="menu startmenu_allprograms">
<li class="startmenuitem">
<div><div style="padding-left:32px;line-height:24px;">
<b>All Programs</b>
<span class="allprograms_arrow"></span>
</div></div>
</li>
<div id="_ui_programsmenu" class="programsmenu" style="display:none;" data-isopen="false">
<ul class="menu">
<li class="menuitem">Test</li>
</ul>
</div>
</ul>
<ul class="menu startmenu_shortcuts">
<li class="startmenuitem" onclick="openLocation('/Documents and Settings/' + xp.profile.name + '/My Documents')">
<img src="` + xp.startmenu.icons.documents + `"/>
<div><div>
<b>My Documents</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openLocation('/Documents and Settings/' + xp.profile.name + '/My Documents')">
<img src="` + xp.startmenu.icons.recentdocuments + `"/>
<div><div>
<b>My Recent Documents</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openLocation('/Documents and Settings/' + xp.profile.name + '/My Documents/My Pictures')">
<img src="` + xp.startmenu.icons.pictures + `"/>
<div><div>
<b>My Pictures</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openLocation('/Documents and Settings/' + xp.profile.name + '/My Documents/My Music')">
<img src="` + xp.startmenu.icons.music + `"/>
<div><div>
<b>My Music</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openLocation('My Computer')">
<img src="` + xp.startmenu.icons.computer + `"/>
<div><div>
<b>My Computer</b>
</div></div>
</li>
<li class="startmenuseparator"></li>
<li class="startmenuitem" onclick="openLocation('Control Panel')">
<img src="` + xp.startmenu.icons.controlpanel + `"/>
<div><div>
<b>Control Panel</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openApp('uac_app')">
<img src="` + xp.startmenu.icons.controlpanel + `"/>
<div><div>
<b>User Accounts and Control</b>
</div></div>
</li>
<li class="startmenuitem" onclick="openApp('themes')">
<img src="` + xp.startmenu.icons.controlpanel + `"/>
<div><div>
<b>Personalize RebornXP</b>
</div></div>
</li>
<li class="startmenuseparator"></li>
<li class="startmenuitem" onclick="openApp('winhelp')">
<img src="` + xp.startmenu.icons.help + `"/>
<div><div>
<b>Help & Support</b>
</div></div>
</li>
<!--Search is under Development-->
<li class="startmenuitem" onclick="">
<img src="` + xp.startmenu.icons.search + `"/>
<div><div>
<b>Search</b>
</div></div>
</li>
<!--Feel free to contribute to this Section-->
<li class="startmenuitem" onclick="runAppDialog()">
<img src="` + xp.startmenu.icons.run + `"/>
<div><div>
<b>Run...</b>
</div></div>
</li>
</ul>
</div>
<div class="startmenu_bottom">
</div>
</div>
<div class="clock" id="_ui_clock">0:00</div>
</taskbar>
<div id="lContextMenu" class="ContextMenu" onselectstart="return false;" oncontextmenu="return false;" style="left: 0px; top: 0px; visibility: hidden;">
Uh oh! Something went wrong!<br>You should <b>not</b> be seeing this!
</div>
`);
$("#_ui_taskbar").css("display", "none");
$("#_ui_boot").css("display", "none");
//$('#_ui_programsmenu').closest('ul').find('li.startmenuitem').on('click', function() {
// $('#_ui_programsmenu').css('display', 'inline');
// $('#_ui_programsmenu').data('isopen', true);
//});
/*$("#_ui_programsmenu").closest("li.startmenuitem").on("mouseleave", function() {
$('#_ui_programsmenu').css('display', 'none');
});*/
$('#_ui_desktop').on('click', function(e) {
var el = $(e.target).closest('.ui_icon');
if (el.hasClass('ui_selected')) {
el.removeClass('ui_selected');
eval(el.data('exe'));
} else {
$('#_ui_desktop .ui_icon').each(function() {
$(this).removeClass('ui_selected');
});
el.addClass('ui_selected');
}
});
$("#_ui_taskbar").css("display", "inline");
$("#_ui_boot").remove();
updateAllWindows();
updateIndexes();
updateAllWindows();
updateTaskbar();
windowsEl.on("mousedown", function(e) {
//var canClose = !(el.is("button.start") || el.is("li.menuitem") || el.is("li.startmenuitem") || el.is("li.startmenuitem img") || el.is("li.startmenuitem div") || el.is("li.startmenuitem div div") || el.is("li.startmenuitem div div b"));
var canClose = !$.contains($('#_ui_taskbar').find('.startmenu')[0], e.target);
if (canClose && !$(e.target).is('button.start')) {
closeStartMenu();
} else {
if ($('#_ui_programsmenu').is(e.target) || $('#_ui_programsmenu').has(e.target).length) return;
if ($('#_ui_programsmenu').closest('ul').find('li.startmenuitem').is(e.target) || $('#_ui_programsmenu').closest('ul').find('li.startmenuitem').has(e.target).length) {
$('#_ui_programsmenu').css('display', $('#_ui_programsmenu').data('isopen') ? 'none ' : 'inline');
$('#_ui_programsmenu').data('isopen', !$('#_ui_programsmenu').data('isopen'));
} else {
$('#_ui_programsmenu').css('display', 'none');
$('#_ui_programsmenu').data('isopen', false);
}
}
});
windowsEl.on('mousedown', function() {
fMouseDown(this);
return false;
});
windowsEl.on('contextmenu', function(e) {
desktopContextMenu(e);
return false;
});
windowsEl.on('click', function() {
rcCloseContext();
});
$('#_ui_taskbar').find('div.startmenu').on('contextmenu', startContextMenu);
updateClock();
};
function runAppDialog() {
closeStartMenu();
var win = new Window({
width: 340,
height: 154,
title: 'Run',
canResize: false,
canMinimize: false
});
win.content(`<form action="#">
<span class="icon icon_run" draggable="false" style="position: absolute; left: 11px; top: 17px;"></span>
<p style="position: absolute; left: 54px; top: 8px;">
Type the name of a program, document, or<br>
Internet resource, and RebornXP will open it for you.
</p>
<p style="position: absolute; top: 54px; left: 12px; line-height: 16px;">Open:</p>
<input type="text" class="prompttext" style="width: 275px; position: absolute; top: 70px; left: 54px;">
<center style="position: absolute; bottom: 17px; right: 16.7px;">
<button class="ok">OK</button>
<button class="cancel">Cancel</button>
<button class="browse">Browse</button>
</center>
</form>`);
win.el.find('.cancel').on('click', function() {
win.close();
});
win.el.find('.ok').on('click', function() {
win.close();
var result = win.el.find('.prompttext').val();
if (!openApp(result)) {
xp.openFileHandler(result);
}
});
win.el.find('.browse').on('click', function() {
xp.filesystem.openFileDialog((file) => {
win.el.find('.prompttext').val(file);
});
});
win.el.find('input[type=text]').focus();
win.el.find('form').on('submit', function() {
return false;
});
}