mirror of
https://gitlab.com/skysthelimit.dev/selenite.git
synced 2025-06-16 10:32:08 -05:00
Add files via upload
This commit is contained in:
parent
c4063a2f44
commit
752b78f300
BIN
xp/boot/assets/logo.png
Normal file
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
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
29
xp/boot/boot.css
Normal 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
150
xp/boot/load.js
Normal 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();
|
||||||
|
});
|
24
xp/index.html
Normal file
24
xp/index.html
Normal 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
102
xp/program/browser.js
Normal 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
18
xp/program/imageviewer.js
Normal 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
23
xp/program/mediaplayer.js
Normal 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
30
xp/program/minesweeper.js
Normal 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
56
xp/program/notepad.js
Normal 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
30
xp/program/paint.js
Normal 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
94
xp/program/terminal.js
Normal 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
38
xp/system/audio.js
Normal 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
34
xp/system/config.js
Normal 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();
|
||||||
|
});
|
||||||
|
}
|
19
xp/system/contextmenus.css
Normal file
19
xp/system/contextmenus.css
Normal 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
196
xp/system/contextmenus.js
Normal 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
254
xp/system/controlpanel.js
Normal 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
620
xp/system/core.js
Normal 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,'&').replace(/</g,'<').replace(/>/g,'>');
|
||||||
|
$('window[guid=' + guid + ']').attr('title', title);
|
||||||
|
$('window[guid=' + guid + ']').find('.windowTitle').html(title.replace(/ /g,' '));
|
||||||
|
$('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) ? ' ' : 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) ? ' ' : 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
19
xp/system/cursors.css
Normal 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
79
xp/system/desktop.css
Normal 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
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
690
xp/system/explorer.js
Normal 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
227
xp/system/filesystem.js
Normal 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
49
xp/system/fonts.css
Normal 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
29
xp/system/help.js
Normal 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
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
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
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
639
xp/system/lambda.js
Normal 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
44
xp/system/script.js
Normal 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
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
396
xp/system/uac.js
Normal 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> </td></tr><tr><td> </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> </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
396
xp/system/uac_app.js
Normal 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> </td></tr><tr><td> </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> </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'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
613
xp/system/unix_formatting.js
Normal file
613
xp/system/unix_formatting.js
Normal 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
108
xp/system/widgets.css
Normal 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
88
xp/system/window-api.js
Normal 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
305
xp/system/window.css
Normal 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
369
xp/system/xp.css
Normal 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
482
xp/system/xp.js
Normal 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/>
|
||||||
|
 ` + msg + `<br/>
|
||||||
|
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;
|
||||||
|
});
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user