blob: 781eab28f01ec22e92779da46d5c14db2514a80b [file] [log] [blame] [raw]
// Copyright (c) 2012-2017, Matt Godbolt
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
define(function (require) {
"use strict";
var CodeMirror = require('codemirror');
var FontScale = require('fontscale');
require('selectize');
function State(id) {
this.id = id;
this.compiler = null;
this.result = null;
}
State.prototype.update = function (id, compiler, result) {
if (this.id !== id) return false;
this.compiler = compiler;
this.result = result;
return true;
}
function Diff(hub, container, state) {
this.container = container;
this.lhs = new State(state.lhs);
this.rhs = new State(state.rhs);
this.eventHub = hub.createEventHub();
this.domRoot = container.getElement();
this.domRoot.html($('#diff').html());
this.compilers = {};
this.outputEditor = CodeMirror.fromTextArea(this.domRoot.find("textarea")[0], {
lineNumbers: true,
mode: "text/x-asm",
readOnly: true,
gutters: ['CodeMirror-linenumbers'],
lineWrapping: true
});
var selectize = this.domRoot.find(".compiler-picker").selectize({
sortField: 'name',
valueField: 'id',
labelField: 'name',
searchField: ['name'],
options: [],
items: []
}).on('change', function () {
console.log("yibble");
// TODO can't do this every time!!
// self.onCompilerChange($(this).val());
});
this.selectize = {lhs: selectize[0].selectize, rhs: selectize[1].selectize};
this.fontScale = new FontScale(this.domRoot, state);
this.fontScale.on('change', _.bind(this.updateState, this));
this.eventHub.on('compileResult', this.onCompileResult, this);
this.eventHub.on('compiler', this.onCompiler, this);
this.eventHub.on('compilerClose', this.onCompilerClose, this);
this.container.on('destroy', function () {
this.eventHub.unsubscribe();
}, this);
container.on('resize', this.resize, this);
container.on('shown', this.refresh, this);
this.eventHub.emit('resendCompilation', this.lhs);
this.eventHub.emit('resendCompilation', this.rhs);
this.eventHub.emit('findCompilers');
this.updateCompilerNames();
this.updateCompilers();
}
// TODO: de-dupe with compiler etc
Diff.prototype.refresh = function () {
this.outputEditor.refresh();
};
// TODO: need to call resize if either .top-bar or .bottom-bar resizes, which needs some work.
// Issue manifests if you make a window where one compiler is small enough that the buttons spill onto two lines:
// reload the page and the bottom-bar is off the bottom until you scroll a tiny bit.
Diff.prototype.resize = function () {
var topBarHeight = this.domRoot.find(".top-bar").outerHeight(true);
var bottomBarHeight = this.domRoot.find(".bottom-bar").outerHeight(true);
this.outputEditor.setSize(this.domRoot.width(), this.domRoot.height() - topBarHeight - bottomBarHeight);
this.refresh();
};
Diff.prototype.onCompileResult = function (id, compiler, result) {
var changed = this.lhs.update(id, compiler, result) || this.rhs.update(id, compiler, result);
if (changed) {
this.updateCompilerNames();
}
};
Diff.prototype.onCompiler = function (id, compiler, options) {
// TODO update lhs, rhs?
this.compilers[id] = {id: id, name: compiler.name + " " + options};
this.updateCompilers();
};
Diff.prototype.onCompilerClose = function (id) {
// TODO update lhs, rhs
delete this.compilers[id];
this.updateCompilers();
};
Diff.prototype.updateCompilerNames = function () {
var name = "Diff";
if (this.lhs.compiler && this.rhs.compiler)
name += " " + this.lhs.compiler.name + " vs " + this.rhs.compiler.name;
this.container.setTitle(name);
};
Diff.prototype.updateCompilersFor = function (selectize) {
selectize.clearOptions();
_.each(this.compilers, function (compiler) {
selectize.addOption(compiler);
}, this);
};
Diff.prototype.updateCompilers = function () {
this.updateCompilersFor(this.selectize.lhs);
this.updateCompilersFor(this.selectize.rhs);
};
Diff.prototype.updateState = function () {
var state = {
lhs: this.lhs.id,
rhs: this.rhs.id
};
this.fontScale.addState(state);
this.container.setState(state);
};
return {
Diff: Diff,
getComponent: function (lhs, rhs) {
return {
type: 'component',
componentName: 'diff',
componentState: {lhs: lhs, rhs: rhs},
};
}
};
});