two-torial-webpatcher/js/dllpatcher.js

311 lines
9.2 KiB
JavaScript
Raw Normal View History

(function(window, document) {
"use strict";
2017-01-28 14:41:42 +10:00
2018-01-07 22:27:01 +10:00
// form labels often need unique IDs - this can be used to generate some
window.DllPatcher_uniqueid = 0;
var createID = function() {
window.DllPatcher_uniqueid++;
return "dllpatch_" + window.DllPatcher_uniqueid;
}
// Each unique kind of patch should have createUI, validatePatch, applyPatch,
// updateUI
var StandardPatch = function(options) {
this.name = options.name;
this.patches = options.patches;
2017-09-04 15:38:54 +10:00
this.tooltip = options.tooltip;
};
StandardPatch.prototype.createUI = function(parent) {
2018-01-07 22:27:01 +10:00
var id = createID();
var label = this.name;
2017-09-04 15:38:54 +10:00
var patch = $('<div>', {'class' : 'patch'});
2018-01-07 22:27:01 +10:00
this.checkbox = $('<input type="checkbox" id="' + id + '">')[0];
patch.append(this.checkbox);
patch.append('<label for="' + id + '">' + label + '</label>');
2017-09-04 15:38:54 +10:00
if(this.tooltip) {
patch.append('<div class="tooltip">' + this.tooltip + '</div>');
2017-09-04 15:38:54 +10:00
}
parent.append(patch);
};
StandardPatch.prototype.updateUI = function(file) {
2018-01-07 22:27:01 +10:00
this.checkbox.checked = this.checkPatchBytes(file) == "on";
};
StandardPatch.prototype.validatePatch = function(file) {
var status = this.checkPatchBytes(file);
if(status == "on") {
console.log('"' + this.name + '"', "is enabled!");
} else if(status == "off") {
console.log('"' + this.name + '"', "is disabled!");
} else {
return '"' + this.name + '" is neither on nor off! Have you got the right dll?';
}
};
StandardPatch.prototype.applyPatch = function(file) {
2018-01-07 22:27:01 +10:00
this.replaceAll(file, this.checkbox.checked);
};
StandardPatch.prototype.replaceAll = function(file, featureOn) {
for(var i = 0; i < this.patches.length; i++) {
replace(file, this.patches[i].offset,
featureOn? this.patches[i].on : this.patches[i].off);
}
}
StandardPatch.prototype.checkPatchBytes = function(file) {
var patchStatus = "";
for(var i = 0; i < this.patches.length; i++) {
var patch = this.patches[i];
if(bytesMatch(file, patch.offset, patch.off)) {
if(patchStatus == "") {
patchStatus = "off";
} else if(patchStatus != "off"){
return "on/off mismatch within patch";
}
} else if(bytesMatch(file, patch.offset, patch.on)) {
if(patchStatus == "") {
patchStatus = "on";
} else if(patchStatus != "on"){
return "on/off mismatch within patch";
}
} else {
return "patch neither on nor off";
}
}
return patchStatus;
}
// Each unique kind of patch should have createUI, validatePatch, applyPatch,
// updateUI
// The DEFAULT state is always the 1st element in the patches array
var UnionPatch = function(options) {
this.name = options.name;
this.offset = options.offset;
this.patches = options.patches;
};
UnionPatch.prototype.createUI = function(parent) {
2018-01-07 22:27:01 +10:00
this.radios = [];
var radio_id = createID();
var container = $("<div>", {"class": "patch-union"});
container.append('<span class="patch-union-title">' + this.name + ':</span>');
for(var i = 0; i < this.patches.length; i++) {
var patch = this.patches[i];
2018-01-07 22:27:01 +10:00
var id = createID();
var label = patch.name;
2017-09-04 20:30:08 +10:00
var patchDiv = $('<div>', {'class' : 'patch'});
2018-01-07 22:27:01 +10:00
var radio = $('<input type="radio" id="' + id + '" name="' + radio_id + '">')[0];
this.radios.push(radio);
patchDiv.append(radio);
patchDiv.append('<label for="' + id + '">' + label + '</label>');
2017-09-04 20:30:08 +10:00
if(patch.tooltip) {
patchDiv.append('<div class="tooltip">' + patch.tooltip + '</div>');
2017-09-04 20:30:08 +10:00
}
container.append(patchDiv);
}
parent.append(container);
};
UnionPatch.prototype.updateUI = function(file) {
for(var i = 0; i < this.patches.length; i++) {
if(bytesMatch(file, this.offset, this.patches[i].patch)) {
2018-01-07 22:27:01 +10:00
this.radios[i].checked = true;
return;
}
}
// Default fallback
2018-01-07 22:27:01 +10:00
this.radios[0].checked = true;
};
UnionPatch.prototype.validatePatch = function(file) {
for(var i = 0; i < this.patches.length; i++) {
if(bytesMatch(file, this.offset, this.patches[i].patch)) {
console.log(this.name, "has", this.patches[i].name, "enabled");
return;
}
}
return '"' + this.name + '" doesn\'t have a valid patch! Have you got the right dll?';
};
UnionPatch.prototype.applyPatch = function(file) {
var patch = this.getSelected();
replace(file, this.offset, patch.patch);
};
UnionPatch.prototype.getSelected = function() {
for(var i = 0; i < this.patches.length; i++) {
2018-01-07 22:27:01 +10:00
if(this.radios[i].checked) {
return this.patches[i];
}
}
return null;
}
var DllPatcher = function(fname, args, description) {
this.mods = [];
for(var i = 0; i < args.length; i++) {
var mod = args[i];
if(mod.type) {
if(mod.type == "union") {
this.mods.push(new UnionPatch(mod));
}
} else { // standard patch
this.mods.push(new StandardPatch(mod));
}
}
this.filename = fname;
this.description = description;
this.createUI();
this.loadPatchUI();
2017-01-28 14:41:42 +10:00
};
DllPatcher.prototype.createUI = function() {
var self = this;
var container = $("<div>", {"class": "patchContainer"});
var header = this.filename + '.dll';
if(this.description) {
header += ' (' + this.description + ')';
}
container.html('<h3>' + header + '</h3>');
$('html').on('dragover dragenter', function() {
container.addClass('dragover');
return true;
})
.on('dragleave dragend drop', function() {
container.removeClass('dragover');
return true;
})
.on('dragover dragenter dragleave dragend drop', function(e) {
e.preventDefault();
});
container.on('drop', function(e) {
var files = e.originalEvent.dataTransfer.files;
if(files && files.length > 0)
self.loadFile(files[0]);
})
this.fileInput = $("<input>",
{"class": "fileInput",
"id" : this.filename + '-file',
"type" : 'file'});
var label = $("<label>", {"class": "fileLabel", "for": this.filename + '-file'});
label.html('<strong>Choose a file</strong> or drag and drop.');
this.fileInput.on('change', function(e) {
if(this.files && this.files.length > 0)
self.loadFile(this.files[0]);
});
this.successDiv = $("<div>", {"class": "success"});
this.errorDiv = $("<div>", {"class": "error"});
this.patchDiv = $("<div>", {"class": "patches"});
var saveButton = $("<button disabled>");
saveButton.text('Load DLL First');
saveButton.on('click', this.saveDll.bind(this));
this.saveButton = saveButton;
container.append(this.fileInput);
container.append(label);
container.append(this.successDiv);
container.append(this.errorDiv);
container.append(this.patchDiv);
container.append(saveButton);
$('body').append(container);
}
2017-01-28 14:41:42 +10:00
DllPatcher.prototype.loadFile = function(file) {
var reader = new FileReader();
var self = this;
2017-01-28 14:41:42 +10:00
reader.onload = function(e) {
self.dllFile = new Uint8Array(e.target.result);
if(self.validatePatches()) {
self.successDiv.removeClass("hidden");
self.successDiv.html("DLL loaded successfully!");
2017-01-28 14:41:42 +10:00
} else {
self.successDiv.addClass("hidden");
2017-01-28 14:41:42 +10:00
}
// Update save button regardless
self.saveButton.prop('disabled', false);
self.saveButton.text('Save DLL');
self.errorDiv.html(self.errorLog);
self.updatePatchUI();
2017-01-28 14:41:42 +10:00
};
reader.readAsArrayBuffer(file);
};
DllPatcher.prototype.saveDll = function() {
if(!this.dllFile || !this.mods || !this.filename)
2017-01-28 14:41:42 +10:00
return;
var fname = this.filename;
for(var i = 0; i < this.mods.length; i++) {
2018-01-07 22:27:01 +10:00
this.mods[i].applyPatch(this.dllFile);
2017-03-13 14:32:53 +10:00
}
2017-01-28 14:41:42 +10:00
fname += '.dll';
var blob = new Blob([this.dllFile], {type: "application/octet-stream"});
2017-01-28 14:41:42 +10:00
saveAs(blob, fname);
}
DllPatcher.prototype.loadPatchUI = function() {
for(var i = 0; i < this.mods.length; i++) {
this.mods[i].createUI(this.patchDiv);
2017-01-28 14:41:42 +10:00
}
}
DllPatcher.prototype.updatePatchUI = function() {
for(var i = 0; i < this.mods.length; i++) {
this.mods[i].updateUI(this.dllFile);
2017-01-28 14:41:42 +10:00
}
}
DllPatcher.prototype.validatePatches = function() {
this.errorLog = "";
var success = true;
for(var i = 0; i < this.mods.length; i++) {
var error = this.mods[i].validatePatch(this.dllFile);
if(error) {
this.errorLog += error + "<br/>";
2017-01-28 14:41:42 +10:00
success = false;
}
}
return success;
}
var bytesMatch = function(buffer, offset, bytes) {
2017-01-28 14:41:42 +10:00
for(var i = 0; i < bytes.length; i++) {
if(buffer[offset+i] != bytes[i])
return false;
}
return true;
};
var replace = function(buffer, offset, bytes) {
2017-01-28 14:41:42 +10:00
for(var i = 0; i < bytes.length; i++) {
buffer[offset+i] = bytes[i];
}
}
var whichBytesMatch = function(buffer, offset, bytesArray) {
2017-01-28 14:41:42 +10:00
for(var i = 0; i < bytesArray.length; i++) {
if(bytesMatch(buffer, offset, bytesArray[i]))
return i;
}
return -1;
}
window.DllPatcher = DllPatcher;
})(window, document);