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);
});