1
0
forked from sent/waves
waves/public/assets/g/xp/system/core.js
2025-04-09 17:11:14 -05:00

621 lines
31 KiB
JavaScript

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