mirror of
https://github.com/EDCD/coriolis.git
synced 2025-12-09 06:43:24 +00:00
Compare commits
192 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
08c5d2256a | ||
|
|
ed6ee4341f | ||
|
|
157c1148fb | ||
|
|
507ea9e09e | ||
|
|
af68cba7be | ||
|
|
224fbe0e8f | ||
|
|
07c936897c | ||
|
|
3786fb907c | ||
|
|
752d9f0c68 | ||
|
|
baf59aafcb | ||
|
|
8787303d2a | ||
|
|
118c80af27 | ||
|
|
d103939e45 | ||
|
|
2a6ade3cab | ||
|
|
c8c42689f9 | ||
|
|
da2f472f4d | ||
|
|
3ba878237b | ||
|
|
7577fb53a2 | ||
|
|
f2509f89ee | ||
|
|
9dd1f78330 | ||
|
|
ebb6c2c420 | ||
|
|
9d23dc1763 | ||
|
|
5fa3f8703e | ||
|
|
797885faea | ||
|
|
68e7e9f5b7 | ||
|
|
d1d165ad51 | ||
|
|
0ebb247666 | ||
|
|
aa73bc2809 | ||
|
|
c3fcdb918f | ||
|
|
5e3722bcfd | ||
|
|
6a4fca2eb1 | ||
|
|
6922cfd047 | ||
|
|
d93fc1d2d0 | ||
|
|
48290b2e75 | ||
|
|
b62abef618 | ||
|
|
e0c0778d82 | ||
|
|
b4a82ae7c2 | ||
|
|
fc73102b30 | ||
|
|
b14e7473f3 | ||
|
|
f4cc9fc722 | ||
|
|
8d19ef7783 | ||
|
|
32be186ec5 | ||
|
|
abb0c7f90d | ||
|
|
71ddbdfe75 | ||
|
|
38463ad9a6 | ||
|
|
70375f94c8 | ||
|
|
2d4336116a | ||
|
|
f52880765e | ||
|
|
dbfe68decb | ||
|
|
659f337de9 | ||
|
|
1d36d41da1 | ||
|
|
0f90efaa54 | ||
|
|
4442930a82 | ||
|
|
912a775088 | ||
|
|
24a229d818 | ||
|
|
66afb61494 | ||
|
|
e7511cc05b | ||
|
|
926f19a936 | ||
|
|
9e6f86b963 | ||
|
|
059de43a9a | ||
|
|
e246b737b2 | ||
|
|
38842417b0 | ||
|
|
82d485a98e | ||
|
|
fac71feea7 | ||
|
|
445c63878b | ||
|
|
7df5953824 | ||
|
|
e391b563fb | ||
|
|
53f62f96d0 | ||
|
|
18745979a0 | ||
|
|
c1966a38ff | ||
|
|
d86973f3b1 | ||
|
|
56b8d19649 | ||
|
|
7cb037e0bc | ||
|
|
56e1b3f9e9 | ||
|
|
c792323a8a | ||
|
|
4d2865de13 | ||
|
|
0d360bc367 | ||
|
|
91c9b46b91 | ||
|
|
7e67bd80dd | ||
|
|
2f775ea09b | ||
|
|
0603c55089 | ||
|
|
41f25a44e9 | ||
|
|
35440b7273 | ||
|
|
7d68b91018 | ||
|
|
deaa61b848 | ||
|
|
2d00cbc41b | ||
|
|
57c1e83c67 | ||
|
|
b7079dbd4e | ||
|
|
be642a5373 | ||
|
|
ffc691c1a2 | ||
|
|
6122d99369 | ||
|
|
b73a8bcdab | ||
|
|
dd624537b7 | ||
|
|
50a67f73fd | ||
|
|
2262a980d4 | ||
|
|
84964ceb5f | ||
|
|
c4bdb7a66e | ||
|
|
21309e129f | ||
|
|
928e02c718 | ||
|
|
7f494dd200 | ||
|
|
f38e743e59 | ||
|
|
58b55eb3da | ||
|
|
8375ad95b3 | ||
|
|
506d027a2d | ||
|
|
fbeb6237cf | ||
|
|
bd95e2c5a5 | ||
|
|
f3276e557a | ||
|
|
8b813e0e7f | ||
|
|
d660d2959f | ||
|
|
014ebda7d2 | ||
|
|
607398d364 | ||
|
|
97d141ce2b | ||
|
|
9f492db9c6 | ||
|
|
6ed82b366c | ||
|
|
a7ca037f48 | ||
|
|
6e21d0e74a | ||
|
|
e6c75bf2af | ||
|
|
270c2f386e | ||
|
|
83f7880c58 | ||
|
|
51d29aee38 | ||
|
|
c7754c0365 | ||
|
|
94037cea38 | ||
|
|
5c1a9d9eea | ||
|
|
3835c73ffd | ||
|
|
b8c1effecb | ||
|
|
dd07241dd9 | ||
|
|
5ad828a613 | ||
|
|
9856df5527 | ||
|
|
7b249900ec | ||
|
|
e3b9267c3f | ||
|
|
a996b8135a | ||
|
|
5e1237390b | ||
|
|
83571b4bef | ||
|
|
c2a0dad9a8 | ||
|
|
f6f057689c | ||
|
|
4378f0020e | ||
|
|
2e9e7c4fc6 | ||
|
|
b7b8ee5580 | ||
|
|
c630dbbed0 | ||
|
|
fa6703a3b8 | ||
|
|
bdcb64c9d1 | ||
|
|
a1afc869a7 | ||
|
|
d8ce26c7cf | ||
|
|
95c474dc05 | ||
|
|
56ca73b4ad | ||
|
|
d82cd6a89e | ||
|
|
bf20d32364 | ||
|
|
e968e62fca | ||
|
|
b9c9ca9fa1 | ||
|
|
6e965e2e98 | ||
|
|
7d569f9036 | ||
|
|
df5a77199d | ||
|
|
3c8dfebfdc | ||
|
|
3cc422596f | ||
|
|
4ed167de22 | ||
|
|
b420647501 | ||
|
|
0c318b5e68 | ||
|
|
1c627297b8 | ||
|
|
f41e2d0552 | ||
|
|
1a1d539c60 | ||
|
|
ba2e46f88f | ||
|
|
c2f1fa81af | ||
|
|
752e03fa0f | ||
|
|
1da69664d7 | ||
|
|
9aa986a133 | ||
|
|
bdbfb28c4a | ||
|
|
162156bb2b | ||
|
|
64c5d542e9 | ||
|
|
31dc789f6e | ||
|
|
b0e2cfd7db | ||
|
|
54ddb0d014 | ||
|
|
e19688e96f | ||
|
|
4f53d75999 | ||
|
|
7277460060 | ||
|
|
93c4f6f3c0 | ||
|
|
c6919a7518 | ||
|
|
ca428e67dc | ||
|
|
c6726cf020 | ||
|
|
56ae1378da | ||
|
|
e982ab1a3b | ||
|
|
0d6aa87e89 | ||
|
|
94d06e4025 | ||
|
|
cee5b297ac | ||
|
|
c549213ce0 | ||
|
|
f3f9112767 | ||
|
|
fb325ea3e2 | ||
|
|
3773f6f7ec | ||
|
|
b90ab6fe48 | ||
|
|
cc2e56dc8a | ||
|
|
ae65af7bbf | ||
|
|
c8fb513cd1 | ||
|
|
7fc3855af4 |
2
.babelrc
2
.babelrc
@@ -1,3 +1,3 @@
|
|||||||
{
|
{
|
||||||
"presets": ["es2015", "react", "stage-0"]
|
"presets": ["env", "react", "stage-0"]
|
||||||
}
|
}
|
||||||
21
.editorconfig
Normal file
21
.editorconfig
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
root = true
|
||||||
|
|
||||||
|
|
||||||
|
[*]
|
||||||
|
|
||||||
|
# change these settings to your own preference
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# we recommend you to keep these unchanged
|
||||||
|
end_of_line = lf
|
||||||
|
charset = utf-8
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
[*.md]
|
||||||
|
trim_trailing_whitespace = false
|
||||||
|
|
||||||
|
[{package,bower}.json]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -7,3 +7,4 @@ nginx.pid
|
|||||||
/bin
|
/bin
|
||||||
env
|
env
|
||||||
*.swp
|
*.swp
|
||||||
|
.project
|
||||||
|
|||||||
21
ChangeLog.md
21
ChangeLog.md
@@ -1,3 +1,24 @@
|
|||||||
|
#2.5.1
|
||||||
|
* Passenger count on main page
|
||||||
|
* AX Modules
|
||||||
|
* Engineering fixes
|
||||||
|
* Use coriolis-data 2.5.1
|
||||||
|
|
||||||
|
#2.5.0
|
||||||
|
* willyb321 and myself have conquered engineering. Mainly him though...
|
||||||
|
* Use coriolis-data 2.5.0
|
||||||
|
|
||||||
|
#2.4.2
|
||||||
|
Lots of kind people have helped out for this release! Check out the PR history!
|
||||||
|
* Uses coriolis-data update:
|
||||||
|
* Fixes issues with repair limpets
|
||||||
|
* Adds requirement data
|
||||||
|
* Adds requirements panel
|
||||||
|
* Adds comma formatting to tooltip numbers
|
||||||
|
|
||||||
|
#2.4.1
|
||||||
|
* Small patches and changes
|
||||||
|
|
||||||
#2.4.0
|
#2.4.0
|
||||||
* Changed compression library to Pako
|
* Changed compression library to Pako
|
||||||
* Use coriolis-data 2.4.0
|
* Use coriolis-data 2.4.0
|
||||||
|
|||||||
878
d3.js
vendored
878
d3.js
vendored
@@ -1340,6 +1340,12 @@ Selection.prototype = selection.prototype = {
|
|||||||
dispatch: selection_dispatch
|
dispatch: selection_dispatch
|
||||||
};
|
};
|
||||||
|
|
||||||
|
var select = function(selector) {
|
||||||
|
return typeof selector === "string"
|
||||||
|
? new Selection([[document.querySelector(selector)]], [document.documentElement])
|
||||||
|
: new Selection([[selector]], root);
|
||||||
|
};
|
||||||
|
|
||||||
var define = function(constructor, factory, prototype) {
|
var define = function(constructor, factory, prototype) {
|
||||||
constructor.prototype = factory.prototype = prototype;
|
constructor.prototype = factory.prototype = prototype;
|
||||||
prototype.constructor = constructor;
|
prototype.constructor = constructor;
|
||||||
@@ -6947,861 +6953,7 @@ var cool = cubehelixLong(cubehelix(260, 0.75, 0.35), cubehelix(80, 1.50, 0.8));
|
|||||||
|
|
||||||
var rainbow = cubehelix();
|
var rainbow = cubehelix();
|
||||||
|
|
||||||
var xhtml$1 = "http://www.w3.org/1999/xhtml";
|
|
||||||
|
|
||||||
var namespaces$1 = {
|
|
||||||
svg: "http://www.w3.org/2000/svg",
|
|
||||||
xhtml: xhtml$1,
|
|
||||||
xlink: "http://www.w3.org/1999/xlink",
|
|
||||||
xml: "http://www.w3.org/XML/1998/namespace",
|
|
||||||
xmlns: "http://www.w3.org/2000/xmlns/"
|
|
||||||
};
|
|
||||||
|
|
||||||
var namespace$1 = function(name) {
|
|
||||||
var prefix = name += "", i = prefix.indexOf(":");
|
|
||||||
if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
|
|
||||||
return namespaces$1.hasOwnProperty(prefix) ? {space: namespaces$1[prefix], local: name} : name;
|
|
||||||
};
|
|
||||||
|
|
||||||
function creatorInherit$1(name) {
|
|
||||||
return function() {
|
|
||||||
var document = this.ownerDocument,
|
|
||||||
uri = this.namespaceURI;
|
|
||||||
return uri === xhtml$1 && document.documentElement.namespaceURI === xhtml$1
|
|
||||||
? document.createElement(name)
|
|
||||||
: document.createElementNS(uri, name);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function creatorFixed$1(fullname) {
|
|
||||||
return function() {
|
|
||||||
return this.ownerDocument.createElementNS(fullname.space, fullname.local);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var creator$1 = function(name) {
|
|
||||||
var fullname = namespace$1(name);
|
|
||||||
return (fullname.local
|
|
||||||
? creatorFixed$1
|
|
||||||
: creatorInherit$1)(fullname);
|
|
||||||
};
|
|
||||||
|
|
||||||
var matcher$2 = function(selector) {
|
|
||||||
return function() {
|
|
||||||
return this.matches(selector);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
if (typeof document !== "undefined") {
|
|
||||||
var element$2 = document.documentElement;
|
|
||||||
if (!element$2.matches) {
|
|
||||||
var vendorMatches$1 = element$2.webkitMatchesSelector
|
|
||||||
|| element$2.msMatchesSelector
|
|
||||||
|| element$2.mozMatchesSelector
|
|
||||||
|| element$2.oMatchesSelector;
|
|
||||||
matcher$2 = function(selector) {
|
|
||||||
return function() {
|
|
||||||
return vendorMatches$1.call(this, selector);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var matcher$3 = matcher$2;
|
|
||||||
|
|
||||||
var filterEvents$1 = {};
|
|
||||||
|
|
||||||
var event$1 = null;
|
|
||||||
|
|
||||||
if (typeof document !== "undefined") {
|
|
||||||
var element$3 = document.documentElement;
|
|
||||||
if (!("onmouseenter" in element$3)) {
|
|
||||||
filterEvents$1 = {mouseenter: "mouseover", mouseleave: "mouseout"};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function filterContextListener$1(listener, index, group) {
|
|
||||||
listener = contextListener$1(listener, index, group);
|
|
||||||
return function(event) {
|
|
||||||
var related = event.relatedTarget;
|
|
||||||
if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
|
|
||||||
listener.call(this, event);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function contextListener$1(listener, index, group) {
|
|
||||||
return function(event1) {
|
|
||||||
var event0 = event$1; // Events can be reentrant (e.g., focus).
|
|
||||||
event$1 = event1;
|
|
||||||
try {
|
|
||||||
listener.call(this, this.__data__, index, group);
|
|
||||||
} finally {
|
|
||||||
event$1 = event0;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseTypenames$2(typenames) {
|
|
||||||
return typenames.trim().split(/^|\s+/).map(function(t) {
|
|
||||||
var name = "", i = t.indexOf(".");
|
|
||||||
if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
|
|
||||||
return {type: t, name: name};
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function onRemove$1(typename) {
|
|
||||||
return function() {
|
|
||||||
var on = this.__on;
|
|
||||||
if (!on) return;
|
|
||||||
for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
|
|
||||||
if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
|
|
||||||
this.removeEventListener(o.type, o.listener, o.capture);
|
|
||||||
} else {
|
|
||||||
on[++i] = o;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (++i) on.length = i;
|
|
||||||
else delete this.__on;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function onAdd$1(typename, value, capture) {
|
|
||||||
var wrap = filterEvents$1.hasOwnProperty(typename.type) ? filterContextListener$1 : contextListener$1;
|
|
||||||
return function(d, i, group) {
|
|
||||||
var on = this.__on, o, listener = wrap(value, i, group);
|
|
||||||
if (on) for (var j = 0, m = on.length; j < m; ++j) {
|
|
||||||
if ((o = on[j]).type === typename.type && o.name === typename.name) {
|
|
||||||
this.removeEventListener(o.type, o.listener, o.capture);
|
|
||||||
this.addEventListener(o.type, o.listener = listener, o.capture = capture);
|
|
||||||
o.value = value;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.addEventListener(typename.type, listener, capture);
|
|
||||||
o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
|
|
||||||
if (!on) this.__on = [o];
|
|
||||||
else on.push(o);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_on$1 = function(typename, value, capture) {
|
|
||||||
var typenames = parseTypenames$2(typename + ""), i, n = typenames.length, t;
|
|
||||||
|
|
||||||
if (arguments.length < 2) {
|
|
||||||
var on = this.node().__on;
|
|
||||||
if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
|
|
||||||
for (i = 0, o = on[j]; i < n; ++i) {
|
|
||||||
if ((t = typenames[i]).type === o.type && t.name === o.name) {
|
|
||||||
return o.value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
on = value ? onAdd$1 : onRemove$1;
|
|
||||||
if (capture == null) capture = false;
|
|
||||||
for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
|
|
||||||
return this;
|
|
||||||
};
|
|
||||||
|
|
||||||
function none$1() {}
|
|
||||||
|
|
||||||
var selector$1 = function(selector) {
|
|
||||||
return selector == null ? none$1 : function() {
|
|
||||||
return this.querySelector(selector);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_select$1 = function(select) {
|
|
||||||
if (typeof select !== "function") select = selector$1(select);
|
|
||||||
|
|
||||||
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
|
|
||||||
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
|
|
||||||
if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
|
|
||||||
if ("__data__" in node) subnode.__data__ = node.__data__;
|
|
||||||
subgroup[i] = subnode;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Selection$2(subgroups, this._parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
function empty$2() {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
var selectorAll$1 = function(selector) {
|
|
||||||
return selector == null ? empty$2 : function() {
|
|
||||||
return this.querySelectorAll(selector);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_selectAll$1 = function(select) {
|
|
||||||
if (typeof select !== "function") select = selectorAll$1(select);
|
|
||||||
|
|
||||||
for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
|
|
||||||
for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
subgroups.push(select.call(node, node.__data__, i, group));
|
|
||||||
parents.push(node);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Selection$2(subgroups, parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_filter$1 = function(match) {
|
|
||||||
if (typeof match !== "function") match = matcher$3(match);
|
|
||||||
|
|
||||||
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
|
|
||||||
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
|
|
||||||
if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
|
|
||||||
subgroup.push(node);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Selection$2(subgroups, this._parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
var sparse$1 = function(update) {
|
|
||||||
return new Array(update.length);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_enter$1 = function() {
|
|
||||||
return new Selection$2(this._enter || this._groups.map(sparse$1), this._parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
function EnterNode$1(parent, datum) {
|
|
||||||
this.ownerDocument = parent.ownerDocument;
|
|
||||||
this.namespaceURI = parent.namespaceURI;
|
|
||||||
this._next = null;
|
|
||||||
this._parent = parent;
|
|
||||||
this.__data__ = datum;
|
|
||||||
}
|
|
||||||
|
|
||||||
EnterNode$1.prototype = {
|
|
||||||
constructor: EnterNode$1,
|
|
||||||
appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
|
|
||||||
insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
|
|
||||||
querySelector: function(selector) { return this._parent.querySelector(selector); },
|
|
||||||
querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
|
|
||||||
};
|
|
||||||
|
|
||||||
var constant$10 = function(x) {
|
var constant$10 = function(x) {
|
||||||
return function() {
|
|
||||||
return x;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
var keyPrefix$2 = "$"; // Protect against keys like “__proto__”.
|
|
||||||
|
|
||||||
function bindIndex$1(parent, group, enter, update, exit, data) {
|
|
||||||
var i = 0,
|
|
||||||
node,
|
|
||||||
groupLength = group.length,
|
|
||||||
dataLength = data.length;
|
|
||||||
|
|
||||||
// Put any non-null nodes that fit into update.
|
|
||||||
// Put any null nodes into enter.
|
|
||||||
// Put any remaining data into enter.
|
|
||||||
for (; i < dataLength; ++i) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
node.__data__ = data[i];
|
|
||||||
update[i] = node;
|
|
||||||
} else {
|
|
||||||
enter[i] = new EnterNode$1(parent, data[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Put any non-null nodes that don’t fit into exit.
|
|
||||||
for (; i < groupLength; ++i) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
exit[i] = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function bindKey$1(parent, group, enter, update, exit, data, key) {
|
|
||||||
var i,
|
|
||||||
node,
|
|
||||||
nodeByKeyValue = {},
|
|
||||||
groupLength = group.length,
|
|
||||||
dataLength = data.length,
|
|
||||||
keyValues = new Array(groupLength),
|
|
||||||
keyValue;
|
|
||||||
|
|
||||||
// Compute the key for each node.
|
|
||||||
// If multiple nodes have the same key, the duplicates are added to exit.
|
|
||||||
for (i = 0; i < groupLength; ++i) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
keyValues[i] = keyValue = keyPrefix$2 + key.call(node, node.__data__, i, group);
|
|
||||||
if (keyValue in nodeByKeyValue) {
|
|
||||||
exit[i] = node;
|
|
||||||
} else {
|
|
||||||
nodeByKeyValue[keyValue] = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compute the key for each datum.
|
|
||||||
// If there a node associated with this key, join and add it to update.
|
|
||||||
// If there is not (or the key is a duplicate), add it to enter.
|
|
||||||
for (i = 0; i < dataLength; ++i) {
|
|
||||||
keyValue = keyPrefix$2 + key.call(parent, data[i], i, data);
|
|
||||||
if (node = nodeByKeyValue[keyValue]) {
|
|
||||||
update[i] = node;
|
|
||||||
node.__data__ = data[i];
|
|
||||||
nodeByKeyValue[keyValue] = null;
|
|
||||||
} else {
|
|
||||||
enter[i] = new EnterNode$1(parent, data[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add any remaining nodes that were not bound to data to exit.
|
|
||||||
for (i = 0; i < groupLength; ++i) {
|
|
||||||
if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
|
|
||||||
exit[i] = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_data$1 = function(value, key) {
|
|
||||||
if (!value) {
|
|
||||||
data = new Array(this.size()), j = -1;
|
|
||||||
this.each(function(d) { data[++j] = d; });
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
var bind = key ? bindKey$1 : bindIndex$1,
|
|
||||||
parents = this._parents,
|
|
||||||
groups = this._groups;
|
|
||||||
|
|
||||||
if (typeof value !== "function") value = constant$10(value);
|
|
||||||
|
|
||||||
for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
|
|
||||||
var parent = parents[j],
|
|
||||||
group = groups[j],
|
|
||||||
groupLength = group.length,
|
|
||||||
data = value.call(parent, parent && parent.__data__, j, parents),
|
|
||||||
dataLength = data.length,
|
|
||||||
enterGroup = enter[j] = new Array(dataLength),
|
|
||||||
updateGroup = update[j] = new Array(dataLength),
|
|
||||||
exitGroup = exit[j] = new Array(groupLength);
|
|
||||||
|
|
||||||
bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);
|
|
||||||
|
|
||||||
// Now connect the enter nodes to their following update node, such that
|
|
||||||
// appendChild can insert the materialized enter node before this node,
|
|
||||||
// rather than at the end of the parent node.
|
|
||||||
for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
|
|
||||||
if (previous = enterGroup[i0]) {
|
|
||||||
if (i0 >= i1) i1 = i0 + 1;
|
|
||||||
while (!(next = updateGroup[i1]) && ++i1 < dataLength);
|
|
||||||
previous._next = next || null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
update = new Selection$2(update, parents);
|
|
||||||
update._enter = enter;
|
|
||||||
update._exit = exit;
|
|
||||||
return update;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_exit$1 = function() {
|
|
||||||
return new Selection$2(this._exit || this._groups.map(sparse$1), this._parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_merge$1 = function(selection) {
|
|
||||||
|
|
||||||
for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
|
|
||||||
for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
|
|
||||||
if (node = group0[i] || group1[i]) {
|
|
||||||
merge[i] = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (; j < m0; ++j) {
|
|
||||||
merges[j] = groups0[j];
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Selection$2(merges, this._parents);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_order$1 = function() {
|
|
||||||
|
|
||||||
for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
|
|
||||||
for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
if (next && next !== node.nextSibling) next.parentNode.insertBefore(node, next);
|
|
||||||
next = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_sort$1 = function(compare) {
|
|
||||||
if (!compare) compare = ascending$2;
|
|
||||||
|
|
||||||
function compareNode(a, b) {
|
|
||||||
return a && b ? compare(a.__data__, b.__data__) : !a - !b;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
|
|
||||||
for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
|
|
||||||
if (node = group[i]) {
|
|
||||||
sortgroup[i] = node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
sortgroup.sort(compareNode);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Selection$2(sortgroups, this._parents).order();
|
|
||||||
};
|
|
||||||
|
|
||||||
function ascending$2(a, b) {
|
|
||||||
return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_call$1 = function() {
|
|
||||||
var callback = arguments[0];
|
|
||||||
arguments[0] = this;
|
|
||||||
callback.apply(null, arguments);
|
|
||||||
return this;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_nodes$1 = function() {
|
|
||||||
var nodes = new Array(this.size()), i = -1;
|
|
||||||
this.each(function() { nodes[++i] = this; });
|
|
||||||
return nodes;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_node$1 = function() {
|
|
||||||
|
|
||||||
for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
|
|
||||||
for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
|
|
||||||
var node = group[i];
|
|
||||||
if (node) return node;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_size$1 = function() {
|
|
||||||
var size = 0;
|
|
||||||
this.each(function() { ++size; });
|
|
||||||
return size;
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_empty$1 = function() {
|
|
||||||
return !this.node();
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_each$1 = function(callback) {
|
|
||||||
|
|
||||||
for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
|
|
||||||
for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
|
|
||||||
if (node = group[i]) callback.call(node, node.__data__, i, group);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
};
|
|
||||||
|
|
||||||
function attrRemove$2(name) {
|
|
||||||
return function() {
|
|
||||||
this.removeAttribute(name);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function attrRemoveNS$2(fullname) {
|
|
||||||
return function() {
|
|
||||||
this.removeAttributeNS(fullname.space, fullname.local);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function attrConstant$2(name, value) {
|
|
||||||
return function() {
|
|
||||||
this.setAttribute(name, value);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function attrConstantNS$2(fullname, value) {
|
|
||||||
return function() {
|
|
||||||
this.setAttributeNS(fullname.space, fullname.local, value);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function attrFunction$2(name, value) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
if (v == null) this.removeAttribute(name);
|
|
||||||
else this.setAttribute(name, v);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function attrFunctionNS$2(fullname, value) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
|
|
||||||
else this.setAttributeNS(fullname.space, fullname.local, v);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_attr$1 = function(name, value) {
|
|
||||||
var fullname = namespace$1(name);
|
|
||||||
|
|
||||||
if (arguments.length < 2) {
|
|
||||||
var node = this.node();
|
|
||||||
return fullname.local
|
|
||||||
? node.getAttributeNS(fullname.space, fullname.local)
|
|
||||||
: node.getAttribute(fullname);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.each((value == null
|
|
||||||
? (fullname.local ? attrRemoveNS$2 : attrRemove$2) : (typeof value === "function"
|
|
||||||
? (fullname.local ? attrFunctionNS$2 : attrFunction$2)
|
|
||||||
: (fullname.local ? attrConstantNS$2 : attrConstant$2)))(fullname, value));
|
|
||||||
};
|
|
||||||
|
|
||||||
var defaultView = function(node) {
|
|
||||||
return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
|
|
||||||
|| (node.document && node) // node is a Window
|
|
||||||
|| node.defaultView; // node is a Document
|
|
||||||
};
|
|
||||||
|
|
||||||
function styleRemove$2(name) {
|
|
||||||
return function() {
|
|
||||||
this.style.removeProperty(name);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function styleConstant$2(name, value, priority) {
|
|
||||||
return function() {
|
|
||||||
this.style.setProperty(name, value, priority);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function styleFunction$2(name, value, priority) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
if (v == null) this.style.removeProperty(name);
|
|
||||||
else this.style.setProperty(name, v, priority);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_style$1 = function(name, value, priority) {
|
|
||||||
var node;
|
|
||||||
return arguments.length > 1
|
|
||||||
? this.each((value == null
|
|
||||||
? styleRemove$2 : typeof value === "function"
|
|
||||||
? styleFunction$2
|
|
||||||
: styleConstant$2)(name, value, priority == null ? "" : priority))
|
|
||||||
: defaultView(node = this.node())
|
|
||||||
.getComputedStyle(node, null)
|
|
||||||
.getPropertyValue(name);
|
|
||||||
};
|
|
||||||
|
|
||||||
function propertyRemove$1(name) {
|
|
||||||
return function() {
|
|
||||||
delete this[name];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function propertyConstant$1(name, value) {
|
|
||||||
return function() {
|
|
||||||
this[name] = value;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function propertyFunction$1(name, value) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
if (v == null) delete this[name];
|
|
||||||
else this[name] = v;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_property$1 = function(name, value) {
|
|
||||||
return arguments.length > 1
|
|
||||||
? this.each((value == null
|
|
||||||
? propertyRemove$1 : typeof value === "function"
|
|
||||||
? propertyFunction$1
|
|
||||||
: propertyConstant$1)(name, value))
|
|
||||||
: this.node()[name];
|
|
||||||
};
|
|
||||||
|
|
||||||
function classArray$1(string) {
|
|
||||||
return string.trim().split(/^|\s+/);
|
|
||||||
}
|
|
||||||
|
|
||||||
function classList$1(node) {
|
|
||||||
return node.classList || new ClassList$1(node);
|
|
||||||
}
|
|
||||||
|
|
||||||
function ClassList$1(node) {
|
|
||||||
this._node = node;
|
|
||||||
this._names = classArray$1(node.getAttribute("class") || "");
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassList$1.prototype = {
|
|
||||||
add: function(name) {
|
|
||||||
var i = this._names.indexOf(name);
|
|
||||||
if (i < 0) {
|
|
||||||
this._names.push(name);
|
|
||||||
this._node.setAttribute("class", this._names.join(" "));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
remove: function(name) {
|
|
||||||
var i = this._names.indexOf(name);
|
|
||||||
if (i >= 0) {
|
|
||||||
this._names.splice(i, 1);
|
|
||||||
this._node.setAttribute("class", this._names.join(" "));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
contains: function(name) {
|
|
||||||
return this._names.indexOf(name) >= 0;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
function classedAdd$1(node, names) {
|
|
||||||
var list = classList$1(node), i = -1, n = names.length;
|
|
||||||
while (++i < n) list.add(names[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
function classedRemove$1(node, names) {
|
|
||||||
var list = classList$1(node), i = -1, n = names.length;
|
|
||||||
while (++i < n) list.remove(names[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
function classedTrue$1(names) {
|
|
||||||
return function() {
|
|
||||||
classedAdd$1(this, names);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function classedFalse$1(names) {
|
|
||||||
return function() {
|
|
||||||
classedRemove$1(this, names);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function classedFunction$1(names, value) {
|
|
||||||
return function() {
|
|
||||||
(value.apply(this, arguments) ? classedAdd$1 : classedRemove$1)(this, names);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_classed$1 = function(name, value) {
|
|
||||||
var names = classArray$1(name + "");
|
|
||||||
|
|
||||||
if (arguments.length < 2) {
|
|
||||||
var list = classList$1(this.node()), i = -1, n = names.length;
|
|
||||||
while (++i < n) if (!list.contains(names[i])) return false;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.each((typeof value === "function"
|
|
||||||
? classedFunction$1 : value
|
|
||||||
? classedTrue$1
|
|
||||||
: classedFalse$1)(names, value));
|
|
||||||
};
|
|
||||||
|
|
||||||
function textRemove$1() {
|
|
||||||
this.textContent = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
function textConstant$2(value) {
|
|
||||||
return function() {
|
|
||||||
this.textContent = value;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function textFunction$2(value) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
this.textContent = v == null ? "" : v;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_text$1 = function(value) {
|
|
||||||
return arguments.length
|
|
||||||
? this.each(value == null
|
|
||||||
? textRemove$1 : (typeof value === "function"
|
|
||||||
? textFunction$2
|
|
||||||
: textConstant$2)(value))
|
|
||||||
: this.node().textContent;
|
|
||||||
};
|
|
||||||
|
|
||||||
function htmlRemove$1() {
|
|
||||||
this.innerHTML = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
function htmlConstant$1(value) {
|
|
||||||
return function() {
|
|
||||||
this.innerHTML = value;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function htmlFunction$1(value) {
|
|
||||||
return function() {
|
|
||||||
var v = value.apply(this, arguments);
|
|
||||||
this.innerHTML = v == null ? "" : v;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_html$1 = function(value) {
|
|
||||||
return arguments.length
|
|
||||||
? this.each(value == null
|
|
||||||
? htmlRemove$1 : (typeof value === "function"
|
|
||||||
? htmlFunction$1
|
|
||||||
: htmlConstant$1)(value))
|
|
||||||
: this.node().innerHTML;
|
|
||||||
};
|
|
||||||
|
|
||||||
function raise$2() {
|
|
||||||
if (this.nextSibling) this.parentNode.appendChild(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_raise$1 = function() {
|
|
||||||
return this.each(raise$2);
|
|
||||||
};
|
|
||||||
|
|
||||||
function lower$1() {
|
|
||||||
if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_lower$1 = function() {
|
|
||||||
return this.each(lower$1);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_append$1 = function(name) {
|
|
||||||
var create = typeof name === "function" ? name : creator$1(name);
|
|
||||||
return this.select(function() {
|
|
||||||
return this.appendChild(create.apply(this, arguments));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
function constantNull$1() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_insert$1 = function(name, before) {
|
|
||||||
var create = typeof name === "function" ? name : creator$1(name),
|
|
||||||
select = before == null ? constantNull$1 : typeof before === "function" ? before : selector$1(before);
|
|
||||||
return this.select(function() {
|
|
||||||
return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
function remove$1() {
|
|
||||||
var parent = this.parentNode;
|
|
||||||
if (parent) parent.removeChild(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_remove$1 = function() {
|
|
||||||
return this.each(remove$1);
|
|
||||||
};
|
|
||||||
|
|
||||||
var selection_datum$1 = function(value) {
|
|
||||||
return arguments.length
|
|
||||||
? this.property("__data__", value)
|
|
||||||
: this.node().__data__;
|
|
||||||
};
|
|
||||||
|
|
||||||
function dispatchEvent$1(node, type, params) {
|
|
||||||
var window = defaultView(node),
|
|
||||||
event = window.CustomEvent;
|
|
||||||
|
|
||||||
if (event) {
|
|
||||||
event = new event(type, params);
|
|
||||||
} else {
|
|
||||||
event = window.document.createEvent("Event");
|
|
||||||
if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
|
|
||||||
else event.initEvent(type, false, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
node.dispatchEvent(event);
|
|
||||||
}
|
|
||||||
|
|
||||||
function dispatchConstant$1(type, params) {
|
|
||||||
return function() {
|
|
||||||
return dispatchEvent$1(this, type, params);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function dispatchFunction$1(type, params) {
|
|
||||||
return function() {
|
|
||||||
return dispatchEvent$1(this, type, params.apply(this, arguments));
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
var selection_dispatch$1 = function(type, params) {
|
|
||||||
return this.each((typeof params === "function"
|
|
||||||
? dispatchFunction$1
|
|
||||||
: dispatchConstant$1)(type, params));
|
|
||||||
};
|
|
||||||
|
|
||||||
var root$2 = [null];
|
|
||||||
|
|
||||||
function Selection$2(groups, parents) {
|
|
||||||
this._groups = groups;
|
|
||||||
this._parents = parents;
|
|
||||||
}
|
|
||||||
|
|
||||||
function selection$2() {
|
|
||||||
return new Selection$2([[document.documentElement]], root$2);
|
|
||||||
}
|
|
||||||
|
|
||||||
Selection$2.prototype = selection$2.prototype = {
|
|
||||||
constructor: Selection$2,
|
|
||||||
select: selection_select$1,
|
|
||||||
selectAll: selection_selectAll$1,
|
|
||||||
filter: selection_filter$1,
|
|
||||||
data: selection_data$1,
|
|
||||||
enter: selection_enter$1,
|
|
||||||
exit: selection_exit$1,
|
|
||||||
merge: selection_merge$1,
|
|
||||||
order: selection_order$1,
|
|
||||||
sort: selection_sort$1,
|
|
||||||
call: selection_call$1,
|
|
||||||
nodes: selection_nodes$1,
|
|
||||||
node: selection_node$1,
|
|
||||||
size: selection_size$1,
|
|
||||||
empty: selection_empty$1,
|
|
||||||
each: selection_each$1,
|
|
||||||
attr: selection_attr$1,
|
|
||||||
style: selection_style$1,
|
|
||||||
property: selection_property$1,
|
|
||||||
classed: selection_classed$1,
|
|
||||||
text: selection_text$1,
|
|
||||||
html: selection_html$1,
|
|
||||||
raise: selection_raise$1,
|
|
||||||
lower: selection_lower$1,
|
|
||||||
append: selection_append$1,
|
|
||||||
insert: selection_insert$1,
|
|
||||||
remove: selection_remove$1,
|
|
||||||
datum: selection_datum$1,
|
|
||||||
on: selection_on$1,
|
|
||||||
dispatch: selection_dispatch$1
|
|
||||||
};
|
|
||||||
|
|
||||||
var select$1 = function(selector) {
|
|
||||||
return typeof selector === "string"
|
|
||||||
? new Selection$2([[document.querySelector(selector)]], [document.documentElement])
|
|
||||||
: new Selection$2([[selector]], root$2);
|
|
||||||
};
|
|
||||||
|
|
||||||
var constant$11 = function(x) {
|
|
||||||
return function constant() {
|
return function constant() {
|
||||||
return x;
|
return x;
|
||||||
};
|
};
|
||||||
@@ -7850,7 +7002,7 @@ function y$3(p) {
|
|||||||
var line = function() {
|
var line = function() {
|
||||||
var x$$1 = x$3,
|
var x$$1 = x$3,
|
||||||
y$$1 = y$3,
|
y$$1 = y$3,
|
||||||
defined = constant$11(true),
|
defined = constant$10(true),
|
||||||
context = null,
|
context = null,
|
||||||
curve = curveLinear,
|
curve = curveLinear,
|
||||||
output = null;
|
output = null;
|
||||||
@@ -7876,15 +7028,15 @@ var line = function() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
line.x = function(_) {
|
line.x = function(_) {
|
||||||
return arguments.length ? (x$$1 = typeof _ === "function" ? _ : constant$11(+_), line) : x$$1;
|
return arguments.length ? (x$$1 = typeof _ === "function" ? _ : constant$10(+_), line) : x$$1;
|
||||||
};
|
};
|
||||||
|
|
||||||
line.y = function(_) {
|
line.y = function(_) {
|
||||||
return arguments.length ? (y$$1 = typeof _ === "function" ? _ : constant$11(+_), line) : y$$1;
|
return arguments.length ? (y$$1 = typeof _ === "function" ? _ : constant$10(+_), line) : y$$1;
|
||||||
};
|
};
|
||||||
|
|
||||||
line.defined = function(_) {
|
line.defined = function(_) {
|
||||||
return arguments.length ? (defined = typeof _ === "function" ? _ : constant$11(!!_), line) : defined;
|
return arguments.length ? (defined = typeof _ === "function" ? _ : constant$10(!!_), line) : defined;
|
||||||
};
|
};
|
||||||
|
|
||||||
line.curve = function(_) {
|
line.curve = function(_) {
|
||||||
@@ -7924,7 +7076,7 @@ function slope2(that, t) {
|
|||||||
// According to https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Representations
|
// According to https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Representations
|
||||||
// "you can express cubic Hermite interpolation in terms of cubic Bézier curves
|
// "you can express cubic Hermite interpolation in terms of cubic Bézier curves
|
||||||
// with respect to the four values p0, p0 + m0 / 3, p1 - m1 / 3, p1".
|
// with respect to the four values p0, p0 + m0 / 3, p1 - m1 / 3, p1".
|
||||||
function point$6(that, t0, t1) {
|
function point$5(that, t0, t1) {
|
||||||
var x0 = that._x0,
|
var x0 = that._x0,
|
||||||
y0 = that._y0,
|
y0 = that._y0,
|
||||||
x1 = that._x1,
|
x1 = that._x1,
|
||||||
@@ -7953,7 +7105,7 @@ MonotoneX.prototype = {
|
|||||||
lineEnd: function() {
|
lineEnd: function() {
|
||||||
switch (this._point) {
|
switch (this._point) {
|
||||||
case 2: this._context.lineTo(this._x1, this._y1); break;
|
case 2: this._context.lineTo(this._x1, this._y1); break;
|
||||||
case 3: point$6(this, this._t0, slope2(this, this._t0)); break;
|
case 3: point$5(this, this._t0, slope2(this, this._t0)); break;
|
||||||
}
|
}
|
||||||
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
|
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
|
||||||
this._line = 1 - this._line;
|
this._line = 1 - this._line;
|
||||||
@@ -7966,8 +7118,8 @@ MonotoneX.prototype = {
|
|||||||
switch (this._point) {
|
switch (this._point) {
|
||||||
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
|
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
|
||||||
case 1: this._point = 2; break;
|
case 1: this._point = 2; break;
|
||||||
case 2: this._point = 3; point$6(this, slope2(this, t1 = slope3(this, x, y)), t1); break;
|
case 2: this._point = 3; point$5(this, slope2(this, t1 = slope3(this, x, y)), t1); break;
|
||||||
default: point$6(this, this._t0, t1 = slope3(this, x, y)); break;
|
default: point$5(this, this._t0, t1 = slope3(this, x, y)); break;
|
||||||
}
|
}
|
||||||
|
|
||||||
this._x0 = this._x1, this._x1 = x;
|
this._x0 = this._x1, this._x1 = x;
|
||||||
@@ -8153,7 +7305,7 @@ exports.line = line;
|
|||||||
exports.scaleBand = band;
|
exports.scaleBand = band;
|
||||||
exports.scaleLinear = linear$2;
|
exports.scaleLinear = linear$2;
|
||||||
exports.scaleOrdinal = ordinal;
|
exports.scaleOrdinal = ordinal;
|
||||||
exports.select = select$1;
|
exports.select = select;
|
||||||
|
|
||||||
Object.defineProperty(exports, '__esModule', { value: true });
|
Object.defineProperty(exports, '__esModule', { value: true });
|
||||||
|
|
||||||
|
|||||||
1
devServer.js
Normal file → Executable file
1
devServer.js
Normal file → Executable file
@@ -5,6 +5,7 @@ var config = require('./webpack.config.dev');
|
|||||||
new WebpackDevServer(webpack(config), {
|
new WebpackDevServer(webpack(config), {
|
||||||
publicPath: config.output.publicPath,
|
publicPath: config.output.publicPath,
|
||||||
hot: true,
|
hot: true,
|
||||||
|
disableHostCheck: true,
|
||||||
headers: { "Access-Control-Allow-Origin": "*" },
|
headers: { "Access-Control-Allow-Origin": "*" },
|
||||||
historyApiFallback: {
|
historyApiFallback: {
|
||||||
rewrites: [
|
rewrites: [
|
||||||
|
|||||||
12853
package-lock.json
generated
Normal file
12853
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
22
package.json
22
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "coriolis_shipyard",
|
"name": "coriolis_shipyard",
|
||||||
"version": "2.4.0",
|
"version": "2.9.15",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/EDCD/coriolis"
|
"url": "https://github.com/EDCD/coriolis"
|
||||||
@@ -19,12 +19,14 @@
|
|||||||
"test": "jest",
|
"test": "jest",
|
||||||
"prod-serve": "nginx -p $(pwd) -c nginx.conf",
|
"prod-serve": "nginx -p $(pwd) -c nginx.conf",
|
||||||
"prod-stop": "kill -QUIT $(cat nginx.pid)",
|
"prod-stop": "kill -QUIT $(cat nginx.pid)",
|
||||||
"build": "npm run clean && NODE_ENV=production webpack -p --config webpack.config.prod.js",
|
"build": "npm run clean && cross-env NODE_ENV=production webpack -p --config webpack.config.prod.js",
|
||||||
"rsync": "rsync -ae \"ssh -i $CORIOLIS_PEM\" --delete build/ $CORIOLIS_USER@$CORIOLIS_HOST:~/wwws",
|
"rsync": "rsync -ae \"ssh -i $CORIOLIS_PEM\" --delete build/ $CORIOLIS_USER@$CORIOLIS_HOST:~/wwws",
|
||||||
"deploy": "npm run lint && npm test && npm run build && npm run rsync"
|
"deploy": "npm run lint && npm test && npm run build && npm run rsync"
|
||||||
},
|
},
|
||||||
"jest": {
|
"jest": {
|
||||||
"transform": {".*": "<rootDir>/node_modules/babel-jest"},
|
"transform": {
|
||||||
|
".*": "<rootDir>/node_modules/babel-jest"
|
||||||
|
},
|
||||||
"testRegex": "(/__tests__/test-.*|\\.(test|spec))\\.js$",
|
"testRegex": "(/__tests__/test-.*|\\.(test|spec))\\.js$",
|
||||||
"moduleFileExtensions": [
|
"moduleFileExtensions": [
|
||||||
"js",
|
"js",
|
||||||
@@ -37,6 +39,7 @@
|
|||||||
"<rootDir>/node_modules/lodash",
|
"<rootDir>/node_modules/lodash",
|
||||||
"<rootDir>/node_modules/react",
|
"<rootDir>/node_modules/react",
|
||||||
"<rootDir>/node_modules/react-dom",
|
"<rootDir>/node_modules/react-dom",
|
||||||
|
"<rootDir>/node_modules/react-transition-group",
|
||||||
"<rootDir>/node_modules/react-testutils-additions",
|
"<rootDir>/node_modules/react-testutils-additions",
|
||||||
"<rootDir>/node_modules/fbjs",
|
"<rootDir>/node_modules/fbjs",
|
||||||
"<rootDir>/node_modules/fbemitter",
|
"<rootDir>/node_modules/fbemitter",
|
||||||
@@ -58,10 +61,12 @@
|
|||||||
"babel-eslint": "*",
|
"babel-eslint": "*",
|
||||||
"babel-jest": "*",
|
"babel-jest": "*",
|
||||||
"babel-loader": "*",
|
"babel-loader": "*",
|
||||||
"babel-preset-es2015": "*",
|
"babel-preset-env": "*",
|
||||||
"babel-preset-react": "*",
|
"babel-preset-react": "*",
|
||||||
"babel-preset-stage-0": "*",
|
"babel-preset-stage-0": "*",
|
||||||
|
"create-react-class": "^15.6.2",
|
||||||
"css-loader": "^0.28.0",
|
"css-loader": "^0.28.0",
|
||||||
|
"cross-env": "^5.1.4",
|
||||||
"d3-selection": "1",
|
"d3-selection": "1",
|
||||||
"eslint": "3.19.0",
|
"eslint": "3.19.0",
|
||||||
"eslint-plugin-react": "^6.10.3",
|
"eslint-plugin-react": "^6.10.3",
|
||||||
@@ -70,7 +75,7 @@
|
|||||||
"extract-text-webpack-plugin": "2.1.0",
|
"extract-text-webpack-plugin": "2.1.0",
|
||||||
"file-loader": "^0.11.1",
|
"file-loader": "^0.11.1",
|
||||||
"html-webpack-plugin": "^2.28.0",
|
"html-webpack-plugin": "^2.28.0",
|
||||||
"jest-cli": "^19.0.2",
|
"jest-cli": "^21.2.1",
|
||||||
"jsen": "^0.6.4",
|
"jsen": "^0.6.4",
|
||||||
"json-loader": "^0.5.4",
|
"json-loader": "^0.5.4",
|
||||||
"less": "^2.7.2",
|
"less": "^2.7.2",
|
||||||
@@ -78,20 +83,23 @@
|
|||||||
"react-addons-perf": "^15.4.2",
|
"react-addons-perf": "^15.4.2",
|
||||||
"react-measure": "^1.4.7",
|
"react-measure": "^1.4.7",
|
||||||
"react-testutils-additions": "^15.2.0",
|
"react-testutils-additions": "^15.2.0",
|
||||||
|
"react-transition-group": "^1.1.2",
|
||||||
"rimraf": "^2.6.1",
|
"rimraf": "^2.6.1",
|
||||||
"rollup": "0.41",
|
"rollup": "0.41",
|
||||||
"rollup-plugin-node-resolve": "3",
|
"rollup-plugin-node-resolve": "3",
|
||||||
"style-loader": "^0.16.1",
|
"style-loader": "^0.16.1",
|
||||||
|
"uglify-js": "^2.4.11",
|
||||||
"url-loader": "^0.5.8",
|
"url-loader": "^0.5.8",
|
||||||
"webpack": "^2.4.1",
|
"webpack": "^2.4.1",
|
||||||
"webpack-dev-server": "^2.4.4",
|
"webpack-dev-server": "^2.4.4",
|
||||||
"uglify-js": "^2.4.11"
|
"webpack-notifier": "^1.6.0",
|
||||||
|
"webpack-bugsnag-plugins": "^1.1.1"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"babel-polyfill": "*",
|
"babel-polyfill": "*",
|
||||||
"browserify-zlib-next": "^1.0.1",
|
"browserify-zlib-next": "^1.0.1",
|
||||||
"classnames": "^2.2.5",
|
"classnames": "^2.2.5",
|
||||||
"coriolis-data": "EDCD/coriolis-data",
|
"coriolis-data": "../coriolis-data",
|
||||||
"d3": "4.8.0",
|
"d3": "4.8.0",
|
||||||
"detect-browser": "^1.7.0",
|
"detect-browser": "^1.7.0",
|
||||||
"fbemitter": "^2.1.1",
|
"fbemitter": "^2.1.1",
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import ModalHelp from './components/ModalHelp';
|
|||||||
import ModalImport from './components/ModalImport';
|
import ModalImport from './components/ModalImport';
|
||||||
import ModalPermalink from './components/ModalPermalink';
|
import ModalPermalink from './components/ModalPermalink';
|
||||||
import * as CompanionApiUtils from './utils/CompanionApiUtils';
|
import * as CompanionApiUtils from './utils/CompanionApiUtils';
|
||||||
|
import * as JournalUtils from './utils/JournalUtils';
|
||||||
|
|
||||||
import AboutPage from './pages/AboutPage';
|
import AboutPage from './pages/AboutPage';
|
||||||
import NotFoundPage from './pages/NotFoundPage';
|
import NotFoundPage from './pages/NotFoundPage';
|
||||||
@@ -92,7 +93,14 @@ export default class Coriolis extends React.Component {
|
|||||||
// Need to decode and gunzip the data, then build the ship
|
// Need to decode and gunzip the data, then build the ship
|
||||||
const data = zlib.inflate(new Buffer(r.params.data, 'base64'), { to: 'string' });
|
const data = zlib.inflate(new Buffer(r.params.data, 'base64'), { to: 'string' });
|
||||||
const json = JSON.parse(data);
|
const json = JSON.parse(data);
|
||||||
const ship = CompanionApiUtils.shipFromJson(json);
|
console.log('Ship import data: ');
|
||||||
|
console.log(json);
|
||||||
|
let ship;
|
||||||
|
if (json && json.modules) {
|
||||||
|
ship = CompanionApiUtils.shipFromJson(json);
|
||||||
|
} else if (json && json.Modules) {
|
||||||
|
ship = JournalUtils.shipFromLoadoutJSON(json);
|
||||||
|
}
|
||||||
r.params.ship = ship.id;
|
r.params.ship = ship.id;
|
||||||
r.params.code = ship.toString();
|
r.params.code = ship.toString();
|
||||||
this._setPage(OutfittingPage, r);
|
this._setPage(OutfittingPage, r);
|
||||||
@@ -126,9 +134,9 @@ export default class Coriolis extends React.Component {
|
|||||||
console && console.error && console.error(arguments); // eslint-disable-line no-console
|
console && console.error && console.error(arguments); // eslint-disable-line no-console
|
||||||
if (errObj) {
|
if (errObj) {
|
||||||
if (errObj instanceof Error) {
|
if (errObj instanceof Error) {
|
||||||
Bugsnag.notifyException(errObj) // eslint-disable-line
|
bugsnagClient.notify(errObj) // eslint-disable-line
|
||||||
} else if (errObj instanceof String) {
|
} else if (errObj instanceof String) {
|
||||||
Bugsnag.notify(msg, errObj) // eslint-disable-line
|
bugsnagClient.notify(msg, errObj) // eslint-disable-line
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.setState({
|
this.setState({
|
||||||
@@ -347,6 +355,8 @@ export default class Coriolis extends React.Component {
|
|||||||
<footer>
|
<footer>
|
||||||
<div className="right cap">
|
<div className="right cap">
|
||||||
<a href="https://github.com/EDCD/coriolis" target="_blank" title="Coriolis Github Project">{window.CORIOLIS_VERSION} - {window.CORIOLIS_DATE}</a>
|
<a href="https://github.com/EDCD/coriolis" target="_blank" title="Coriolis Github Project">{window.CORIOLIS_VERSION} - {window.CORIOLIS_DATE}</a>
|
||||||
|
<br/>
|
||||||
|
<a href={"https://github.com/EDCD/coriolis/compare/edcd:develop@{" + window.CORIOLIS_DATE + "}...edcd:develop"} target="_blank" title={"Coriolis Commits since" + window.CORIOLIS_DATE}>Commits since last release ({window.CORIOLIS_DATE})</a>
|
||||||
</div>
|
</div>
|
||||||
</footer>
|
</footer>
|
||||||
</div>;
|
</div>;
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ const GRPCAT = {
|
|||||||
'fx': 'limpet controllers',
|
'fx': 'limpet controllers',
|
||||||
'hb': 'limpet controllers',
|
'hb': 'limpet controllers',
|
||||||
'pc': 'limpet controllers',
|
'pc': 'limpet controllers',
|
||||||
|
'rpl': 'limpet controllers',
|
||||||
'pce': 'passenger cabins',
|
'pce': 'passenger cabins',
|
||||||
'pci': 'passenger cabins',
|
'pci': 'passenger cabins',
|
||||||
'pcm': 'passenger cabins',
|
'pcm': 'passenger cabins',
|
||||||
@@ -36,10 +37,14 @@ const GRPCAT = {
|
|||||||
'ml': 'lasers',
|
'ml': 'lasers',
|
||||||
'c': 'projectiles',
|
'c': 'projectiles',
|
||||||
'mc': 'projectiles',
|
'mc': 'projectiles',
|
||||||
|
'axmc': 'experimental',
|
||||||
'fc': 'projectiles',
|
'fc': 'projectiles',
|
||||||
|
'rfl': 'experimental',
|
||||||
'pa': 'projectiles',
|
'pa': 'projectiles',
|
||||||
'rg': 'projectiles',
|
'rg': 'projectiles',
|
||||||
'mr': 'ordnance',
|
'mr': 'ordnance',
|
||||||
|
'axmr': 'experimental',
|
||||||
|
'rcpl': 'experimental',
|
||||||
'tp': 'ordnance',
|
'tp': 'ordnance',
|
||||||
'nl': 'ordnance',
|
'nl': 'ordnance',
|
||||||
'sc': 'scanners',
|
'sc': 'scanners',
|
||||||
@@ -48,9 +53,15 @@ const GRPCAT = {
|
|||||||
'cs': 'scanners',
|
'cs': 'scanners',
|
||||||
'kw': 'scanners',
|
'kw': 'scanners',
|
||||||
'ws': 'scanners',
|
'ws': 'scanners',
|
||||||
|
'xs': 'scanners',
|
||||||
'ch': 'defence',
|
'ch': 'defence',
|
||||||
'po': 'defence',
|
'po': 'defence',
|
||||||
'ec': 'defence',
|
'ec': 'defence',
|
||||||
|
'sfn': 'defence',
|
||||||
|
// Standard
|
||||||
|
'gpp': 'guardian',
|
||||||
|
'gpc': 'guardian',
|
||||||
|
'ggc': 'guardian'
|
||||||
};
|
};
|
||||||
// Order here is the order in which items will be shown in the modules menu
|
// Order here is the order in which items will be shown in the modules menu
|
||||||
const CATEGORIES = {
|
const CATEGORIES = {
|
||||||
@@ -60,7 +71,7 @@ const CATEGORIES = {
|
|||||||
'fi': ['fi'],
|
'fi': ['fi'],
|
||||||
'fuel': ['ft', 'fs'],
|
'fuel': ['ft', 'fs'],
|
||||||
'hangars': ['fh', 'pv'],
|
'hangars': ['fh', 'pv'],
|
||||||
'limpet controllers': ['cc', 'fx', 'hb', 'pc'],
|
'limpet controllers': ['cc', 'fx', 'hb', 'pc', 'rpl'],
|
||||||
'passenger cabins': ['pce', 'pci', 'pcm', 'pcq'],
|
'passenger cabins': ['pce', 'pci', 'pcm', 'pcq'],
|
||||||
'rf': ['rf'],
|
'rf': ['rf'],
|
||||||
'shields': ['sg', 'bsg', 'psg', 'scb'],
|
'shields': ['sg', 'bsg', 'psg', 'scb'],
|
||||||
@@ -75,6 +86,11 @@ const CATEGORIES = {
|
|||||||
'hs': ['hs'],
|
'hs': ['hs'],
|
||||||
'defence': ['ch', 'po', 'ec'],
|
'defence': ['ch', 'po', 'ec'],
|
||||||
'scanners': ['sc', 'ss', 'cs', 'kw', 'ws'], // Overloaded with internal scanners
|
'scanners': ['sc', 'ss', 'cs', 'kw', 'ws'], // Overloaded with internal scanners
|
||||||
|
// Experimental
|
||||||
|
'experimental': ['axmc', 'axmr', 'rfl', 'xs', 'sfn', 'rcpl'],
|
||||||
|
|
||||||
|
// Guardian
|
||||||
|
'guardian': ['gpp', 'gpc', 'ggc']
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -88,7 +104,11 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
diffDetails: PropTypes.func,
|
diffDetails: PropTypes.func,
|
||||||
m: PropTypes.object,
|
m: PropTypes.object,
|
||||||
shipMass: PropTypes.number,
|
shipMass: PropTypes.number,
|
||||||
warning: PropTypes.func
|
warning: PropTypes.func,
|
||||||
|
firstSlotId: PropTypes.string,
|
||||||
|
lastSlotId: PropTypes.string,
|
||||||
|
activeSlotId: PropTypes.string,
|
||||||
|
slotDiv: PropTypes.object
|
||||||
};
|
};
|
||||||
|
|
||||||
static defaultProps = {
|
static defaultProps = {
|
||||||
@@ -104,6 +124,7 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
super(props);
|
super(props);
|
||||||
this._hideDiff = this._hideDiff.bind(this);
|
this._hideDiff = this._hideDiff.bind(this);
|
||||||
this.state = this._initState(props, context);
|
this.state = this._initState(props, context);
|
||||||
|
this.slotItems = [];// Array to hold <li> refs.
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -114,8 +135,9 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
*/
|
*/
|
||||||
_initState(props, context) {
|
_initState(props, context) {
|
||||||
let translate = context.language.translate;
|
let translate = context.language.translate;
|
||||||
let { m, warning, shipMass, onSelect, modules } = props;
|
let { m, warning, shipMass, onSelect, modules, firstSlotId, lastSlotId } = props;
|
||||||
let list, currentGroup;
|
let list, currentGroup;
|
||||||
|
|
||||||
let buildGroup = this._buildGroup.bind(
|
let buildGroup = this._buildGroup.bind(
|
||||||
this,
|
this,
|
||||||
translate,
|
translate,
|
||||||
@@ -126,7 +148,7 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
this._hideDiff(event);
|
this._hideDiff(event);
|
||||||
onSelect(m);
|
onSelect(m);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
if (modules instanceof Array) {
|
if (modules instanceof Array) {
|
||||||
list = buildGroup(modules[0].grp, modules);
|
list = buildGroup(modules[0].grp, modules);
|
||||||
@@ -134,7 +156,10 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
list = [];
|
list = [];
|
||||||
// At present time slots with grouped options (Hardpoints and Internal) can be empty
|
// At present time slots with grouped options (Hardpoints and Internal) can be empty
|
||||||
if (m) {
|
if (m) {
|
||||||
list.push(<div className='empty-c upp' key='empty' onClick={onSelect.bind(null, null)} >{translate('empty')}</div>);
|
let emptyId = 'empty';
|
||||||
|
if(this.firstSlotId == null) this.firstSlotId = emptyId;
|
||||||
|
let keyDown = this._keyDown.bind(this, onSelect);
|
||||||
|
list.push(<div className='empty-c upp' key={emptyId} data-id={emptyId} onClick={onSelect.bind(null, null)} onKeyDown={keyDown} tabIndex="0" ref={slotItem => this.slotItems[emptyId] = slotItem} >{translate('empty')}</div>);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Need to regroup the modules by our own categorisation
|
// Need to regroup the modules by our own categorisation
|
||||||
@@ -184,37 +209,38 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
let trackingFocus = false;
|
||||||
return { list, currentGroup };
|
return { list, currentGroup, trackingFocus};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate React Components for Module Group
|
* Generate React Components for Module Group
|
||||||
* @param {Function} translate Translate function
|
* @param {Function} translate Translate function
|
||||||
* @param {Objecy} mountedModule Mounted Module
|
* @param {Object} mountedModule Mounted Module
|
||||||
* @param {Funciton} warningFunc Warning function
|
* @param {Function} warningFunc Warning function
|
||||||
* @param {number} mass Mass
|
* @param {number} mass Mass
|
||||||
* @param {function} onSelect Select/Mount callback
|
* @param {function} onSelect Select/Mount callback
|
||||||
* @param {string} grp Group name
|
* @param {string} grp Group name
|
||||||
* @param {Array} modules Available modules
|
* @param {Array} modules Available modules
|
||||||
* @return {React.Component} Available Module Group contents
|
* @return {React.Component} Available Module Group contents
|
||||||
*/
|
*/
|
||||||
_buildGroup(translate, mountedModule, warningFunc, mass, onSelect, grp, modules) {
|
_buildGroup(translate, mountedModule, warningFunc, mass, onSelect, grp, modules, firstSlotId, lastSlotId) {
|
||||||
let prevClass = null, prevRating = null;
|
let prevClass = null, prevRating = null, prevName;
|
||||||
let elems = [];
|
let elems = [];
|
||||||
|
|
||||||
const sortedModules = modules.sort(this._moduleOrder);
|
const sortedModules = modules.sort(this._moduleOrder);
|
||||||
|
|
||||||
|
|
||||||
// Calculate the number of items per class. Used so we don't have long lists with only a few items in each row
|
// Calculate the number of items per class. Used so we don't have long lists with only a few items in each row
|
||||||
const tmp = sortedModules.map((v, i) => v['class']).reduce((count, cls) => { count[cls] = ++count[cls] || 1; return count; }, {});
|
const tmp = sortedModules.map((v, i) => v['class']).reduce((count, cls) => { count[cls] = ++count[cls] || 1; return count; }, {});
|
||||||
const itemsPerClass = Math.max.apply(null, Object.keys(tmp).map(key => tmp[key]));
|
const itemsPerClass = Math.max.apply(null, Object.keys(tmp).map(key => tmp[key]));
|
||||||
|
|
||||||
let itemsOnThisRow = 0;
|
let itemsOnThisRow = 0;
|
||||||
|
|
||||||
for (let i = 0; i < sortedModules.length; i++) {
|
for (let i = 0; i < sortedModules.length; i++) {
|
||||||
let m = sortedModules[i];
|
let m = sortedModules[i];
|
||||||
let mount = null;
|
let mount = null;
|
||||||
let disabled = false;
|
let disabled = false;
|
||||||
|
prevName = m.name
|
||||||
if (ModuleUtils.isShieldGenerator(m.grp)) {
|
if (ModuleUtils.isShieldGenerator(m.grp)) {
|
||||||
// Shield generators care about maximum hull mass
|
// Shield generators care about maximum hull mass
|
||||||
disabled = mass > m.maxmass;
|
disabled = mass > m.maxmass;
|
||||||
@@ -230,9 +256,21 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
});
|
});
|
||||||
let eventHandlers;
|
let eventHandlers;
|
||||||
|
|
||||||
if (disabled || active) {
|
if (disabled) {
|
||||||
eventHandlers = {};
|
eventHandlers = {
|
||||||
|
onKeyDown: this._keyDown.bind(this, null),
|
||||||
|
onKeyUp: this._keyUp.bind(this, null)
|
||||||
|
|
||||||
|
};
|
||||||
} else {
|
} else {
|
||||||
|
/**
|
||||||
|
* Get the ids of the first and last <li> elements in the <ul> that are focusable (i.e. are not active or disabled)
|
||||||
|
* Will be used to keep focus inside the <ul> on Tab and Shift-Tab while it is visible
|
||||||
|
*/
|
||||||
|
if (this.firstSlotId == null) this.firstSlotId = sortedModules[i].id;
|
||||||
|
if (active) this.activeSlotId = sortedModules[i].id;
|
||||||
|
this.lastSlotId = sortedModules[i].id;
|
||||||
|
|
||||||
let showDiff = this._showDiff.bind(this, mountedModule, m);
|
let showDiff = this._showDiff.bind(this, mountedModule, m);
|
||||||
let select = onSelect.bind(null, m);
|
let select = onSelect.bind(null, m);
|
||||||
|
|
||||||
@@ -241,7 +279,9 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
onTouchStart: this._touchStart.bind(this, showDiff),
|
onTouchStart: this._touchStart.bind(this, showDiff),
|
||||||
onTouchEnd: this._touchEnd.bind(this, select),
|
onTouchEnd: this._touchEnd.bind(this, select),
|
||||||
onMouseLeave: this._hideDiff,
|
onMouseLeave: this._hideDiff,
|
||||||
onClick: select
|
onClick: select,
|
||||||
|
onKeyDown: this._keyDown.bind(this, select),
|
||||||
|
onKeyUp: this._keyUp.bind(this, select)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -250,24 +290,28 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
case 'G': mount = <MountGimballed className={'lg'}/>; break;
|
case 'G': mount = <MountGimballed className={'lg'}/>; break;
|
||||||
case 'T': mount = <MountTurret className={'lg'}/>; break;
|
case 'T': mount = <MountTurret className={'lg'}/>; break;
|
||||||
}
|
}
|
||||||
|
if (m.name && m.name === prevName) {
|
||||||
|
// elems.push(<br key={'b' + m.grp + i} />);
|
||||||
|
itemsOnThisRow = 0;
|
||||||
|
}
|
||||||
if (itemsOnThisRow == 6 || i > 0 && sortedModules.length > 3 && itemsPerClass > 2 && m.class != prevClass && (m.rating != prevRating || m.mount)) {
|
if (itemsOnThisRow == 6 || i > 0 && sortedModules.length > 3 && itemsPerClass > 2 && m.class != prevClass && (m.rating != prevRating || m.mount)) {
|
||||||
elems.push(<br key={'b' + m.grp + i} />);
|
elems.push(<br key={'b' + m.grp + i} />);
|
||||||
itemsOnThisRow = 0;
|
itemsOnThisRow = 0;
|
||||||
}
|
}
|
||||||
|
let tbIdx = (classes.indexOf('disabled') < 0) ? 0 : undefined;
|
||||||
elems.push(
|
elems.push(
|
||||||
<li key={m.id} className={classes} {...eventHandlers}>
|
<li key={m.id} data-id={m.id} className={classes} {...eventHandlers} tabIndex={tbIdx} ref={slotItem => this.slotItems[m.id] = slotItem}>
|
||||||
{mount}
|
{mount}
|
||||||
{(mount ? ' ' : '') + m.class + m.rating + (m.missile ? '/' + m.missile : '') + (m.name ? ' ' + translate(m.name) : '')}
|
{(mount ? ' ' : '') + m.class + m.rating + (m.missile ? '/' + m.missile : '') + (m.name ? ' ' + translate(m.name) : '')}
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
|
|
||||||
itemsOnThisRow++;
|
itemsOnThisRow++;
|
||||||
prevClass = m.class;
|
prevClass = m.class;
|
||||||
prevRating = m.rating;
|
prevRating = m.rating;
|
||||||
|
prevName = m.name;
|
||||||
}
|
}
|
||||||
|
return <ul key={'modules' + grp}>{elems}</ul>;
|
||||||
return <ul key={'modules' + grp} >{elems}</ul>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -318,6 +362,41 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
this._hideDiff();
|
this._hideDiff();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key down - select module on Enter key, move to next/previous module on Tab/Shift-Tab, close on Esc
|
||||||
|
* @param {Function} select Select module callback
|
||||||
|
* @param {SyntheticEvent} event Event
|
||||||
|
*/
|
||||||
|
|
||||||
|
_keyDown(select, event) {
|
||||||
|
var className = event.currentTarget.attributes['class'].value;
|
||||||
|
if (event.key == 'Enter' && className.indexOf('disabled') < 0 && className.indexOf('active') < 0) {
|
||||||
|
select();
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var elemId = event.currentTarget.attributes['data-id'].value;
|
||||||
|
if (className.indexOf('disabled') < 0 && event.key == 'Tab') {
|
||||||
|
if (event.shiftKey && elemId == this.firstSlotId) {
|
||||||
|
event.preventDefault();
|
||||||
|
this.slotItems[this.lastSlotId].focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!event.shiftKey && elemId == this.lastSlotId) {
|
||||||
|
event.preventDefault();
|
||||||
|
this.slotItems[this.firstSlotId].focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key Up
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
_keyUp(select,event) {
|
||||||
|
//nothing here yet
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hide diff tooltip
|
* Hide diff tooltip
|
||||||
* @param {SyntheticEvent} event Event
|
* @param {SyntheticEvent} event Event
|
||||||
@@ -375,6 +454,23 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
if (this.groupElem) { // Scroll to currently selected group
|
if (this.groupElem) { // Scroll to currently selected group
|
||||||
this.node.scrollTop = this.groupElem.offsetTop;
|
this.node.scrollTop = this.groupElem.offsetTop;
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Set focus on active or first slot element, if applicable.
|
||||||
|
*/
|
||||||
|
if (this.slotItems[this.activeSlotId]) {
|
||||||
|
this.slotItems[this.activeSlotId].focus();
|
||||||
|
} else if (this.slotItems[this.firstSlotId]) {
|
||||||
|
this.slotItems[this.firstSlotId].focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
/**
|
||||||
|
* Set focus to slot element ref (if we have one) after modules component unmounts
|
||||||
|
*/
|
||||||
|
if(this.props.slotDiv) {
|
||||||
|
this.props.slotDiv.focus();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -391,6 +487,7 @@ export default class AvailableModulesMenu extends TranslatedComponent {
|
|||||||
* @return {React.Component} List
|
* @return {React.Component} List
|
||||||
*/
|
*/
|
||||||
render() {
|
render() {
|
||||||
|
console.log("Tracking focus? " + this.state.trackingFocus);
|
||||||
return (
|
return (
|
||||||
<div ref={node => this.node = node}
|
<div ref={node => this.node = node}
|
||||||
className={cn('select', this.props.className)}
|
className={cn('select', this.props.className)}
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ export default class EngagementRange extends TranslatedComponent {
|
|||||||
|
|
||||||
const { ship } = props;
|
const { ship } = props;
|
||||||
|
|
||||||
const maxRange = this._calcMaxRange(ship);
|
const maxRange = Math.round(this._calcMaxRange(ship));
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
maxRange
|
maxRange
|
||||||
|
|||||||
@@ -97,11 +97,12 @@ export default class HardpointSlot extends Slot {
|
|||||||
{ showModuleResistances && m.getKineticResistance() ? <div className='l'>{translate('kinres')}: {formats.pct(m.getKineticResistance())}</div> : null }
|
{ showModuleResistances && m.getKineticResistance() ? <div className='l'>{translate('kinres')}: {formats.pct(m.getKineticResistance())}</div> : null }
|
||||||
{ showModuleResistances && m.getThermalResistance() ? <div className='l'>{translate('thermres')}: {formats.pct(m.getThermalResistance())}</div> : null }
|
{ showModuleResistances && m.getThermalResistance() ? <div className='l'>{translate('thermres')}: {formats.pct(m.getThermalResistance())}</div> : null }
|
||||||
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
||||||
{ m && validMods.length > 0 ? <div className='r' ><button onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
{ m && validMods.length > 0 ? <div className='r' tabIndex="0" ref={ modButton => this.modButton = modButton }><button tabIndex="-1" onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
||||||
</div>
|
</div>
|
||||||
</div>;
|
</div>;
|
||||||
} else {
|
} else {
|
||||||
return <div className={'empty'}>{translate('empty')}</div>;
|
return <div className={'empty'}>{translate('empty')}</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -505,6 +505,9 @@ export default class Header extends TranslatedComponent {
|
|||||||
return (
|
return (
|
||||||
<header>
|
<header>
|
||||||
{this.props.appCacheUpdate && <div id="app-update" onClick={() => window.location.reload() }>{translate('PHRASE_UPDATE_RDY')}</div>}
|
{this.props.appCacheUpdate && <div id="app-update" onClick={() => window.location.reload() }>{translate('PHRASE_UPDATE_RDY')}</div>}
|
||||||
|
{this.props.appCache ? <a className={"view-changes"} href={"https://github.com/EDCD/coriolis/compare/edcd:develop@{" + window.CORIOLIS_DATE + "}...edcd:develop"} target="_blank">
|
||||||
|
{"View Release Changes"}
|
||||||
|
</a> : null}
|
||||||
<Link className='l' href='/' style={{ marginRight: '1em' }} title='Home'><CoriolisLogo className='icon xl' /></Link>
|
<Link className='l' href='/' style={{ marginRight: '1em' }} title='Home'><CoriolisLogo className='icon xl' /></Link>
|
||||||
|
|
||||||
<div className='l menu'>
|
<div className='l menu'>
|
||||||
|
|||||||
@@ -21,18 +21,23 @@ export default class InternalSlot extends Slot {
|
|||||||
* @param {Object} u Localized Units Map
|
* @param {Object} u Localized Units Map
|
||||||
* @return {React.Component} Slot contents
|
* @return {React.Component} Slot contents
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
_getSlotDetails(m, enabled, translate, formats, u) {
|
_getSlotDetails(m, enabled, translate, formats, u) {
|
||||||
if (m) {
|
if (m) {
|
||||||
let classRating = m.class + m.rating;
|
let classRating = m.class + m.rating;
|
||||||
let { drag, drop, ship } = this.props;
|
let { drag, drop, ship } = this.props;
|
||||||
let { termtip, tooltip } = this.context;
|
let { termtip, tooltip } = this.context;
|
||||||
let validMods = Modifications.modules[m.grp].modifications || [];
|
let validMods = (Modifications.modules[m.grp] ? Modifications.modules[m.grp].modifications : []);
|
||||||
let showModuleResistances = Persist.showModuleResistances();
|
let showModuleResistances = Persist.showModuleResistances();
|
||||||
|
|
||||||
// Modifications tooltip shows blueprint and grade, if available
|
// Modifications tooltip shows blueprint and grade, if available
|
||||||
let modTT = translate('modified');
|
let modTT = translate('modified');
|
||||||
if (m && m.blueprint && m.blueprint.name) {
|
if (m && m.blueprint && m.blueprint.name) {
|
||||||
modTT = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
modTT = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
||||||
|
if (m.blueprint.special && m.blueprint.special.id >= 0) {
|
||||||
|
modTT += ', ' + translate(m.blueprint.special.name);
|
||||||
|
}
|
||||||
modTT = (
|
modTT = (
|
||||||
<div>
|
<div>
|
||||||
<div>{modTT}</div>
|
<div>{modTT}</div>
|
||||||
@@ -67,6 +72,7 @@ export default class InternalSlot extends Slot {
|
|||||||
{ m.getRange() ? <div className={'l'}>{translate('range')} {formats.f2(m.getRange())}{u.km}</div> : null }
|
{ m.getRange() ? <div className={'l'}>{translate('range')} {formats.f2(m.getRange())}{u.km}</div> : null }
|
||||||
{ m.getRangeT() ? <div className={'l'}>{translate('ranget')} {formats.f1(m.getRangeT())}{u.s}</div> : null }
|
{ m.getRangeT() ? <div className={'l'}>{translate('ranget')} {formats.f1(m.getRangeT())}{u.s}</div> : null }
|
||||||
{ m.getTime() ? <div className={'l'}>{translate('time')}: {formats.time(m.getTime())}</div> : null }
|
{ m.getTime() ? <div className={'l'}>{translate('time')}: {formats.time(m.getTime())}</div> : null }
|
||||||
|
{ m.getHackTime() ? <div className={'l'}>{translate('hacktime')}: {formats.time(m.getHackTime())}</div> : null }
|
||||||
{ m.maximum ? <div className={'l'}>{translate('max')}: {(m.maximum)}</div> : null }
|
{ m.maximum ? <div className={'l'}>{translate('max')}: {(m.maximum)}</div> : null }
|
||||||
{ m.rangeLS ? <div className={'l'}>{translate('range')}: {m.rangeLS}{u.Ls}</div> : null }
|
{ m.rangeLS ? <div className={'l'}>{translate('range')}: {m.rangeLS}{u.Ls}</div> : null }
|
||||||
{ m.rangeLS === null ? <div className={'l'}>∞{u.Ls}</div> : null }
|
{ m.rangeLS === null ? <div className={'l'}>∞{u.Ls}</div> : null }
|
||||||
@@ -81,7 +87,7 @@ export default class InternalSlot extends Slot {
|
|||||||
{ m.getHullReinforcement() ? <div className='l'>{translate('armour')}: {formats.int(m.getHullReinforcement() + ship.baseArmour * m.getModValue('hullboost') / 10000)}</div> : null }
|
{ m.getHullReinforcement() ? <div className='l'>{translate('armour')}: {formats.int(m.getHullReinforcement() + ship.baseArmour * m.getModValue('hullboost') / 10000)}</div> : null }
|
||||||
{ m.getProtection() ? <div className='l'>{translate('protection')}: {formats.rPct(m.getProtection())}</div> : null }
|
{ m.getProtection() ? <div className='l'>{translate('protection')}: {formats.rPct(m.getProtection())}</div> : null }
|
||||||
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
||||||
{ m && validMods.length > 0 ? <div className='r' ><button onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
{ m && validMods.length > 0 ? <div className='r' tabIndex="0" ref={ modButton => this.modButton = modButton }><button tabIndex="-1" onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
||||||
</div>
|
</div>
|
||||||
</div>;
|
</div>;
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -14,7 +14,10 @@ export default class Modification extends TranslatedComponent {
|
|||||||
m: PropTypes.object.isRequired,
|
m: PropTypes.object.isRequired,
|
||||||
name: PropTypes.string.isRequired,
|
name: PropTypes.string.isRequired,
|
||||||
value: PropTypes.number.isRequired,
|
value: PropTypes.number.isRequired,
|
||||||
onChange: PropTypes.func.isRequired
|
onChange: PropTypes.func.isRequired,
|
||||||
|
onKeyDown: PropTypes.func.isRequired,
|
||||||
|
modItems: PropTypes.array.isRequired,
|
||||||
|
handleModChange: PropTypes.func.isRequired
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -36,7 +39,6 @@ export default class Modification extends TranslatedComponent {
|
|||||||
*/
|
*/
|
||||||
_updateValue(value) {
|
_updateValue(value) {
|
||||||
const name = this.props.name;
|
const name = this.props.name;
|
||||||
|
|
||||||
let scaledValue = Math.round(Number(value) * 100);
|
let scaledValue = Math.round(Number(value) * 100);
|
||||||
// Limit to +1000% / -99.99%
|
// Limit to +1000% / -99.99%
|
||||||
if (scaledValue > 100000) {
|
if (scaledValue > 100000) {
|
||||||
@@ -57,9 +59,15 @@ export default class Modification extends TranslatedComponent {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Triggered when an update to slider value is finished i.e. when losing focus
|
* Triggered when an update to slider value is finished i.e. when losing focus
|
||||||
|
*
|
||||||
|
* pnellesen (24/05/2018): added value check below - this should prevent experimental effects from being recalculated
|
||||||
|
* with each onBlur event, even when no change has actually been made to the field.
|
||||||
*/
|
*/
|
||||||
_updateFinished() {
|
_updateFinished() {
|
||||||
this.props.onChange();
|
if (this.props.value != this.state.value) {
|
||||||
|
this.props.handleModChange(true);
|
||||||
|
this.props.onChange();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -86,9 +94,9 @@ export default class Modification extends TranslatedComponent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div onBlur={this._updateFinished.bind(this)} className={'cb'} key={name}>
|
<div onBlur={this._updateFinished.bind(this)} className={'cb'} key={name} ref={ modItem => this.props.modItems[name] = modItem }>
|
||||||
<div className={'cb'}>{translate(name, m.grp)}{symbol}</div>
|
<div className={'cb'}>{translate(name, m.grp)}{symbol}</div>
|
||||||
<NumberEditor className={'cb'} style={{ width: '90%', textAlign: 'center' }} step={0.01} stepModifier={1} decimals={2} value={this.state.value} onValueChange={this._updateValue.bind(this)} />
|
<NumberEditor className={'cb'} style={{ width: '90%', textAlign: 'center' }} step={0.01} stepModifier={1} decimals={2} value={this.state.value} onValueChange={this._updateValue.bind(this)} onKeyDown={ this.props.onKeyDown } />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,14 @@ import { isEmpty, stopCtxPropagation } from '../utils/UtilityFunctions';
|
|||||||
import cn from 'classnames';
|
import cn from 'classnames';
|
||||||
import { Modifications } from 'coriolis-data/dist';
|
import { Modifications } from 'coriolis-data/dist';
|
||||||
import Modification from './Modification';
|
import Modification from './Modification';
|
||||||
import { getBlueprint, blueprintTooltip, setWorst, setBest, setExtreme, setRandom } from '../utils/BlueprintFunctions';
|
import {
|
||||||
|
getBlueprint,
|
||||||
|
blueprintTooltip,
|
||||||
|
setPercent,
|
||||||
|
getPercent,
|
||||||
|
setRandom,
|
||||||
|
specialToolTip
|
||||||
|
} from '../utils/BlueprintFunctions'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Modifications menu
|
* Modifications menu
|
||||||
@@ -17,7 +24,8 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
ship: PropTypes.object.isRequired,
|
ship: PropTypes.object.isRequired,
|
||||||
m: PropTypes.object.isRequired,
|
m: PropTypes.object.isRequired,
|
||||||
marker: PropTypes.string.isRequired,
|
marker: PropTypes.string.isRequired,
|
||||||
onChange: PropTypes.func.isRequired
|
onChange: PropTypes.func.isRequired,
|
||||||
|
modButton:PropTypes.object
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -30,14 +38,24 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
|
|
||||||
this._toggleBlueprintsMenu = this._toggleBlueprintsMenu.bind(this);
|
this._toggleBlueprintsMenu = this._toggleBlueprintsMenu.bind(this);
|
||||||
this._toggleSpecialsMenu = this._toggleSpecialsMenu.bind(this);
|
this._toggleSpecialsMenu = this._toggleSpecialsMenu.bind(this);
|
||||||
this._rollWorst = this._rollWorst.bind(this);
|
this._rollFifty = this._rollFifty.bind(this);
|
||||||
this._rollRandom = this._rollRandom.bind(this);
|
this._rollRandom = this._rollRandom.bind(this);
|
||||||
this._rollBest = this._rollBest.bind(this);
|
this._rollBest = this._rollBest.bind(this);
|
||||||
this._rollExtreme = this._rollExtreme.bind(this);
|
this._rollWorst = this._rollWorst.bind(this);
|
||||||
this._reset = this._reset.bind(this);
|
this._reset = this._reset.bind(this);
|
||||||
|
|
||||||
|
this._keyDown = this._keyDown.bind(this);
|
||||||
|
|
||||||
|
this.modItems = [];// Array to hold various element refs (<li>, <div>, <ul>, etc.)
|
||||||
|
this.firstModId = null;
|
||||||
|
this.firstBPLabel = null;// First item in mod menu
|
||||||
|
this.lastModId = null;
|
||||||
|
this.lastNeId = null;//Last number editor id. Used to set focus to last number editor when shift-tab pressed on first element in mod menu.
|
||||||
|
this.modValDidChange = false; //used to determine if component update was caused by change in modification value.
|
||||||
|
this._handleModChange = this._handleModChange.bind(this);
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
blueprintMenuOpened: false,
|
blueprintMenuOpened: !(props.m.blueprint && props.m.blueprint.name),
|
||||||
specialMenuOpened: false
|
specialMenuOpened: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -52,7 +70,6 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
const { m } = props;
|
const { m } = props;
|
||||||
const { language, tooltip, termtip } = context;
|
const { language, tooltip, termtip } = context;
|
||||||
const translate = language.translate;
|
const translate = language.translate;
|
||||||
|
|
||||||
const blueprints = [];
|
const blueprints = [];
|
||||||
for (const blueprintName in Modifications.modules[m.grp].blueprints) {
|
for (const blueprintName in Modifications.modules[m.grp].blueprints) {
|
||||||
const blueprint = getBlueprint(blueprintName, m);
|
const blueprint = getBlueprint(blueprintName, m);
|
||||||
@@ -66,9 +83,14 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
const close = this._blueprintSelected.bind(this, blueprintName, grade);
|
const close = this._blueprintSelected.bind(this, blueprintName, grade);
|
||||||
const key = blueprintName + ':' + grade;
|
const key = blueprintName + ':' + grade;
|
||||||
const tooltipContent = blueprintTooltip(translate, blueprint.grades[grade], Modifications.modules[m.grp].blueprints[blueprintName].grades[grade].engineers, m.grp);
|
const tooltipContent = blueprintTooltip(translate, blueprint.grades[grade], Modifications.modules[m.grp].blueprints[blueprintName].grades[grade].engineers, m.grp);
|
||||||
blueprintGrades.unshift(<li key={key} className={classes} style={{ width: '2em' }} onMouseOver={termtip.bind(null, tooltipContent)} onMouseOut={tooltip.bind(null, null)} onClick={close}>{grade}</li>);
|
|
||||||
|
|
||||||
|
blueprintGrades.unshift(<li key={key} tabIndex="0" data-id={key} className={classes} style={{ width: '2em' }} onMouseOver={termtip.bind(null, tooltipContent)} onMouseOut={tooltip.bind(null, null)} onClick={close} onKeyDown={this._keyDown} ref={modItem => this.modItems[key] = modItem}>{grade}</li>);
|
||||||
}
|
}
|
||||||
if (blueprintGrades) {
|
if (blueprintGrades) {
|
||||||
|
const thisLen = blueprintGrades.length;
|
||||||
|
if (this.firstModId == null) this.firstModId = blueprintGrades[0].key;
|
||||||
|
this.lastModId = blueprintGrades[thisLen-1].key;
|
||||||
blueprints.push(<div key={blueprint.name} className={'select-group cap'}>{translate(blueprint.name)}</div>);
|
blueprints.push(<div key={blueprint.name} className={'select-group cap'}>{translate(blueprint.name)}</div>);
|
||||||
blueprints.push(<ul key={blueprintName}>{blueprintGrades}</ul>);
|
blueprints.push(<ul key={blueprintName}>{blueprintGrades}</ul>);
|
||||||
}
|
}
|
||||||
@@ -76,6 +98,68 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
return blueprints;
|
return blueprints;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key down - select module on Enter key, move to next/previous module on Tab/Shift-Tab, close on Esc
|
||||||
|
* @param {Function} select Select module callback
|
||||||
|
* @param {SyntheticEvent} event Event
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
_keyDown(event) {
|
||||||
|
var className = null;
|
||||||
|
var elemId = null;
|
||||||
|
if (event.currentTarget.attributes['class']) className = event.currentTarget.attributes['class'].value;
|
||||||
|
if (event.currentTarget.attributes['data-id']) elemId = event.currentTarget.attributes['data-id'].value;
|
||||||
|
|
||||||
|
if (event.key == 'Enter' && className.indexOf('disabled') < 0 && className.indexOf('active') < 0) {
|
||||||
|
event.stopPropagation();
|
||||||
|
if (elemId != null) {
|
||||||
|
this.modItems[elemId].click();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
event.currentTarget.click();
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (event.key == 'Tab') {
|
||||||
|
//Shift-Tab
|
||||||
|
if(event.shiftKey) {
|
||||||
|
if (elemId == this.firstModId && elemId != null) {
|
||||||
|
// Initial modification menu
|
||||||
|
event.preventDefault();
|
||||||
|
this.modItems[this.lastModId].focus();
|
||||||
|
return;
|
||||||
|
} else if (event.currentTarget.className.indexOf("button-inline-menu") >= 0 && event.currentTarget.previousElementSibling == null && this.lastNeId != null && this.modItems[this.lastNeId] != null) {
|
||||||
|
// shift-tab on first element in modifications menu. set focus to last number editor field if open
|
||||||
|
event.preventDefault();
|
||||||
|
this.modItems[this.lastNeId].lastChild.focus();
|
||||||
|
return;
|
||||||
|
} else if (event.currentTarget.className.indexOf("button-inline-menu") >= 0 && event.currentTarget.previousElementSibling == null) {
|
||||||
|
// shift-tab on button-inline-menu with no number editor
|
||||||
|
event.preventDefault();
|
||||||
|
event.currentTarget.parentElement.lastElementChild.focus();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (elemId == this.lastModId && elemId != null) {
|
||||||
|
// Initial modification menu
|
||||||
|
event.preventDefault();
|
||||||
|
this.modItems[this.firstModId].focus();
|
||||||
|
return;
|
||||||
|
} else if (event.currentTarget.className.indexOf("button-inline-menu") >= 0 && event.currentTarget.nextSibling == null && event.currentTarget.nodeName != "TD") {
|
||||||
|
// Experimental menu
|
||||||
|
event.preventDefault();
|
||||||
|
event.currentTarget.parentElement.firstElementChild.focus();
|
||||||
|
return;
|
||||||
|
} else if (event.currentTarget.className == 'cb' && event.currentTarget.parentElement.nextSibling == null) {
|
||||||
|
event.preventDefault();
|
||||||
|
this.modItems[this.firstBPLabel].focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the specials
|
* Render the specials
|
||||||
* @param {Object} props React component properties
|
* @param {Object} props React component properties
|
||||||
@@ -83,20 +167,40 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
* @return {Object} list: Array of React Components
|
* @return {Object} list: Array of React Components
|
||||||
*/
|
*/
|
||||||
_renderSpecials(props, context) {
|
_renderSpecials(props, context) {
|
||||||
|
|
||||||
const { m } = props;
|
const { m } = props;
|
||||||
const { language, tooltip, termtip } = context;
|
const { language, tooltip, termtip } = context;
|
||||||
const translate = language.translate;
|
const translate = language.translate;
|
||||||
|
|
||||||
const specials = [];
|
const specials = [];
|
||||||
const specialsId = m.missile && Modifications.modules[m.grp]['specials_' + m.missile] ? 'specials_' + m.missile : 'specials';
|
const specialsId = m.missile && Modifications.modules[m.grp]['specials_' + m.missile] ? 'specials_' + m.missile : 'specials';
|
||||||
if (Modifications.modules[m.grp][specialsId] && Modifications.modules[m.grp][specialsId].length > 0) {
|
if (Modifications.modules[m.grp][specialsId] && Modifications.modules[m.grp][specialsId].length > 0) {
|
||||||
const close = this._specialSelected.bind(this, null);
|
const close = this._specialSelected.bind(this, null);
|
||||||
specials.push(<div style={{ cursor: 'pointer' }} key={ 'none' } onClick={ close }>{translate('PHRASE_NO_SPECIAL')}</div>);
|
specials.push(<div tabIndex="0" style={{ cursor: 'pointer', fontWeight: 'bold' }} className={ 'button-inline-menu warning' } key={ 'none' } data-id={ 'none' } onClick={ close } onKeyDown={this._keyDown} ref={modItem => this.modItems['none'] = modItem}>{translate('PHRASE_NO_SPECIAL')}</div>);
|
||||||
for (const specialName of Modifications.modules[m.grp][specialsId]) {
|
for (const specialName of Modifications.modules[m.grp][specialsId]) {
|
||||||
|
if (Modifications.specials[specialName].name.search('Legacy') >= 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const classes = cn('button-inline-menu', {
|
||||||
|
active: m.blueprint && m.blueprint.special && m.blueprint.special.edname == specialName
|
||||||
|
});
|
||||||
const close = this._specialSelected.bind(this, specialName);
|
const close = this._specialSelected.bind(this, specialName);
|
||||||
specials.push(<div style={{ cursor: 'pointer' }} key={ specialName } onClick={ close }>{translate(Modifications.specials[specialName].name)}</div>);
|
if (m.blueprint && m.blueprint.name) {
|
||||||
|
let tmp = {};
|
||||||
|
if (m.blueprint.special) {
|
||||||
|
tmp = m.blueprint.special;
|
||||||
|
} else {
|
||||||
|
tmp = undefined;
|
||||||
|
}
|
||||||
|
m.blueprint.special = Modifications.specials[specialName];
|
||||||
|
let specialTt = specialToolTip(translate, m.blueprint.grades[m.blueprint.grade], m.grp, m, specialName);
|
||||||
|
m.blueprint.special = tmp;
|
||||||
|
specials.push(<div tabIndex="0" style={{ cursor: 'pointer' }} className={classes} key={ specialName } data-id={ specialName } onMouseOver={termtip.bind(null, specialTt)} onMouseOut={tooltip.bind(null, null)} onClick={ close } onKeyDown={this._keyDown} ref={modItem => this.modItems[specialName] = modItem}>{translate(Modifications.specials[specialName].name)}</div>);
|
||||||
|
} else {
|
||||||
|
specials.push(<div tabIndex="0" style={{ cursor: 'pointer' }} className={classes} key={ specialName } data-id={ specialName }onClick={ close } onKeyDown={this._keyDown} ref={modItem => this.modItems[specialName] = modItem}>{translate(Modifications.specials[specialName].name)}</div>);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
console.log("_renderSpecials. specials: %O", specials);
|
||||||
return specials;
|
return specials;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -111,9 +215,11 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
for (const modName of Modifications.modules[m.grp].modifications) {
|
for (const modName of Modifications.modules[m.grp].modifications) {
|
||||||
if (!Modifications.modifications[modName].hidden) {
|
if (!Modifications.modifications[modName].hidden) {
|
||||||
const key = modName + (m.getModValue(modName) / 100 || 0);
|
const key = modName + (m.getModValue(modName) / 100 || 0);
|
||||||
modifications.push(<Modification key={ key } ship={ ship } m={ m } name={ modName } value={ m.getModValue(modName) / 100 || 0 } onChange={ onChange }/>);
|
this.lastNeId = modName;
|
||||||
|
modifications.push(<Modification key={ key } ship={ ship } m={ m } name={ modName } value={ m.getModValue(modName) / 100 || 0 } onChange={ onChange } onKeyDown={ this._keyDown } modItems={ this.modItems } handleModChange = {this._handleModChange} />);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
console.log("_renderModifications. modItems: %O", this.modItems);
|
||||||
return modifications;
|
return modifications;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,8 +242,9 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
const blueprint = getBlueprint(fdname, m);
|
const blueprint = getBlueprint(fdname, m);
|
||||||
blueprint.grade = grade;
|
blueprint.grade = grade;
|
||||||
ship.setModuleBlueprint(m, blueprint);
|
ship.setModuleBlueprint(m, blueprint);
|
||||||
|
setPercent(ship, m, 100);
|
||||||
|
|
||||||
this.setState({ blueprintMenuOpened: false });
|
this.setState({ blueprintMenuOpened: false, specialMenuOpened: true });
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -168,11 +275,15 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provide a 'worst' roll within the information we have
|
* Provide a '50%' roll within the information we have
|
||||||
*/
|
*/
|
||||||
_rollWorst() {
|
_rollFifty() {
|
||||||
const { m, ship } = this.props;
|
const { m, ship } = this.props;
|
||||||
setWorst(ship, m);
|
setPercent(ship, m, 50);
|
||||||
|
|
||||||
|
// this will change the values in the modifications. Set modDidChange to true to prevent focus change when component updates
|
||||||
|
this._handleModChange(true);
|
||||||
|
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -182,6 +293,10 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
_rollRandom() {
|
_rollRandom() {
|
||||||
const { m, ship } = this.props;
|
const { m, ship } = this.props;
|
||||||
setRandom(ship, m);
|
setRandom(ship, m);
|
||||||
|
|
||||||
|
// this will change the values in the modifications. Set modDidChange to true to prevent focus change when component updates
|
||||||
|
this._handleModChange(true);
|
||||||
|
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -190,17 +305,27 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
*/
|
*/
|
||||||
_rollBest() {
|
_rollBest() {
|
||||||
const { m, ship } = this.props;
|
const { m, ship } = this.props;
|
||||||
setBest(ship, m);
|
setPercent(ship, m, 100);
|
||||||
|
|
||||||
|
// this will change the values in the modifications. Set modDidChange to true to prevent focus change when component updates
|
||||||
|
this._handleModChange(true);
|
||||||
|
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provide an 'extreme' roll within the information we have
|
* Provide a 'worst' roll within the information we have
|
||||||
*/
|
*/
|
||||||
_rollExtreme() {
|
_rollWorst() {
|
||||||
const { m, ship } = this.props;
|
const { m, ship } = this.props;
|
||||||
setExtreme(ship, m);
|
setPercent(ship, m, 0);
|
||||||
|
|
||||||
|
// this will change the values in the modifications. Set modDidChange to true to prevent focus change when component updates
|
||||||
|
this._handleModChange(true);
|
||||||
|
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -210,10 +335,56 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
const { m, ship } = this.props;
|
const { m, ship } = this.props;
|
||||||
ship.clearModifications(m);
|
ship.clearModifications(m);
|
||||||
ship.clearModuleBlueprint(m);
|
ship.clearModuleBlueprint(m);
|
||||||
|
|
||||||
this.props.onChange();
|
this.props.onChange();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* set mod did change boolean
|
||||||
|
*/
|
||||||
|
_handleModChange(b) {
|
||||||
|
this.modValDidChange = b;
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
/**
|
||||||
|
* Set focus on first element in modifications menu
|
||||||
|
* after it first mounts
|
||||||
|
*/
|
||||||
|
let firstEleCn = this.modItems['modMainDiv'].children.length > 0 ? this.modItems['modMainDiv'].children[0].className : null;
|
||||||
|
if (firstEleCn.indexOf('select-group cap') >= 0) {
|
||||||
|
this.modItems['modMainDiv'].children[1].firstElementChild.focus();
|
||||||
|
} else {
|
||||||
|
this.modItems['modMainDiv'].firstElementChild.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate() {
|
||||||
|
/**
|
||||||
|
* Set focus on first element in modifications menu
|
||||||
|
* if component updates, unless update is due to value change
|
||||||
|
* in a modification
|
||||||
|
*/
|
||||||
|
if (!this.modValDidChange) {
|
||||||
|
if (this.modItems['modMainDiv'].children.length > 0) {
|
||||||
|
let firstEleCn = this.modItems['modMainDiv'].children[0].className;
|
||||||
|
if (firstEleCn.indexOf('button-inline-menu') >= 0) {
|
||||||
|
this.modItems['modMainDiv'].firstElementChild.focus();
|
||||||
|
} else if (firstEleCn.indexOf('select-group cap') >= 0) {
|
||||||
|
this.modItems['modMainDiv'].children[1].firstElementChild.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this._handleModChange(false);//Need to reset if component update due to value change
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
if (this.props.modButton) {
|
||||||
|
this.props.modButton.focus();// set focus to the modification menu icon after mod menu is unmounted.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the list
|
* Render the list
|
||||||
* @return {React.Component} List
|
* @return {React.Component} List
|
||||||
@@ -226,66 +397,82 @@ export default class ModificationsMenu extends TranslatedComponent {
|
|||||||
|
|
||||||
const _toggleBlueprintsMenu = this._toggleBlueprintsMenu;
|
const _toggleBlueprintsMenu = this._toggleBlueprintsMenu;
|
||||||
const _toggleSpecialsMenu = this._toggleSpecialsMenu;
|
const _toggleSpecialsMenu = this._toggleSpecialsMenu;
|
||||||
const _rollBest = this._rollBest;
|
const _rollFull = this._rollBest;
|
||||||
const _rollExtreme = this._rollExtreme;
|
|
||||||
const _rollWorst = this._rollWorst;
|
const _rollWorst = this._rollWorst;
|
||||||
|
const _rollFifty = this._rollFifty;
|
||||||
const _rollRandom = this._rollRandom;
|
const _rollRandom = this._rollRandom;
|
||||||
const _reset = this._reset;
|
const _reset = this._reset;
|
||||||
|
|
||||||
let blueprintLabel;
|
let blueprintLabel;
|
||||||
let haveBlueprint = false;
|
let haveBlueprint = false;
|
||||||
let blueprintTt;
|
let blueprintTt;
|
||||||
if (m.blueprint && m.blueprint.name) {
|
let blueprintCv;
|
||||||
|
//TODO: Fix this to actually find the correct blueprint.
|
||||||
|
if (!m.blueprint || !m.blueprint.name || !m.blueprint.fdname || !Modifications.modules[m.grp].blueprints || !Modifications.modules[m.grp].blueprints[m.blueprint.fdname]) {
|
||||||
|
this.props.ship.clearModuleBlueprint(m);
|
||||||
|
this.props.ship.clearModuleSpecial(m);
|
||||||
|
}
|
||||||
|
if (m.blueprint && m.blueprint.name && Modifications.modules[m.grp].blueprints[m.blueprint.fdname].grades[m.blueprint.grade]) {
|
||||||
blueprintLabel = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
blueprintLabel = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
||||||
haveBlueprint = true;
|
haveBlueprint = true;
|
||||||
blueprintTt = blueprintTooltip(translate, m.blueprint.grades[m.blueprint.grade], Modifications.modules[m.grp].blueprints[m.blueprint.fdname].grades[m.blueprint.grade].engineers, m.grp);
|
blueprintTt = blueprintTooltip(translate, m.blueprint.grades[m.blueprint.grade], Modifications.modules[m.grp].blueprints[m.blueprint.fdname].grades[m.blueprint.grade].engineers, m.grp);
|
||||||
|
blueprintCv = getPercent(m);
|
||||||
}
|
}
|
||||||
|
|
||||||
let specialLabel;
|
let specialLabel;
|
||||||
|
let specialTt;
|
||||||
if (m.blueprint && m.blueprint.special) {
|
if (m.blueprint && m.blueprint.special) {
|
||||||
specialLabel = m.blueprint.special.name;
|
specialLabel = m.blueprint.special.name;
|
||||||
|
specialTt = specialToolTip(translate, m.blueprint.grades[m.blueprint.grade], m.grp, m, m.blueprint.special.edname);
|
||||||
} else {
|
} else {
|
||||||
specialLabel = translate('PHRASE_SELECT_SPECIAL');
|
specialLabel = translate('PHRASE_SELECT_SPECIAL');
|
||||||
}
|
}
|
||||||
|
|
||||||
const specials = this._renderSpecials(this.props, this.context);
|
const specials = this._renderSpecials(this.props, this.context);
|
||||||
|
/**
|
||||||
|
* pnellesen - 05/28/2018 - added additional checks for specials.length below to ensure menus
|
||||||
|
* display correctly in cases where there are no specials (ex: AFMUs.)
|
||||||
|
*/
|
||||||
const showBlueprintsMenu = blueprintMenuOpened;
|
const showBlueprintsMenu = blueprintMenuOpened;
|
||||||
const showSpecial = haveBlueprint && specials.length && !blueprintMenuOpened;
|
const showSpecial = haveBlueprint && specials.length && !blueprintMenuOpened;
|
||||||
const showSpecialsMenu = specialMenuOpened;
|
const showSpecialsMenu = specialMenuOpened && specials.length;
|
||||||
const showRolls = haveBlueprint && !blueprintMenuOpened && !specialMenuOpened;
|
const showRolls = haveBlueprint && !blueprintMenuOpened && (!specialMenuOpened || !specials.length);
|
||||||
const showReset = !blueprintMenuOpened && !specialMenuOpened;
|
const showReset = !blueprintMenuOpened && (!specialMenuOpened || !specials.length) && haveBlueprint;
|
||||||
const showMods = !blueprintMenuOpened && !specialMenuOpened;
|
const showMods = !blueprintMenuOpened && (!specialMenuOpened || !specials.length) && haveBlueprint;
|
||||||
|
if (haveBlueprint) {
|
||||||
|
this.firstBPLabel = blueprintLabel
|
||||||
|
} else {
|
||||||
|
this.firstBPLabel = 'selectBP';
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={cn('select', this.props.className)}
|
className={cn('select', this.props.className)}
|
||||||
onClick={(e) => e.stopPropagation() }
|
onClick={(e) => e.stopPropagation() }
|
||||||
onContextMenu={stopCtxPropagation}
|
onContextMenu={stopCtxPropagation}
|
||||||
|
ref={modItem => this.modItems['modMainDiv'] = modItem}
|
||||||
>
|
>
|
||||||
{ showBlueprintsMenu ? '' : haveBlueprint ?
|
{ showBlueprintsMenu | showSpecialsMenu ? '' : haveBlueprint ?
|
||||||
<div className={ cn('section-menu', { selected: blueprintMenuOpened })} style={{ cursor: 'pointer' }} onMouseOver={termtip.bind(null, blueprintTt)} onMouseOut={tooltip.bind(null, null)} onClick={_toggleBlueprintsMenu}>{blueprintLabel}</div> :
|
<div tabIndex="0" className={ cn('section-menu button-inline-menu', { selected: blueprintMenuOpened })} style={{ cursor: 'pointer' }} onMouseOver={termtip.bind(null, blueprintTt)} onMouseOut={tooltip.bind(null, null)} onClick={_toggleBlueprintsMenu} onKeyDown={ this._keyDown } ref={modItems => this.modItems[this.firstBPLabel] = modItems}>{blueprintLabel}</div> :
|
||||||
<div className={ cn('section-menu', { selected: blueprintMenuOpened })} style={{ cursor: 'pointer' }} onClick={_toggleBlueprintsMenu}>{translate('PHRASE_SELECT_BLUEPRINT')}</div> }
|
<div tabIndex="0" className={ cn('section-menu button-inline-menu', { selected: blueprintMenuOpened })} style={{ cursor: 'pointer' }} onClick={_toggleBlueprintsMenu} onKeyDown={ this._keyDown } ref={modItems => this.modItems[this.firstBPLabel] = modItems}>{translate('PHRASE_SELECT_BLUEPRINT')}</div> }
|
||||||
{ showBlueprintsMenu ? this._renderBlueprints(this.props, this.context) : null }
|
{ showBlueprintsMenu ? this._renderBlueprints(this.props, this.context) : null }
|
||||||
{ showSpecial ? <div className={ cn('section-menu', { selected: specialMenuOpened })} style={{ cursor: 'pointer' }} onClick={_toggleSpecialsMenu}>{specialLabel}</div> : null }
|
{ showSpecial & !showSpecialsMenu ? <div tabIndex="0" className={ cn('section-menu button-inline-menu', { selected: specialMenuOpened })} style={{ cursor: 'pointer' }} onMouseOver={specialTt ? termtip.bind(null, specialTt) : null} onMouseOut={specialTt ? tooltip.bind(null, null) : null} onClick={_toggleSpecialsMenu} onKeyDown={ this._keyDown }>{specialLabel}</div> : null }
|
||||||
{ showSpecialsMenu ? specials : null }
|
{ showSpecialsMenu ? specials : null }
|
||||||
{ showRolls || showReset ?
|
{ showReset ? <div tabIndex="0" className={'section-menu button-inline-menu warning'} style={{ cursor: 'pointer' }} onClick={_reset} onKeyDown={ this._keyDown } onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_RESET')} onMouseOut={tooltip.bind(null, null)}> { translate('reset') } </div> : null }
|
||||||
|
{ showRolls ?
|
||||||
|
|
||||||
<table style={{ width: '100%', backgroundColor: 'transparent' }}>
|
<table style={{ width: '100%', backgroundColor: 'transparent' }}>
|
||||||
<tbody>
|
<tbody>
|
||||||
{ showRolls ?
|
{ showRolls ?
|
||||||
<tr>
|
<tr>
|
||||||
<td> { translate('roll') }: </td>
|
<td tabIndex="0" className={ cn('section-menu button-inline-menu', {active: false})}> { translate('roll') }: </td>
|
||||||
<td style={{ cursor: 'pointer' }} onClick={_rollWorst} onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_WORST')} onMouseOut={tooltip.bind(null, null)}> { translate('worst') } </td>
|
<td tabIndex="0" className={ cn('section-menu button-inline-menu', { active: blueprintCv === 0 })} style={{ cursor: 'pointer' }} onClick={_rollWorst} onKeyDown={ this._keyDown } onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_WORST')} onMouseOut={tooltip.bind(null, null)}> { translate('0%') } </td>
|
||||||
<td style={{ cursor: 'pointer' }} onClick={_rollBest}onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_BEST')} onMouseOut={tooltip.bind(null, null)}> { translate('best') } </td>
|
<td tabIndex="0" className={ cn('section-menu button-inline-menu', { active: blueprintCv === 50 })} style={{ cursor: 'pointer' }} onClick={_rollFifty} onKeyDown={ this._keyDown } onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_FIFTY')} onMouseOut={tooltip.bind(null, null)}> { translate('50%') } </td>
|
||||||
<td style={{ cursor: 'pointer' }} onClick={_rollExtreme}onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_EXTREME')} onMouseOut={tooltip.bind(null, null)}> { translate('extreme') } </td>
|
<td tabIndex="0" className={ cn('section-menu button-inline-menu', { active: blueprintCv === 100 })} style={{ cursor: 'pointer' }} onClick={_rollFull} onKeyDown={ this._keyDown } onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_BEST')} onMouseOut={tooltip.bind(null, null)}> { translate('100%') } </td>
|
||||||
<td style={{ cursor: 'pointer' }} onClick={_rollRandom} onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_RANDOM')} onMouseOut={tooltip.bind(null, null)}> { translate('random') } </td>
|
<td tabIndex="0" className={ cn('section-menu button-inline-menu', { active: blueprintCv === null || blueprintCv % 50 != 0 })} style={{ cursor: 'pointer' }} onClick={_rollRandom} onKeyDown={ this._keyDown } onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_RANDOM')} onMouseOut={tooltip.bind(null, null)}> { translate('random') } </td>
|
||||||
</tr> : null }
|
|
||||||
{ showReset ?
|
|
||||||
<tr>
|
|
||||||
<td colSpan={'5'} style={{ cursor: 'pointer' }} onClick={_reset}onMouseOver={termtip.bind(null, 'PHRASE_BLUEPRINT_RESET')} onMouseOut={tooltip.bind(null, null)}> { translate('reset') } </td>
|
|
||||||
</tr> : null }
|
</tr> : null }
|
||||||
</tbody>
|
</tbody>
|
||||||
</table> : null }
|
</table> : null }
|
||||||
|
{ showMods ? <hr /> : null }
|
||||||
{ showMods ?
|
{ showMods ?
|
||||||
<span onMouseOver={termtip.bind(null, 'HELP_MODIFICATIONS_MENU')} onMouseOut={tooltip.bind(null, null)} >
|
<span onMouseOver={termtip.bind(null, 'HELP_MODIFICATIONS_MENU')} onMouseOut={tooltip.bind(null, null)} >
|
||||||
{ this._renderModifications(this.props) }
|
{ this._renderModifications(this.props) }
|
||||||
|
|||||||
@@ -15,22 +15,30 @@ export default class ShipSummaryTable extends TranslatedComponent {
|
|||||||
cargo: PropTypes.number.isRequired,
|
cargo: PropTypes.number.isRequired,
|
||||||
fuel: PropTypes.number.isRequired,
|
fuel: PropTypes.number.isRequired,
|
||||||
marker: PropTypes.string.isRequired,
|
marker: PropTypes.string.isRequired,
|
||||||
|
pips: PropTypes.object.isRequired
|
||||||
};
|
};
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props)
|
||||||
|
this.didContextChange = this.didContextChange.bind(this);
|
||||||
|
this.state = {
|
||||||
|
shieldColour: 'blue'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the table
|
* Render the table
|
||||||
* @return {React.Component} Summary table
|
* @return {React.Component} Summary table
|
||||||
*/
|
*/
|
||||||
render() {
|
render() {
|
||||||
const { ship, cargo, fuel } = this.props;
|
const { ship, cargo, fuel, pips } = this.props;
|
||||||
let { language, tooltip, termtip } = this.context;
|
let { language, tooltip, termtip } = this.context;
|
||||||
let translate = language.translate;
|
let translate = language.translate;
|
||||||
let u = language.units;
|
let u = language.units;
|
||||||
let formats = language.formats;
|
let formats = language.formats;
|
||||||
let { time, int, round, f1, f2 } = formats;
|
let { time, int, round, f1, f2 } = formats;
|
||||||
let hide = tooltip.bind(null, null);
|
let hide = tooltip.bind(null, null);
|
||||||
|
const shieldGenerator = ship.findInternalByGroup('sg') || ship.findInternalByGroup('psg');
|
||||||
const shieldGenerator = ship.findInternalByGroup('sg');
|
|
||||||
const sgClassNames = cn({ warning: shieldGenerator && !ship.shield, muted: !shieldGenerator });
|
const sgClassNames = cn({ warning: shieldGenerator && !ship.shield, muted: !shieldGenerator });
|
||||||
const sgTooltip = shieldGenerator ? 'TT_SUMMARY_SHIELDS' : 'TT_SUMMARY_SHIELDS_NONFUNCTIONAL';
|
const sgTooltip = shieldGenerator ? 'TT_SUMMARY_SHIELDS' : 'TT_SUMMARY_SHIELDS_NONFUNCTIONAL';
|
||||||
const timeToDrain = Calc.timeToDrainWep(ship, 4);
|
const timeToDrain = Calc.timeToDrainWep(ship, 4);
|
||||||
@@ -38,9 +46,19 @@ export default class ShipSummaryTable extends TranslatedComponent {
|
|||||||
const speedTooltip = canThrust ? 'TT_SUMMARY_SPEED' : 'TT_SUMMARY_SPEED_NONFUNCTIONAL';
|
const speedTooltip = canThrust ? 'TT_SUMMARY_SPEED' : 'TT_SUMMARY_SPEED_NONFUNCTIONAL';
|
||||||
const canBoost = ship.canBoost(cargo, ship.fuelCapacity);
|
const canBoost = ship.canBoost(cargo, ship.fuelCapacity);
|
||||||
const boostTooltip = canBoost ? 'TT_SUMMARY_BOOST' : canThrust ? 'TT_SUMMARY_BOOST_NONFUNCTIONAL' : 'TT_SUMMARY_SPEED_NONFUNCTIONAL';
|
const boostTooltip = canBoost ? 'TT_SUMMARY_BOOST' : canThrust ? 'TT_SUMMARY_BOOST_NONFUNCTIONAL' : 'TT_SUMMARY_SPEED_NONFUNCTIONAL';
|
||||||
|
const sgMetrics = Calc.shieldMetrics(ship, pips.sys || 2);
|
||||||
|
const armourMetrics = Calc.armourMetrics(ship);
|
||||||
|
let shieldColour = 'blue';
|
||||||
|
if (shieldGenerator && shieldGenerator.m.grp === 'psg') {
|
||||||
|
shieldColour = 'green';
|
||||||
|
} else if (shieldGenerator && shieldGenerator.m.grp === 'bsg') {
|
||||||
|
shieldColour = 'purple';
|
||||||
|
}
|
||||||
|
this.state = {
|
||||||
|
shieldColour
|
||||||
|
}
|
||||||
return <div id='summary'>
|
return <div id='summary'>
|
||||||
<table id='summaryTable'>
|
<table className={'summaryTable'}>
|
||||||
<thead>
|
<thead>
|
||||||
<tr className='main'>
|
<tr className='main'>
|
||||||
<th rowSpan={2} className={ cn({ 'bg-warning-disabled': !canThrust }) }>{translate('speed')}</th>
|
<th rowSpan={2} className={ cn({ 'bg-warning-disabled': !canThrust }) }>{translate('speed')}</th>
|
||||||
@@ -53,6 +71,7 @@ export default class ShipSummaryTable extends TranslatedComponent {
|
|||||||
<th rowSpan={2}>{translate('TTD')}</th>
|
<th rowSpan={2}>{translate('TTD')}</th>
|
||||||
{/* <th onMouseEnter={termtip.bind(null, 'heat per second')} onMouseLeave={hide} rowSpan={2}>{translate('HPS')}</th> */}
|
{/* <th onMouseEnter={termtip.bind(null, 'heat per second')} onMouseLeave={hide} rowSpan={2}>{translate('HPS')}</th> */}
|
||||||
<th rowSpan={2}>{translate('cargo')}</th>
|
<th rowSpan={2}>{translate('cargo')}</th>
|
||||||
|
<th rowSpan={2}>{translate('pax')}</th>
|
||||||
<th rowSpan={2}>{translate('fuel')}</th>
|
<th rowSpan={2}>{translate('fuel')}</th>
|
||||||
<th colSpan={3}>{translate('mass')}</th>
|
<th colSpan={3}>{translate('mass')}</th>
|
||||||
<th onMouseEnter={termtip.bind(null, 'hull hardness', { cap: 0 })} onMouseLeave={hide} rowSpan={2}>{translate('hrd')}</th>
|
<th onMouseEnter={termtip.bind(null, 'hull hardness', { cap: 0 })} onMouseLeave={hide} rowSpan={2}>{translate('hrd')}</th>
|
||||||
@@ -86,6 +105,7 @@ export default class ShipSummaryTable extends TranslatedComponent {
|
|||||||
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_TTD', { cap: 0 })} onMouseLeave={hide}>{timeToDrain === Infinity ? '∞' : time(timeToDrain)}</td>
|
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_TTD', { cap: 0 })} onMouseLeave={hide}>{timeToDrain === Infinity ? '∞' : time(timeToDrain)}</td>
|
||||||
{/* <td>{f1(ship.totalHps)}</td> */}
|
{/* <td>{f1(ship.totalHps)}</td> */}
|
||||||
<td>{round(ship.cargoCapacity)}{u.T}</td>
|
<td>{round(ship.cargoCapacity)}{u.T}</td>
|
||||||
|
<td>{ship.passengerCapacity}</td>
|
||||||
<td>{round(ship.fuelCapacity)}{u.T}</td>
|
<td>{round(ship.fuelCapacity)}{u.T}</td>
|
||||||
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_HULL_MASS', { cap: 0 })} onMouseLeave={hide}>{ship.hullMass}{u.T}</td>
|
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_HULL_MASS', { cap: 0 })} onMouseLeave={hide}>{ship.hullMass}{u.T}</td>
|
||||||
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_UNLADEN_MASS', { cap: 0 })} onMouseLeave={hide}>{int(ship.unladenMass)}{u.T}</td>
|
<td onMouseEnter={termtip.bind(null, 'TT_SUMMARY_UNLADEN_MASS', { cap: 0 })} onMouseLeave={hide}>{int(ship.unladenMass)}{u.T}</td>
|
||||||
@@ -96,6 +116,70 @@ export default class ShipSummaryTable extends TranslatedComponent {
|
|||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
<table className={'summaryTable'}>
|
||||||
|
<thead className={this.state.shieldColour}>
|
||||||
|
<tr>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'shield', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('shield')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'explres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('explres')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'kinres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('kinres')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'thermres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('thermres')}</th>
|
||||||
|
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'TT_SUMMARY_SHIELDS_SCB', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('absolute')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'TT_SUMMARY_SHIELDS_SCB', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('explosive')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'TT_SUMMARY_SHIELDS_SCB', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('kinetic')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'TT_SUMMARY_SHIELDS_SCB', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('thermal')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_SG_RECOVER', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('recovery')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_SG_RECHARGE', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('recharge')}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>{translate(shieldGenerator && shieldGenerator.m.grp || 'No Shield')}</td>
|
||||||
|
<td>{int(ship.shieldExplRes * 100) + '%'}</td>
|
||||||
|
<td>{int(ship.shieldKinRes * 100) + '%'}</td>
|
||||||
|
<td>{int(ship.shieldThermRes * 100) + '%'}</td>
|
||||||
|
<td>{int(sgMetrics && sgMetrics.generator ? sgMetrics.total / sgMetrics.absolute.total : 0)}</td>
|
||||||
|
<td>{int(sgMetrics && sgMetrics.generator ? sgMetrics.total / sgMetrics.explosive.total : 0)}</td>
|
||||||
|
<td>{int(sgMetrics && sgMetrics.generator ? sgMetrics.total / sgMetrics.kinetic.total : 0 )}</td>
|
||||||
|
<td>{int(sgMetrics && sgMetrics.generator ? sgMetrics.total / sgMetrics.thermal.total : 0 )}</td>
|
||||||
|
<td>{sgMetrics && sgMetrics.recover ? formats.time(sgMetrics.recover) : 0}</td>
|
||||||
|
<td>{sgMetrics && sgMetrics.recharge ? formats.time(sgMetrics.recharge) : 0}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'armour', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('armour')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'explres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('explres')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'kinres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('kinres')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'thermres', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('thermres')}</th>
|
||||||
|
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_EFFECTIVE_ARMOUR', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('absolute')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_EFFECTIVE_ARMOUR', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('explosive')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_EFFECTIVE_ARMOUR', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('kinetic')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'PHRASE_EFFECTIVE_ARMOUR', { cap: 0 })} onMouseLeave={hide} className='lft'>{`${translate('thermal')} ${translate('HP')}`}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'TT_MODULE_ARMOUR', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('raw module armour')}</th>
|
||||||
|
<th onMouseEnter={termtip.bind(null, 'armour', { cap: 0 })} onMouseLeave={hide} className='lft'>{translate('internal protection')}</th>
|
||||||
|
|
||||||
|
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>{translate(ship && ship.bulkheads && ship.bulkheads.m && ship.bulkheads.m.name || 'No Armour')}</td>
|
||||||
|
<td>{int(ship.hullExplRes * 100) + '%'}</td>
|
||||||
|
<td>{int(ship.hullKinRes * 100) + '%'}</td>
|
||||||
|
<td>{int(ship.hullThermRes * 100) + '%'}</td>
|
||||||
|
<td>{int(armourMetrics.total / armourMetrics.absolute.total)}</td>
|
||||||
|
<td>{int(armourMetrics.total / armourMetrics.explosive.total)}</td>
|
||||||
|
<td>{int(armourMetrics.total / armourMetrics.kinetic.total)}</td>
|
||||||
|
<td>{int(armourMetrics.total / armourMetrics.thermal.total)}</td>
|
||||||
|
<td>{int(armourMetrics.modulearmour)}</td>
|
||||||
|
<td>{int(armourMetrics.moduleprotection * 100) + '%'}</td>
|
||||||
|
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,12 +17,12 @@ export default class Slider extends React.Component {
|
|||||||
|
|
||||||
static propTypes = {
|
static propTypes = {
|
||||||
axis: PropTypes.bool,
|
axis: PropTypes.bool,
|
||||||
axisUnit: PropTypes.string,
|
axisUnit: PropTypes.string,//units (T, M, etc.)
|
||||||
max: PropTypes.number,
|
max: PropTypes.number,
|
||||||
min: PropTypes.number,
|
min: PropTypes.number,
|
||||||
onChange: PropTypes.func.isRequired,
|
onChange: PropTypes.func.isRequired,// function which determins percent value
|
||||||
onResize: PropTypes.func,
|
onResize: PropTypes.func,
|
||||||
percent: PropTypes.number.isRequired,
|
percent: PropTypes.number.isRequired,//value of slider
|
||||||
scale: PropTypes.number
|
scale: PropTypes.number
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -35,6 +35,11 @@ export default class Slider extends React.Component {
|
|||||||
this._down = this._down.bind(this);
|
this._down = this._down.bind(this);
|
||||||
this._move = this._move.bind(this);
|
this._move = this._move.bind(this);
|
||||||
this._up = this._up.bind(this);
|
this._up = this._up.bind(this);
|
||||||
|
this._keyup = this._keyup.bind(this);
|
||||||
|
this._keydown = this._keydown.bind(this);
|
||||||
|
this._touchstart = this._touchstart.bind(this);
|
||||||
|
this._touchend = this._touchend.bind(this);
|
||||||
|
|
||||||
this._updatePercent = this._updatePercent.bind(this);
|
this._updatePercent = this._updatePercent.bind(this);
|
||||||
this._updateDimensions = this._updateDimensions.bind(this);
|
this._updateDimensions = this._updateDimensions.bind(this);
|
||||||
|
|
||||||
@@ -46,10 +51,12 @@ export default class Slider extends React.Component {
|
|||||||
* @param {SyntheticEvent} event Event
|
* @param {SyntheticEvent} event Event
|
||||||
*/
|
*/
|
||||||
_down(event) {
|
_down(event) {
|
||||||
|
|
||||||
let rect = event.currentTarget.getBoundingClientRect();
|
let rect = event.currentTarget.getBoundingClientRect();
|
||||||
this.left = rect.left;
|
this.left = rect.left;
|
||||||
this.width = rect.width;
|
this.width = rect.width;
|
||||||
this._move(event);
|
this._move(event);
|
||||||
|
this.touchStartTimer = setTimeout(() => this.sliderInputBox._setDisplay('block'), 1500);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -69,11 +76,67 @@ export default class Slider extends React.Component {
|
|||||||
* @param {Event} event DOM Event
|
* @param {Event} event DOM Event
|
||||||
*/
|
*/
|
||||||
_up(event) {
|
_up(event) {
|
||||||
|
this.sliderInputBox.sliderVal.focus();
|
||||||
|
clearTimeout(this.touchStartTimer);
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
this.left = null;
|
this.left = null;
|
||||||
this.width = null;
|
this.width = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key up handler for keyboard.
|
||||||
|
* display the number field then set focus to it
|
||||||
|
* when "Enter" key is pressed
|
||||||
|
* @param {Event} event Keyboard event
|
||||||
|
*/
|
||||||
|
_keyup(event) {
|
||||||
|
switch (event.key) {
|
||||||
|
case 'Enter':
|
||||||
|
event.preventDefault();
|
||||||
|
this.sliderInputBox._setDisplay('block');
|
||||||
|
//this.enterTimer = setTimeout(() => this.sliderInputBox.sliderVal.focus(), 10);
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Key down handler
|
||||||
|
* increment slider position by +/- 1 when right/left arrow key is pressed or held
|
||||||
|
* @param {Event} event
|
||||||
|
*/
|
||||||
|
_keydown(event) {
|
||||||
|
|
||||||
|
switch (event.key) {
|
||||||
|
case 'ArrowRight':
|
||||||
|
var newVal = this.props.percent*this.props.max + 1;
|
||||||
|
if (newVal <= this.props.max) this.props.onChange(newVal/this.props.max);
|
||||||
|
return;
|
||||||
|
case 'ArrowLeft':
|
||||||
|
var newVal = this.props.percent*this.props.max - 1;
|
||||||
|
if (newVal >= 0) this.props.onChange(newVal/this.props.max);
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Touch start handler
|
||||||
|
* @param {Event} event DOM Event
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
_touchstart(event) {
|
||||||
|
this.touchStartTimer = setTimeout(() => this.sliderInputBox._setDisplay('block'), 1500);
|
||||||
|
}
|
||||||
|
|
||||||
|
_touchend(event) {
|
||||||
|
this.sliderInputBox.sliderVal.focus();
|
||||||
|
clearTimeout(this.touchStartTimer);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Determine if the user is still dragging
|
* Determine if the user is still dragging
|
||||||
* @param {SyntheticEvent} event Event
|
* @param {SyntheticEvent} event Event
|
||||||
@@ -118,6 +181,7 @@ export default class Slider extends React.Component {
|
|||||||
*/
|
*/
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this._updateDimensions();
|
this._updateDimensions();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -151,16 +215,147 @@ export default class Slider extends React.Component {
|
|||||||
let width = outerWidth - (margin * 2);
|
let width = outerWidth - (margin * 2);
|
||||||
let pctPos = width * this.props.percent;
|
let pctPos = width * this.props.percent;
|
||||||
|
|
||||||
return <svg onMouseUp={this._up} onMouseEnter={this._enter.bind(this)} onMouseMove={this._move} onTouchEnd={this._up} style={style} ref={node => this.node = node}>
|
return <div><svg
|
||||||
|
onMouseUp={this._up} onMouseEnter={this._enter.bind(this)} onMouseMove={this._move} onKeyUp={this._keyup} onKeyDown={this._keydown} style={style} ref={node => this.node = node} tabIndex="0">
|
||||||
<rect className='primary' style={{ opacity: 0.3 }} x={margin} y='0.25em' rx='0.3em' ry='0.3em' width={width} height='0.7em' />
|
<rect className='primary' style={{ opacity: 0.3 }} x={margin} y='0.25em' rx='0.3em' ry='0.3em' width={width} height='0.7em' />
|
||||||
<rect className='primary-disabled' x={margin} y='0.45em' rx='0.15em' ry='0.15em' width={pctPos} height='0.3em' />
|
<rect className='primary-disabled' x={margin} y='0.45em' rx='0.15em' ry='0.15em' width={pctPos} height='0.3em' />
|
||||||
<circle className='primary' r={margin} cy='0.6em' cx={pctPos + margin} />
|
<circle className='primary' r={margin} cy='0.6em' cx={pctPos + margin} />
|
||||||
<rect x={margin} width={width} height='100%' fillOpacity='0' style={{ cursor: 'col-resize' }} onMouseDown={this._down} onTouchMove={this._move} onTouchStart={this._down} />
|
<rect x={margin} width={width} height='100%' fillOpacity='0' style={{ cursor: 'col-resize' }} onMouseDown={this._down} onTouchMove={this._move} onTouchStart={this._down} onTouchEnd={this._touchend} />
|
||||||
{axis && <g style={{ fontSize: '.7em' }}>
|
{axis && <g style={{ fontSize: '.7em' }}>
|
||||||
<text className='primary-disabled' y='3em' x={margin} style={{ textAnchor: 'middle' }}>{min + axisUnit}</text>
|
<text className='primary-disabled' y='3em' x={margin} style={{ textAnchor: 'middle' }}>{min + axisUnit}</text>
|
||||||
<text className='primary-disabled' y='3em' x='50%' style={{ textAnchor: 'middle' }}>{(min + max / 2) + axisUnit}</text>
|
<text className='primary-disabled' y='3em' x='50%' style={{ textAnchor: 'middle' }}>{(min + max / 2) + axisUnit}</text>
|
||||||
<text className='primary-disabled' y='3em' x='100%' style={{ textAnchor: 'end' }}>{max + axisUnit}</text>
|
<text className='primary-disabled' y='3em' x='100%' style={{ textAnchor: 'end' }}>{max + axisUnit}</text>
|
||||||
</g>}
|
</g>}
|
||||||
</svg>;
|
</svg>
|
||||||
|
<TextInputBox ref={(tb) => this.sliderInputBox = tb}
|
||||||
|
onChange={this.props.onChange}
|
||||||
|
percent={this.props.percent}
|
||||||
|
axisUnit={this.props.axisUnit}
|
||||||
|
scale={this.props.scale}
|
||||||
|
max={this.props.max}
|
||||||
|
|
||||||
|
/>
|
||||||
|
</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* New component to add keyboard support for sliders - works on all devices (desktop, iOS, Android)
|
||||||
|
**/
|
||||||
|
class TextInputBox extends React.Component {
|
||||||
|
static propTypes = {
|
||||||
|
axisUnit: PropTypes.string,//units (T, M, etc.)
|
||||||
|
max: PropTypes.number,
|
||||||
|
onChange: PropTypes.func.isRequired,// function which determins percent value
|
||||||
|
percent: PropTypes.number.isRequired,//value of slider
|
||||||
|
scale: PropTypes.number
|
||||||
|
};
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this._handleFocus = this._handleFocus.bind(this);
|
||||||
|
this._handleBlur = this._handleBlur.bind(this);
|
||||||
|
this._handleChange = this._handleChange.bind(this);
|
||||||
|
//this._keydown = this._keydown.bind(this);
|
||||||
|
this._keyup = this._keyup.bind(this);
|
||||||
|
this.state = this._getInitialState();
|
||||||
|
this.percent = this.props.percent;
|
||||||
|
this.max = this.props.max;
|
||||||
|
this.state.inputValue = this.percent * this.max;
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillReceiveProps(nextProps, nextState) {
|
||||||
|
var nextValue = nextProps.percent * nextProps.max;
|
||||||
|
// See https://stackoverflow.com/questions/32414308/updating-state-on-props-change-in-react-form
|
||||||
|
if (nextValue !== this.state.inputValue && nextValue <= nextProps.max) {
|
||||||
|
this.setState({ inputValue: nextValue });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
componentDidUpdate(prevProps, prevState) {
|
||||||
|
|
||||||
|
if (prevState.divStyle.display == 'none' && this.state.divStyle.display == 'block') {
|
||||||
|
this.enterTimer = setTimeout(() => this.sliderVal.focus(), 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prevProps.max !== this.props.max && this.state.inputValue > this.props.max) {
|
||||||
|
// they chose a different module
|
||||||
|
this.setState({ inputValue: this.props.max });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.state.inputValue != prevState.inputValue && prevProps.max == this.props.max) {
|
||||||
|
this.props.onChange(this.state.inputValue/this.props.max);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
_getInitialState() {
|
||||||
|
return {
|
||||||
|
divStyle: {display:'none'},
|
||||||
|
inputStyle: {width:'4em'},
|
||||||
|
labelStyle: {marginLeft: '.1em'},
|
||||||
|
maxLength:5,
|
||||||
|
size:5,
|
||||||
|
min:0,
|
||||||
|
tabIndex:-1,
|
||||||
|
type:'number',
|
||||||
|
readOnly: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_setDisplay(val) {
|
||||||
|
this.setState({
|
||||||
|
divStyle: {display:val}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_handleFocus() {
|
||||||
|
this.setState({
|
||||||
|
inputValue:this._getValue()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_handleBlur() {
|
||||||
|
this._setDisplay('none');
|
||||||
|
if (this.state.inputValue !== '') {
|
||||||
|
this.props.onChange(this.state.inputValue/this.props.max);
|
||||||
|
} else {
|
||||||
|
this.state.inputValue = this.props.percent * this.props.max;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
_getValue() {
|
||||||
|
return this.state.inputValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
_handleChange(event) {
|
||||||
|
if (event.target.value < 0) {
|
||||||
|
this.setState({inputValue: 0});
|
||||||
|
} else if (event.target.value <= this.props.max) {
|
||||||
|
this.setState({inputValue: event.target.value});
|
||||||
|
} else {
|
||||||
|
this.setState({inputValue: this.props.max});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key up handler for input field.
|
||||||
|
* If user hits Enter key, blur/close the input field
|
||||||
|
* @param {Event} event Keyboard event
|
||||||
|
*/
|
||||||
|
_keyup(event) {
|
||||||
|
switch (event.key) {
|
||||||
|
case 'Enter':
|
||||||
|
this.sliderVal.blur();
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
let { axisUnit, onChange, percent, scale } = this.props;
|
||||||
|
return <div style={this.state.divStyle}><input style={this.state.inputStyle} value={this._getValue()} min={this.state.min} max={this.props.max} onChange={this._handleChange} onKeyUp={this._keyup} tabIndex={this.state.tabIndex} maxLength={this.state.maxLength} size={this.state.size} onBlur={() => {this._handleBlur()}} onFocus={() => {this._handleFocus()}} type={this.state.type} ref={(ip) => this.sliderVal = ip}/><text className="primary upp" style={this.state.labelStyle}>{this.props.axisUnit}</text></div>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -40,6 +40,8 @@ export default class Slot extends TranslatedComponent {
|
|||||||
|
|
||||||
this._contextMenu = wrapCtxMenu(this._contextMenu.bind(this));
|
this._contextMenu = wrapCtxMenu(this._contextMenu.bind(this));
|
||||||
this._getMaxClassLabel = this._getMaxClassLabel.bind(this);
|
this._getMaxClassLabel = this._getMaxClassLabel.bind(this);
|
||||||
|
this._keyDown = this._keyDown.bind(this);
|
||||||
|
this.slotDiv = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Must be implemented by subclasses:
|
// Must be implemented by subclasses:
|
||||||
@@ -73,6 +75,25 @@ export default class Slot extends TranslatedComponent {
|
|||||||
this.props.onSelect(null,null);
|
this.props.onSelect(null,null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Key Down handler
|
||||||
|
* @param {SyntheticEvent} event Event
|
||||||
|
* ToDo: see if this can be moved up
|
||||||
|
* we do more or less the same thing
|
||||||
|
* in every section when Enter key is pressed
|
||||||
|
* on a focusable item
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
_keyDown(event) {
|
||||||
|
if (event.key == 'Enter') {
|
||||||
|
if(event.target.className == 'r') {
|
||||||
|
console.log("Slot: Enter key pressed on mod icon");
|
||||||
|
this._toggleModifications();
|
||||||
|
} else {
|
||||||
|
console.log("Slot: Enter key pressed on: %O", event.target);
|
||||||
|
}
|
||||||
|
this.props.onOpen(event);
|
||||||
|
}
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* Render the slot
|
* Render the slot
|
||||||
* @return {React.Component} The slot
|
* @return {React.Component} The slot
|
||||||
@@ -104,6 +125,7 @@ export default class Slot extends TranslatedComponent {
|
|||||||
ship={ship}
|
ship={ship}
|
||||||
m={m}
|
m={m}
|
||||||
marker={modificationsMarker}
|
marker={modificationsMarker}
|
||||||
|
modButton = {this.modButton}
|
||||||
/>;
|
/>;
|
||||||
} else {
|
} else {
|
||||||
menu = <AvailableModulesMenu
|
menu = <AvailableModulesMenu
|
||||||
@@ -114,6 +136,7 @@ export default class Slot extends TranslatedComponent {
|
|||||||
onSelect={onSelect}
|
onSelect={onSelect}
|
||||||
warning={warning}
|
warning={warning}
|
||||||
diffDetails={diffDetails.bind(ship, this.context.language)}
|
diffDetails={diffDetails.bind(ship, this.context.language)}
|
||||||
|
slotDiv = {this.slotDiv}
|
||||||
/>;
|
/>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -121,7 +144,7 @@ export default class Slot extends TranslatedComponent {
|
|||||||
// TODO: implement touch dragging
|
// TODO: implement touch dragging
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn('slot', dropClass, { selected })} onClick={onOpen} onContextMenu={this._contextMenu} onDragOver={dragOver}>
|
<div className={cn('slot', dropClass, { selected })} onClick={onOpen} onKeyDown={this._keyDown} onContextMenu={this._contextMenu} onDragOver={dragOver} tabIndex="0" ref={slotDiv => this.slotDiv = slotDiv}>
|
||||||
<div className='details-container'>
|
<div className='details-container'>
|
||||||
<div className='sz'>{this._getMaxClassLabel(translate)}</div>
|
<div className='sz'>{this._getMaxClassLabel(translate)}</div>
|
||||||
{slotDetails}
|
{slotDetails}
|
||||||
@@ -131,6 +154,7 @@ export default class Slot extends TranslatedComponent {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Toggle the modifications flag when selecting the modifications icon
|
* Toggle the modifications flag when selecting the modifications icon
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -35,6 +35,18 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
this._modificationsSelected = false;
|
this._modificationsSelected = false;
|
||||||
|
this._keyDown = this._keyDown.bind(this);
|
||||||
|
this.modButton = null;
|
||||||
|
this.slotDiv = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
_keyDown(event) {
|
||||||
|
if (event.key == 'Enter') {
|
||||||
|
if(event.target.className == 'r') {
|
||||||
|
this._toggleModifications();
|
||||||
|
}
|
||||||
|
this.props.onOpen(event);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -48,7 +60,7 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
let m = slot.m;
|
let m = slot.m;
|
||||||
let classRating = m.class + m.rating;
|
let classRating = m.class + m.rating;
|
||||||
let menu;
|
let menu;
|
||||||
let validMods = m == null ? [] : (Modifications.modules[m.grp].modifications || []);
|
let validMods = m == null || !Modifications.modules[m.grp] ? [] : (Modifications.modules[m.grp].modifications || []);
|
||||||
let showModuleResistances = Persist.showModuleResistances();
|
let showModuleResistances = Persist.showModuleResistances();
|
||||||
let mass = m.getMass() || m.cargo || m.fuel || 0;
|
let mass = m.getMass() || m.cargo || m.fuel || 0;
|
||||||
|
|
||||||
@@ -56,6 +68,9 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
let modTT = translate('modified');
|
let modTT = translate('modified');
|
||||||
if (m && m.blueprint && m.blueprint.name) {
|
if (m && m.blueprint && m.blueprint.name) {
|
||||||
modTT = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
modTT = translate(m.blueprint.name) + ' ' + translate('grade') + ' ' + m.blueprint.grade;
|
||||||
|
if (m.blueprint.special && m.blueprint.special.id >= 0) {
|
||||||
|
modTT += ', ' + translate(m.blueprint.special.name);
|
||||||
|
}
|
||||||
modTT = (
|
modTT = (
|
||||||
<div>
|
<div>
|
||||||
<div>{modTT}</div>
|
<div>{modTT}</div>
|
||||||
@@ -79,6 +94,7 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
ship={ship}
|
ship={ship}
|
||||||
m={m}
|
m={m}
|
||||||
marker={modificationsMarker}
|
marker={modificationsMarker}
|
||||||
|
modButton = {this.modButton}
|
||||||
/>;
|
/>;
|
||||||
} else {
|
} else {
|
||||||
menu = <AvailableModulesMenu
|
menu = <AvailableModulesMenu
|
||||||
@@ -89,14 +105,15 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
onSelect={onSelect}
|
onSelect={onSelect}
|
||||||
warning={warning}
|
warning={warning}
|
||||||
diffDetails={diffDetails.bind(ship, this.context.language)}
|
diffDetails={diffDetails.bind(ship, this.context.language)}
|
||||||
|
slotDiv = {this.slotDiv}
|
||||||
/>;
|
/>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn('slot', { selected: this.props.selected })} onClick={this.props.onOpen} onContextMenu={stopCtxPropagation}>
|
<div className={cn('slot', { selected: this.props.selected })} onClick={this.props.onOpen} onKeyDown={this._keyDown} onContextMenu={stopCtxPropagation} tabIndex="0" ref={ slotDiv => this.slotDiv = slotDiv }>
|
||||||
<div className={cn('details-container', { warning: warning && warning(slot.m), disabled: m.grp !== 'bh' && !slot.enabled })}>
|
<div className={cn('details-container', { warning: warning && warning(slot.m), disabled: m.grp !== 'bh' && !slot.enabled })}>
|
||||||
<div className={'sz'}>{slot.maxClass}</div>
|
<div className={'sz'}>{m.grp == 'bh' ? m.name.charAt(0) : slot.maxClass}</div>
|
||||||
<div>
|
<div>
|
||||||
<div className={'l'}>{classRating} {translate(m.name || m.grp)}{m.mods && Object.keys(m.mods).length > 0 ? <span className='r' onMouseOver={termtip.bind(null, modTT)} onMouseOut={tooltip.bind(null, null)}><Modified /></span> : null }</div>
|
<div className={'l'}>{classRating} {translate(m.name || m.grp)}{m.mods && Object.keys(m.mods).length > 0 ? <span className='r' onMouseOver={termtip.bind(null, modTT)} onMouseOut={tooltip.bind(null, null)}><Modified /></span> : null }</div>
|
||||||
<div className={'r'}>{formats.round(mass)}{units.T}</div>
|
<div className={'r'}>{formats.round(mass)}{units.T}</div>
|
||||||
@@ -118,7 +135,7 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
{ showModuleResistances && m.getKineticResistance() ? <div className='l'>{translate('kinres')}: {formats.pct(m.getKineticResistance())}</div> : null }
|
{ showModuleResistances && m.getKineticResistance() ? <div className='l'>{translate('kinres')}: {formats.pct(m.getKineticResistance())}</div> : null }
|
||||||
{ showModuleResistances && m.getThermalResistance() ? <div className='l'>{translate('thermres')}: {formats.pct(m.getThermalResistance())}</div> : null }
|
{ showModuleResistances && m.getThermalResistance() ? <div className='l'>{translate('thermres')}: {formats.pct(m.getThermalResistance())}</div> : null }
|
||||||
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
{ m.getIntegrity() ? <div className='l'>{translate('integrity')}: {formats.int(m.getIntegrity())}</div> : null }
|
||||||
{ validMods.length > 0 ? <div className='r' ><button onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
{ validMods.length > 0 ? <div className='r' tabIndex="0" ref={ modButton => this.modButton = modButton }><button tabIndex="-1" onClick={this._toggleModifications.bind(this)} onContextMenu={stopCtxPropagation} onMouseOver={termtip.bind(null, 'modifications')} onMouseOut={tooltip.bind(null, null)}><ListModifications /></button></div> : null }
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -131,6 +148,7 @@ export default class StandardSlot extends TranslatedComponent {
|
|||||||
* Toggle the modifications flag when selecting the modifications icon
|
* Toggle the modifications flag when selecting the modifications icon
|
||||||
*/
|
*/
|
||||||
_toggleModifications() {
|
_toggleModifications() {
|
||||||
|
|
||||||
this._modificationsSelected = !this._modificationsSelected;
|
this._modificationsSelected = !this._modificationsSelected;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -235,12 +235,10 @@ export default class StandardSlotSection extends SlotSection {
|
|||||||
<ul>
|
<ul>
|
||||||
<li className='lc' onClick={this._multiPurpose.bind(this, false, 0)}>{translate('Multi-purpose')}</li>
|
<li className='lc' onClick={this._multiPurpose.bind(this, false, 0)}>{translate('Multi-purpose')}</li>
|
||||||
<li className='lc' onClick={this._multiPurpose.bind(this, true, 2)}>{translate('Combat')}</li>
|
<li className='lc' onClick={this._multiPurpose.bind(this, true, 2)}>{translate('Combat')}</li>
|
||||||
<li className='lc' onClick={this._optimizeCargo.bind(this, false)}>{translate('Trader')}</li>
|
<li className='lc' onClick={this._optimizeCargo.bind(this, true)}>{translate('Trader')}</li>
|
||||||
<li className='lc' onClick={this._optimizeCargo.bind(this, true)}>{translate('Shielded Trader')}</li>
|
|
||||||
<li className='lc' onClick={this._optimizeExplorer.bind(this, false)}>{translate('Explorer')}</li>
|
<li className='lc' onClick={this._optimizeExplorer.bind(this, false)}>{translate('Explorer')}</li>
|
||||||
<li className={cn('lc', { disabled: planetaryDisabled })} onClick={!planetaryDisabled && this._optimizeExplorer.bind(this, true)}>{translate('Planetary Explorer')}</li>
|
<li className={cn('lc', { disabled: planetaryDisabled })} onClick={!planetaryDisabled && this._optimizeExplorer.bind(this, true)}>{translate('Planetary Explorer')}</li>
|
||||||
<li className='lc' onClick={this._optimizeMiner.bind(this, false)}>{translate('Miner')}</li>
|
<li className='lc' onClick={this._optimizeMiner.bind(this, true)}>{translate('Miner')}</li>
|
||||||
<li className='lc' onClick={this._optimizeMiner.bind(this, true)}>{translate('Shielded Miner')}</li>
|
|
||||||
<li className='lc' onClick={this._optimizeRacer.bind(this)}>{translate('Racer')}</li>
|
<li className='lc' onClick={this._optimizeRacer.bind(this)}>{translate('Racer')}</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>;
|
</div>;
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import * as FR from './fr';
|
|||||||
import * as IT from './it';
|
import * as IT from './it';
|
||||||
import * as RU from './ru';
|
import * as RU from './ru';
|
||||||
import * as PL from './pl';
|
import * as PL from './pl';
|
||||||
|
import * as PT from './pt';
|
||||||
import * as d3 from 'd3';
|
import * as d3 from 'd3';
|
||||||
|
|
||||||
let fallbackTerms = EN.terms;
|
let fallbackTerms = EN.terms;
|
||||||
@@ -25,6 +26,7 @@ export function getLanguage(langCode) {
|
|||||||
case 'it': lang = IT; break;
|
case 'it': lang = IT; break;
|
||||||
case 'ru': lang = RU; break;
|
case 'ru': lang = RU; break;
|
||||||
case 'pl': lang = PL; break;
|
case 'pl': lang = PL; break;
|
||||||
|
case 'pt': lang = PT; break;
|
||||||
default:
|
default:
|
||||||
lang = EN;
|
lang = EN;
|
||||||
}
|
}
|
||||||
@@ -88,5 +90,6 @@ export const Languages = {
|
|||||||
es: 'Español',
|
es: 'Español',
|
||||||
fr: 'Français',
|
fr: 'Français',
|
||||||
ru: 'ру́сский',
|
ru: 'ру́сский',
|
||||||
pl: 'polski'
|
pl: 'polski',
|
||||||
|
pt: 'português'
|
||||||
};
|
};
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
16
src/app/i18n/pt.js
Normal file
16
src/app/i18n/pt.js
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
export const formats = {
|
||||||
|
decimal: ',',
|
||||||
|
thousands: '.',
|
||||||
|
grouping: [3],
|
||||||
|
currency: ['', ' €'],
|
||||||
|
dateTime: '%A, %e de %B de %Y, %X',
|
||||||
|
date: '%d/%m/%Y',
|
||||||
|
time: '%H:%M:%S',
|
||||||
|
periods: ['AM', 'PM'],
|
||||||
|
days: ['domingo', 'lunes', 'martes', 'miércoles', 'jueves', 'viernes', 'sábado'],
|
||||||
|
shortDays: ['dom', 'lun', 'mar', 'mié', 'jue', 'vie', 'sáb'],
|
||||||
|
months: ['enero', 'febrero', 'marzo', 'abril', 'mayo', 'junio', 'julio', 'agosto', 'septiembre', 'octubre', 'noviembre', 'diciembre'],
|
||||||
|
shortMonths: ['ene', 'feb', 'mar', 'abr', 'may', 'jun', 'jul', 'ago', 'sep', 'oct', 'nov', 'dic']
|
||||||
|
};
|
||||||
|
|
||||||
|
export { default as terms } from './pt.json';
|
||||||
278
src/app/i18n/pt.json
Normal file
278
src/app/i18n/pt.json
Normal file
File diff suppressed because one or more lines are too long
@@ -41,6 +41,9 @@ export default class AboutPage extends Page {
|
|||||||
|
|
||||||
<h3>Chat</h3>
|
<h3>Chat</h3>
|
||||||
<p>You can chat to us on our <a href='https://discord.gg/0uwCh6R62aPRjk9w' target='_blank'>EDCD Discord server</a>.</p>
|
<p>You can chat to us on our <a href='https://discord.gg/0uwCh6R62aPRjk9w' target='_blank'>EDCD Discord server</a>.</p>
|
||||||
|
|
||||||
|
<h3>Supporting Coriolis</h3>
|
||||||
|
<p>Coriolis is an open source project, and I work on it in my free time. I have set up a patreon at <a href='https://www.patreon.com/coriolis_elite'>patreon.com/coriolis_elite</a>, which will be used to keep Coriolis up to date and the servers running.</p>
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,13 +73,10 @@ export default class OutfittingPage extends Page {
|
|||||||
let buildName = params.bn;
|
let buildName = params.bn;
|
||||||
let data = Ships[shipId]; // Retrieve the basic ship properties, slots and defaults
|
let data = Ships[shipId]; // Retrieve the basic ship properties, slots and defaults
|
||||||
let savedCode = Persist.getBuild(shipId, buildName);
|
let savedCode = Persist.getBuild(shipId, buildName);
|
||||||
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return { error: { message: 'Ship not found: ' + shipId } };
|
return { error: { message: 'Ship not found: ' + shipId } };
|
||||||
}
|
}
|
||||||
|
|
||||||
let ship = new Ship(shipId, data.properties, data.slots); // Create a new Ship instance
|
let ship = new Ship(shipId, data.properties, data.slots); // Create a new Ship instance
|
||||||
|
|
||||||
if (code) {
|
if (code) {
|
||||||
ship.buildFrom(code); // Populate modules from serialized 'code' URL param
|
ship.buildFrom(code); // Populate modules from serialized 'code' URL param
|
||||||
} else {
|
} else {
|
||||||
@@ -556,10 +553,25 @@ export default class OutfittingPage extends Page {
|
|||||||
const boostMarker = `${ship.canBoost(cargo, fuel)}`;
|
const boostMarker = `${ship.canBoost(cargo, fuel)}`;
|
||||||
const shipSummaryMarker = `${ship.name}${_sStr}${_iStr}${_hStr}${_pStr}${_mStr}${ship.ladenMass}${cargo}${fuel}`;
|
const shipSummaryMarker = `${ship.name}${_sStr}${_iStr}${_hStr}${_pStr}${_mStr}${ship.ladenMass}${cargo}${fuel}`;
|
||||||
|
|
||||||
|
const requirements = Ships[ship.id].requirements;
|
||||||
|
var requirementElements = [];
|
||||||
|
|
||||||
|
function renderRequirement(className, textKey, tooltipTextKey) {
|
||||||
|
requirementElements.push(<div key={textKey} className={className} onMouseEnter={termtip.bind(null, tooltipTextKey)} onMouseLeave={hide}>{translate(textKey)}</div>);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (requirements) {
|
||||||
|
requirements.federationRank && renderRequirement('federation', 'federation rank ' + requirements.federationRank, 'federation rank required');
|
||||||
|
requirements.empireRank && renderRequirement('empire', 'empire rank ' + requirements.empireRank, 'empire rank required');
|
||||||
|
requirements.horizons && renderRequirement('horizons', 'horizons', 'horizons required');
|
||||||
|
requirements.horizonsEarlyAdoption && renderRequirement('horizons', 'horizons early adoption', 'horizons early adoption required');
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div id='outfit' className={'page'} style={{ fontSize: (sizeRatio * 0.9) + 'em' }}>
|
<div id='outfit' className={'page'} style={{ fontSize: (sizeRatio * 0.9) + 'em' }}>
|
||||||
<div id='overview'>
|
<div id='overview'>
|
||||||
<h1>{ship.name}</h1>
|
<h1>{ship.name}</h1>
|
||||||
|
<div id='requirements'>{requirementElements}</div>
|
||||||
<div id='build'>
|
<div id='build'>
|
||||||
<input value={newBuildName || ''} onChange={this._buildNameChange} placeholder={translate('Enter Name')} maxLength={50} />
|
<input value={newBuildName || ''} onChange={this._buildNameChange} placeholder={translate('Enter Name')} maxLength={50} />
|
||||||
<button onClick={canSave && this._saveBuild} disabled={!canSave} onMouseOver={termtip.bind(null, 'save')} onMouseOut={hide}>
|
<button onClick={canSave && this._saveBuild} disabled={!canSave} onMouseOver={termtip.bind(null, 'save')} onMouseOut={hide}>
|
||||||
@@ -590,7 +602,7 @@ export default class OutfittingPage extends Page {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Main tables */}
|
{/* Main tables */}
|
||||||
<ShipSummaryTable ship={ship} fuel={fuel} cargo={cargo} marker={shipSummaryMarker} />
|
<ShipSummaryTable ship={ship} fuel={fuel} cargo={cargo} marker={shipSummaryMarker} pips={{sys: this.state.sys, wep: this.state.wep, eng: this.state.eng}} />
|
||||||
<StandardSlotSection ship={ship} fuel={fuel} cargo={cargo} code={standardSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
<StandardSlotSection ship={ship} fuel={fuel} cargo={cargo} code={standardSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
||||||
<InternalSlotSection ship={ship} code={internalSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
<InternalSlotSection ship={ship} code={internalSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
||||||
<HardpointSlotSection ship={ship} code={hardpointsSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
<HardpointSlotSection ship={ship} code={hardpointsSlotMarker} onChange={shipUpdated} onCargoChange={this._cargoUpdated} onFuelChange={this._fuelUpdated} currentMenu={menu} />
|
||||||
|
|||||||
@@ -25,6 +25,25 @@ function countInt(slot) {
|
|||||||
this.int[slot.maxClass - 1]++; // Subtract 1 since there is no Class 0 Internal compartment
|
this.int[slot.maxClass - 1]++; // Subtract 1 since there is no Class 0 Internal compartment
|
||||||
this.intCount++;
|
this.intCount++;
|
||||||
this.maxCargo += crEligible ? ModuleUtils.findInternal('cr', slot.maxClass, 'E').cargo : 0;
|
this.maxCargo += crEligible ? ModuleUtils.findInternal('cr', slot.maxClass, 'E').cargo : 0;
|
||||||
|
|
||||||
|
// if no eligiblity, then assume pce
|
||||||
|
let passSlotType = null;
|
||||||
|
let passSlotRating = null;
|
||||||
|
if (!slot.eligible || slot.eligible.pce) {
|
||||||
|
passSlotType = 'pce';
|
||||||
|
passSlotRating = 'E';
|
||||||
|
} else if (slot.eligible.pci) {
|
||||||
|
passSlotType = 'pci';
|
||||||
|
passSlotRating = 'D';
|
||||||
|
} else if (slot.eligible.pcm) {
|
||||||
|
passSlotType = 'pcm';
|
||||||
|
passSlotRating = 'C';
|
||||||
|
} else if (slot.eligible.pcq) {
|
||||||
|
passSlotType = 'pcq';
|
||||||
|
passSlotRating = 'B';
|
||||||
|
}
|
||||||
|
let passengerBay = passSlotType ? ModuleUtils.findMaxInternal(passSlotType, slot.maxClass, passSlotRating) : null;
|
||||||
|
this.maxPassengers += passengerBay ? passengerBay.passengers : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -39,6 +58,7 @@ function shipSummary(shipId, shipData) {
|
|||||||
hpCount: 0,
|
hpCount: 0,
|
||||||
intCount: 0,
|
intCount: 0,
|
||||||
maxCargo: 0,
|
maxCargo: 0,
|
||||||
|
maxPassengers: 0,
|
||||||
hp: [0, 0, 0, 0, 0], // Utility, Small, Medium, Large, Huge
|
hp: [0, 0, 0, 0, 0], // Utility, Small, Medium, Large, Huge
|
||||||
int: [0, 0, 0, 0, 0, 0, 0, 0], // Sizes 1 - 8
|
int: [0, 0, 0, 0, 0, 0, 0, 0], // Sizes 1 - 8
|
||||||
standard: shipData.slots.standard,
|
standard: shipData.slots.standard,
|
||||||
@@ -139,15 +159,16 @@ export default class ShipyardPage extends Page {
|
|||||||
* @param {Object} u Localized unit map
|
* @param {Object} u Localized unit map
|
||||||
* @param {Function} fInt Localized integer formatter
|
* @param {Function} fInt Localized integer formatter
|
||||||
* @param {Function} fRound Localized round formatter
|
* @param {Function} fRound Localized round formatter
|
||||||
|
* @param {Boolean} highlight Should this row be highlighted
|
||||||
* @return {React.Component} Table Row
|
* @return {React.Component} Table Row
|
||||||
*/
|
*/
|
||||||
_shipRowElement(s, translate, u, fInt, fRound) {
|
_shipRowElement(s, translate, u, fInt, fRound, highlight) {
|
||||||
let noTouch = this.context.noTouch;
|
let noTouch = this.context.noTouch;
|
||||||
|
|
||||||
return <tr
|
return <tr
|
||||||
key={s.id}
|
key={s.id}
|
||||||
style={{ height: '1.5em' }}
|
style={{ height: '1.5em' }}
|
||||||
className={cn({ highlighted: noTouch && this.state.shipId === s.id })}
|
className={cn({ highlighted: noTouch && this.state.shipId === s.id, alt: highlight })}
|
||||||
onMouseEnter={noTouch && this._highlightShip.bind(this, s.id)}
|
onMouseEnter={noTouch && this._highlightShip.bind(this, s.id)}
|
||||||
>
|
>
|
||||||
<td className='ri'>{s.manufacturer}</td>
|
<td className='ri'>{s.manufacturer}</td>
|
||||||
@@ -166,12 +187,14 @@ export default class ShipyardPage extends Page {
|
|||||||
<td className='ri'>{fInt(s.topBoost)}</td>
|
<td className='ri'>{fInt(s.topBoost)}</td>
|
||||||
<td className='ri'>{fRound(s.maxJumpRange)}</td>
|
<td className='ri'>{fRound(s.maxJumpRange)}</td>
|
||||||
<td className='ri'>{fInt(s.maxCargo)}</td>
|
<td className='ri'>{fInt(s.maxCargo)}</td>
|
||||||
|
<td className='ri'>{fInt(s.maxPassengers)}</td>
|
||||||
<td className='cn'>{s.standard[0]}</td>
|
<td className='cn'>{s.standard[0]}</td>
|
||||||
<td className='cn'>{s.standard[1]}</td>
|
<td className='cn'>{s.standard[1]}</td>
|
||||||
<td className='cn'>{s.standard[2]}</td>
|
<td className='cn'>{s.standard[2]}</td>
|
||||||
<td className='cn'>{s.standard[3]}</td>
|
<td className='cn'>{s.standard[3]}</td>
|
||||||
<td className='cn'>{s.standard[4]}</td>
|
<td className='cn'>{s.standard[4]}</td>
|
||||||
<td className='cn'>{s.standard[5]}</td>
|
<td className='cn'>{s.standard[5]}</td>
|
||||||
|
<td className='cn'>{s.standard[6]}</td>
|
||||||
<td className={cn({ disabled: !s.hp[1] })}>{s.hp[1]}</td>
|
<td className={cn({ disabled: !s.hp[1] })}>{s.hp[1]}</td>
|
||||||
<td className={cn({ disabled: !s.hp[2] })}>{s.hp[2]}</td>
|
<td className={cn({ disabled: !s.hp[2] })}>{s.hp[2]}</td>
|
||||||
<td className={cn({ disabled: !s.hp[3] })}>{s.hp[3]}</td>
|
<td className={cn({ disabled: !s.hp[3] })}>{s.hp[3]}</td>
|
||||||
@@ -246,13 +269,26 @@ export default class ShipyardPage extends Page {
|
|||||||
let shipRows = new Array(shipSummaries.length);
|
let shipRows = new Array(shipSummaries.length);
|
||||||
let detailRows = new Array(shipSummaries.length);
|
let detailRows = new Array(shipSummaries.length);
|
||||||
|
|
||||||
|
let lastShipSortValue = null;
|
||||||
|
let backgroundHighlight = false;
|
||||||
|
|
||||||
for (let s of shipSummaries) {
|
for (let s of shipSummaries) {
|
||||||
detailRows[i] = this._shipRowElement(s, translate, units, fInt, formats.f1);
|
let shipSortValue = s[shipPredicate];
|
||||||
|
if( shipPredicateIndex != undefined ) {
|
||||||
|
shipSortValue = shipSortValue[shipPredicateIndex];
|
||||||
|
}
|
||||||
|
|
||||||
|
if( shipSortValue != lastShipSortValue ) {
|
||||||
|
backgroundHighlight = !backgroundHighlight;
|
||||||
|
lastShipSortValue = shipSortValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
detailRows[i] = this._shipRowElement(s, translate, units, fInt, formats.f1, backgroundHighlight);
|
||||||
shipRows[i] = (
|
shipRows[i] = (
|
||||||
<tr
|
<tr
|
||||||
key={i}
|
key={i}
|
||||||
style={{ height: '1.5em' }}
|
style={{ height: '1.5em' }}
|
||||||
className={cn({ highlighted: noTouch && this.state.shipId === s.id })}
|
className={cn({ highlighted: noTouch && this.state.shipId === s.id, alt: backgroundHighlight })}
|
||||||
onMouseEnter={noTouch && this._highlightShip.bind(this, s.id)}
|
onMouseEnter={noTouch && this._highlightShip.bind(this, s.id)}
|
||||||
>
|
>
|
||||||
<td className='le'><Link href={'/outfit/' + s.id}>{s.name}</Link></td>
|
<td className='le'><Link href={'/outfit/' + s.id}>{s.name}</Link></td>
|
||||||
@@ -293,8 +329,8 @@ export default class ShipyardPage extends Page {
|
|||||||
<th rowSpan={3} className='sortable' onMouseEnter={termtip.bind(null, 'hardness')} onMouseLeave={hide} onClick={sortShips('hardness')}>{translate('hrd')}</th>
|
<th rowSpan={3} className='sortable' onMouseEnter={termtip.bind(null, 'hardness')} onMouseLeave={hide} onClick={sortShips('hardness')}>{translate('hrd')}</th>
|
||||||
<th> </th>
|
<th> </th>
|
||||||
<th colSpan={4}>{translate('base')}</th>
|
<th colSpan={4}>{translate('base')}</th>
|
||||||
<th colSpan={4}>{translate('max')}</th>
|
<th colSpan={5}>{translate('max')}</th>
|
||||||
<th className='lft' colSpan={6}></th>
|
<th className='lft' colSpan={7}></th>
|
||||||
<th className='lft' colSpan={5}></th>
|
<th className='lft' colSpan={5}></th>
|
||||||
<th className='lft' colSpan={8}></th>
|
<th className='lft' colSpan={8}></th>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -310,29 +346,31 @@ export default class ShipyardPage extends Page {
|
|||||||
<th className='sortable' onClick={sortShips('topBoost')}>{translate('boost')}</th>
|
<th className='sortable' onClick={sortShips('topBoost')}>{translate('boost')}</th>
|
||||||
<th className='sortable' onClick={sortShips('maxJumpRange')}>{translate('jump')}</th>
|
<th className='sortable' onClick={sortShips('maxJumpRange')}>{translate('jump')}</th>
|
||||||
<th className='sortable' onClick={sortShips('maxCargo')}>{translate('cargo')}</th>
|
<th className='sortable' onClick={sortShips('maxCargo')}>{translate('cargo')}</th>
|
||||||
|
<th className='sortable' onClick={sortShips('maxPassengers')}>{translate('pax')}</th>
|
||||||
|
|
||||||
<th className='lft' colSpan={6}>{translate('core module classes')}</th>
|
<th className='lft' colSpan={7}>{translate('core module classes')}</th>
|
||||||
<th colSpan={5} className='sortable lft' onClick={sortShips('hpCount')}>{translate('hardpoints')}</th>
|
<th colSpan={5} className='sortable lft' onClick={sortShips('hpCount')}>{translate('hardpoints')}</th>
|
||||||
<th colSpan={8} className='sortable lft' onClick={sortShips('intCount')}>{translate('internal compartments')}</th>
|
<th colSpan={8} className='sortable lft' onClick={sortShips('intCount')}>{translate('internal compartments')}</th>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th className='sortable lft' onClick={sortShips('retailCost')}>{units.CR}</th>
|
<th className='sortable lft' onClick={sortShips('retailCost')}>{units.CR}</th>
|
||||||
<th className='sortable lft' onClick={sortShips('hullMass')}>{units.T}</th>
|
<th className='sortable lft' onClick={sortShips('hullMass')}>{units.T}</th>
|
||||||
<th className='sortable lft' onClick={sortShips('speed')}>{units['m/s']}</th>
|
<th className='sortable lft' onClick={sortShips('speed')}>{units['m/s']}</th>
|
||||||
<th className='sortable' onClick={sortShips('boost')}>{units['m/s']}</th>
|
<th className='sortable' onClick={sortShips('boost')}>{units['m/s']}</th>
|
||||||
<th> </th>
|
<th> </th>
|
||||||
<th className='sortable' onClick={sortShips('baseShieldStrength')}>{units.MJ}</th>
|
<th className='sortable' onClick={sortShips('baseShieldStrength')}>{units.MJ}</th>
|
||||||
<th className='sortable lft' onClick={sortShips('topSpeed')}>{units['m/s']}</th>
|
<th className='sortable lft' onClick={sortShips('topSpeed')}>{units['m/s']}</th>
|
||||||
<th className='sortable' onClick={sortShips('topBoost')}>{units['m/s']}</th>
|
<th className='sortable' onClick={sortShips('topBoost')}>{units['m/s']}</th>
|
||||||
<th className='sortable' onClick={sortShips('maxJumpRange')}>{units.LY}</th>
|
<th className='sortable' onClick={sortShips('maxJumpRange')}>{units.LY}</th>
|
||||||
<th className='sortable' onClick={sortShips('maxCargo')}>{units.T}</th>
|
<th className='sortable' onClick={sortShips('maxCargo')}>{units.T}</th>
|
||||||
|
<th> </th>
|
||||||
<th className='sortable lft' onMouseEnter={termtip.bind(null, 'power plant')} onMouseLeave={hide} onClick={sortShips('standard', 0)}>{'pp'}</th>
|
<th className='sortable lft' onMouseEnter={termtip.bind(null, 'power plant')} onMouseLeave={hide} onClick={sortShips('standard', 0)}>{'pp'}</th>
|
||||||
<th className='sortable' onMouseEnter={termtip.bind(null, 'thrusters')} onMouseLeave={hide} onClick={sortShips('standard', 1)}>{'th'}</th>
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'thrusters')} onMouseLeave={hide} onClick={sortShips('standard', 1)}>{'th'}</th>
|
||||||
<th className='sortable' onMouseEnter={termtip.bind(null, 'frame shift drive')} onMouseLeave={hide} onClick={sortShips('standard', 2)}>{'fsd'}</th>
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'frame shift drive')} onMouseLeave={hide} onClick={sortShips('standard', 2)}>{'fsd'}</th>
|
||||||
<th className='sortable' onMouseEnter={termtip.bind(null, 'life support')} onMouseLeave={hide} onClick={sortShips('standard', 3)}>{'ls'}</th>
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'life support')} onMouseLeave={hide} onClick={sortShips('standard', 3)}>{'ls'}</th>
|
||||||
<th className='sortable' onMouseEnter={termtip.bind(null, 'power distriubtor')} onMouseLeave={hide} onClick={sortShips('standard', 4)}>{'pd'}</th>
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'power distriubtor')} onMouseLeave={hide} onClick={sortShips('standard', 4)}>{'pd'}</th>
|
||||||
<th className='sortable' onMouseEnter={termtip.bind(null, 'sensors')} onMouseLeave={hide} onClick={sortShips('standard', 5)}>{'s'}</th>
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'sensors')} onMouseLeave={hide} onClick={sortShips('standard', 5)}>{'s'}</th>
|
||||||
|
<th className='sortable' onMouseEnter={termtip.bind(null, 'fuel tank')} onMouseLeave={hide} onClick={sortShips('standard', 6)}>{'ft'}</th>
|
||||||
<th className='sortable lft' onClick={sortShips('hp',1)}>{translate('S')}</th>
|
<th className='sortable lft' onClick={sortShips('hp',1)}>{translate('S')}</th>
|
||||||
<th className='sortable' onClick={sortShips('hp', 2)}>{translate('M')}</th>
|
<th className='sortable' onClick={sortShips('hp', 2)}>{translate('M')}</th>
|
||||||
<th className='sortable' onClick={sortShips('hp', 3)}>{translate('L')}</th>
|
<th className='sortable' onClick={sortShips('hp', 3)}>{translate('L')}</th>
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ export function totalJumpRange(mass, fsd, fuel) {
|
|||||||
* @param {number} multiplier Shield multiplier for ship (1 + shield boosters if any)
|
* @param {number} multiplier Shield multiplier for ship (1 + shield boosters if any)
|
||||||
* @return {number} Approximate shield strengh in MJ
|
* @return {number} Approximate shield strengh in MJ
|
||||||
*/
|
*/
|
||||||
export function shieldStrength(mass, baseShield, sg, multiplier) {
|
export function shieldStrength(mass, baseShield, sg, multiplier, ship) {
|
||||||
// sg might be a module or a template; handle either here
|
// sg might be a module or a template; handle either here
|
||||||
let minMass = sg instanceof Module ? sg.getMinMass() : sg.minmass;
|
let minMass = sg instanceof Module ? sg.getMinMass() : sg.minmass;
|
||||||
let optMass = sg instanceof Module ? sg.getOptMass() : sg.optmass;
|
let optMass = sg instanceof Module ? sg.getOptMass() : sg.optmass;
|
||||||
@@ -55,7 +55,17 @@ export function shieldStrength(mass, baseShield, sg, multiplier) {
|
|||||||
let minMul = sg instanceof Module ? sg.getMinMul() : sg.minmul;
|
let minMul = sg instanceof Module ? sg.getMinMul() : sg.minmul;
|
||||||
let optMul = sg instanceof Module ? sg.getOptMul() : sg.optmul;
|
let optMul = sg instanceof Module ? sg.getOptMul() : sg.optmul;
|
||||||
let maxMul = sg instanceof Module ? sg.getMaxMul() : sg.maxmul;
|
let maxMul = sg instanceof Module ? sg.getMaxMul() : sg.maxmul;
|
||||||
|
if (ship) {
|
||||||
|
for (const i of ship.hardpoints) {
|
||||||
|
if (!i.maxClass) {
|
||||||
|
if (i.grp === 'sb' || (i.m && i.m.grp === 'sb')) {
|
||||||
|
if (!isNaN(i.m.getModValue('optmul'))) {
|
||||||
|
optMul += i.m.getModValue('optmul') / 10000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
let xnorm = Math.min(1, (maxMass - mass) / (maxMass - minMass));
|
let xnorm = Math.min(1, (maxMass - mass) / (maxMass - minMass));
|
||||||
let exponent = Math.log((optMul - minMul) / (maxMul - minMul)) / Math.log(Math.min(1, (maxMass - optMass) / (maxMass - minMass)));
|
let exponent = Math.log((optMul - minMul) / (maxMul - minMul)) / Math.log(Math.min(1, (maxMass - optMass) / (maxMass - minMass)));
|
||||||
let ynorm = Math.pow(xnorm, exponent);
|
let ynorm = Math.pow(xnorm, exponent);
|
||||||
@@ -351,7 +361,7 @@ export function shieldMetrics(ship, sys) {
|
|||||||
boosterKinDmg = boosterKinDmg > 0.7 ? boosterKinDmg : 0.7 - (0.7 - boosterKinDmg) / 2;
|
boosterKinDmg = boosterKinDmg > 0.7 ? boosterKinDmg : 0.7 - (0.7 - boosterKinDmg) / 2;
|
||||||
boosterThermDmg = boosterThermDmg > 0.7 ? boosterThermDmg : 0.7 - (0.7 - boosterThermDmg) / 2;
|
boosterThermDmg = boosterThermDmg > 0.7 ? boosterThermDmg : 0.7 - (0.7 - boosterThermDmg) / 2;
|
||||||
|
|
||||||
const generatorStrength = this.shieldStrength(ship.hullMass, ship.baseShieldStrength, shieldGenerator, 1);
|
const generatorStrength = this.shieldStrength(ship.hullMass, ship.baseShieldStrength, shieldGenerator, 1, ship);
|
||||||
const boostersStrength = generatorStrength * boost;
|
const boostersStrength = generatorStrength * boost;
|
||||||
|
|
||||||
// Recover time is the time taken to go from 0 to 50%. It includes a 16-second wait before shields start to recover
|
// Recover time is the time taken to go from 0 to 50%. It includes a 16-second wait before shields start to recover
|
||||||
|
|||||||
@@ -9,12 +9,16 @@ export const StandardArray = [
|
|||||||
'pd', // Power Distributor
|
'pd', // Power Distributor
|
||||||
's', // Sensors
|
's', // Sensors
|
||||||
'ft', // Fuel Tank
|
'ft', // Fuel Tank
|
||||||
|
'gpp', // Guardian Hybrid Power Plant
|
||||||
|
'gpd' // Guardian Hybrid Power Distributor
|
||||||
];
|
];
|
||||||
|
|
||||||
// Map to lookup group labels/names for component grp, used for JSON Serialization
|
// Map to lookup group labels/names for component grp, used for JSON Serialization
|
||||||
export const ModuleGroupToName = {
|
export const ModuleGroupToName = {
|
||||||
// Standard
|
// Standard
|
||||||
pp: 'Power Plant',
|
pp: 'Power Plant',
|
||||||
|
gpp: 'Guardian Hybrid Power Plant',
|
||||||
|
gpd: 'Guardian Hybrid Power Distributor',
|
||||||
t: 'Thrusters',
|
t: 'Thrusters',
|
||||||
fsd: 'Frame Shift Drive',
|
fsd: 'Frame Shift Drive',
|
||||||
ls: 'Life Support',
|
ls: 'Life Support',
|
||||||
@@ -58,19 +62,27 @@ export const ModuleGroupToName = {
|
|||||||
cm: 'Countermeasure',
|
cm: 'Countermeasure',
|
||||||
ec: 'Electronic Countermeasure',
|
ec: 'Electronic Countermeasure',
|
||||||
fc: 'Fragment Cannon',
|
fc: 'Fragment Cannon',
|
||||||
|
rfl: 'Remote Release Flak Launcher',
|
||||||
hs: 'Heat Sink Launcher',
|
hs: 'Heat Sink Launcher',
|
||||||
ws: 'Frame Shift Wake Scanner',
|
ws: 'Frame Shift Wake Scanner',
|
||||||
kw: 'Kill Warrant Scanner',
|
kw: 'Kill Warrant Scanner',
|
||||||
nl: 'Mine Launcher',
|
nl: 'Mine Launcher',
|
||||||
ml: 'Mining Laser',
|
ml: 'Mining Laser',
|
||||||
mr: 'Missile Rack',
|
mr: 'Missile Rack',
|
||||||
|
axmr: 'AX Missile Rack',
|
||||||
pa: 'Plasma Accelerator',
|
pa: 'Plasma Accelerator',
|
||||||
po: 'Point Defence',
|
po: 'Point Defence',
|
||||||
mc: 'Multi-cannon',
|
mc: 'Multi-cannon',
|
||||||
|
axmc: 'AX Multi-cannon',
|
||||||
pl: 'Pulse Laser',
|
pl: 'Pulse Laser',
|
||||||
rg: 'Rail Gun',
|
rg: 'Rail Gun',
|
||||||
sb: 'Shield Booster',
|
sb: 'Shield Booster',
|
||||||
tp: 'Torpedo Pylon'
|
tp: 'Torpedo Pylon',
|
||||||
|
sfn: 'Shutdown Field Neutraliser',
|
||||||
|
xs: 'Xeno Scanner',
|
||||||
|
rcpl: 'Recon Limpet Controller',
|
||||||
|
gpc: 'Guardian Plasma Charger',
|
||||||
|
ggc: 'Guardian Gauss Cannon',
|
||||||
};
|
};
|
||||||
|
|
||||||
let GrpNameToCodeMap = {};
|
let GrpNameToCodeMap = {};
|
||||||
|
|||||||
@@ -64,7 +64,6 @@ export default class Module {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sanitise the resultant value to 4dp equivalent
|
// Sanitise the resultant value to 4dp equivalent
|
||||||
return isNaN(result) ? result : Math.round(result);
|
return isNaN(result) ? result : Math.round(result);
|
||||||
}
|
}
|
||||||
@@ -79,6 +78,9 @@ export default class Module {
|
|||||||
if (!this.mods) {
|
if (!this.mods) {
|
||||||
this.mods = {};
|
this.mods = {};
|
||||||
}
|
}
|
||||||
|
if (!this.origVals) {
|
||||||
|
this.origVals = {};
|
||||||
|
}
|
||||||
if (valueiswithspecial && this.blueprint && this.blueprint.special) {
|
if (valueiswithspecial && this.blueprint && this.blueprint.special) {
|
||||||
// This module has a special effect, see if we need to alter the stored value
|
// This module has a special effect, see if we need to alter the stored value
|
||||||
const modifierActions = Modifications.modifierActions[this.blueprint.special.edname];
|
const modifierActions = Modifications.modifierActions[this.blueprint.special.edname];
|
||||||
@@ -117,7 +119,6 @@ export default class Module {
|
|||||||
_getModifiedValue(name) {
|
_getModifiedValue(name) {
|
||||||
const modification = Modifications.modifications[name];
|
const modification = Modifications.modifications[name];
|
||||||
let result = this[name];
|
let result = this[name];
|
||||||
|
|
||||||
if (!result) {
|
if (!result) {
|
||||||
if (modification && modification.method === 'additive') {
|
if (modification && modification.method === 'additive') {
|
||||||
// Additive modifications start at 0 rather than NULL
|
// Additive modifications start at 0 rather than NULL
|
||||||
@@ -135,7 +136,7 @@ export default class Module {
|
|||||||
if (modification.type === 'percentage') {
|
if (modification.type === 'percentage') {
|
||||||
modValue = this.getModValue(name) / 10000;
|
modValue = this.getModValue(name) / 10000;
|
||||||
} else if (modification.type === 'numeric') {
|
} else if (modification.type === 'numeric') {
|
||||||
modValue = this.getModValue(name) / 100;
|
modValue = this.getModValue(name)/ 100;
|
||||||
} else {
|
} else {
|
||||||
modValue = this.getModValue(name);
|
modValue = this.getModValue(name);
|
||||||
}
|
}
|
||||||
@@ -721,4 +722,13 @@ export default class Module {
|
|||||||
getTime() {
|
getTime() {
|
||||||
return this._getModifiedValue('time');
|
return this._getModifiedValue('time');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the hack time for this module, taking in to account modifications
|
||||||
|
* @return {string} the time for this module
|
||||||
|
*/
|
||||||
|
getHackTime() {
|
||||||
|
return this._getModifiedValue('hacktime');
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -63,7 +63,6 @@ export function standard(type, id) {
|
|||||||
if (!isNaN(type)) {
|
if (!isNaN(type)) {
|
||||||
type = StandardArray[type];
|
type = StandardArray[type];
|
||||||
}
|
}
|
||||||
|
|
||||||
let s = Modules.standard[type].find(e => e.id == id || (e.class == id.charAt(0) && e.rating == id.charAt(1)));
|
let s = Modules.standard[type].find(e => e.id == id || (e.class == id.charAt(0) && e.rating == id.charAt(1)));
|
||||||
if (s) {
|
if (s) {
|
||||||
s = new Module({ template: s });
|
s = new Module({ template: s });
|
||||||
@@ -196,6 +195,29 @@ export function findInternal(groupName, clss, rating, name) {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Finds an internal module based on Class, Rating, Group and/or name.
|
||||||
|
* At least one of Group name or unique module name must be provided.
|
||||||
|
* will start searching at specified class and proceed lower until a
|
||||||
|
* module is found or 0 is hit
|
||||||
|
* Uses findInternal internally
|
||||||
|
*
|
||||||
|
* @param {String} groupName [Optional] Full name or abbreviated name for module group
|
||||||
|
* @param {integer} clss module Class
|
||||||
|
* @param {String} rating module Rating
|
||||||
|
* @param {String} name [Optional] Long/unique name for module -e.g. 'Advanced Discover Scanner'
|
||||||
|
* @return {Object} The module if found, null if not found
|
||||||
|
*/
|
||||||
|
export function findMaxInternal(groupName, clss, rating, name) {
|
||||||
|
let foundModule = null;
|
||||||
|
let currentClss = clss;
|
||||||
|
while (currentClss > 0 && foundModule == null) {
|
||||||
|
foundModule = findInternal(groupName, currentClss, rating, name);
|
||||||
|
currentClss = currentClss - 1;
|
||||||
|
}
|
||||||
|
return foundModule;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Finds an internal Module ID based on Class, Rating, Group and/or name.
|
* Finds an internal Module ID based on Class, Rating, Group and/or name.
|
||||||
* At least one ofGroup name or unique module name must be provided
|
* At least one ofGroup name or unique module name must be provided
|
||||||
|
|||||||
@@ -241,7 +241,7 @@ export default class Ship {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// TODO Not accurate if the ship has modified shield boosters
|
// TODO Not accurate if the ship has modified shield boosters
|
||||||
return Calc.shieldStrength(this.hullMass, this.baseShieldStrength, sg, 1 + (multiplierDelta || 0));
|
return Calc.shieldStrength(this.hullMass, this.baseShieldStrength, sg, 1 + (multiplierDelta || 0), this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -437,12 +437,15 @@ export default class Ship {
|
|||||||
m.blueprint = bp;
|
m.blueprint = bp;
|
||||||
this.clearModifications(m);
|
this.clearModifications(m);
|
||||||
// Set any hidden items for the blueprint now
|
// Set any hidden items for the blueprint now
|
||||||
const features = m.blueprint.grades[m.blueprint.grade].features;
|
if (m.blueprint.grades[m.blueprint.grade] && m.blueprint.grades[m.blueprint.grade].features) {
|
||||||
for (const featureName in features) {
|
const features = m.blueprint.grades[m.blueprint.grade].features;
|
||||||
if (Modifications.modifications[featureName].hidden) {
|
for (const featureName in features) {
|
||||||
this.setModification(m, featureName, bp.grades[bp.grade].features[featureName][0]);
|
if (Modifications.modifications[featureName].hidden) {
|
||||||
|
this.setModification(m, featureName, bp.grades[bp.grade].features[featureName][0]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.updateModificationsString();
|
this.updateModificationsString();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -464,7 +467,17 @@ export default class Ship {
|
|||||||
if (m.blueprint) {
|
if (m.blueprint) {
|
||||||
m.blueprint.special = special;
|
m.blueprint.special = special;
|
||||||
}
|
}
|
||||||
this.recalculateDps().recalculateHps().recalculateEps();
|
this.updatePowerGenerated()
|
||||||
|
.updatePowerUsed()
|
||||||
|
.recalculateMass()
|
||||||
|
.updateJumpStats()
|
||||||
|
.recalculateShield()
|
||||||
|
.recalculateShieldCells()
|
||||||
|
.recalculateArmour()
|
||||||
|
.recalculateDps()
|
||||||
|
.recalculateEps()
|
||||||
|
.recalculateHps()
|
||||||
|
.updateMovement();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -490,7 +503,6 @@ export default class Ship {
|
|||||||
// Value passed is invalid; reset it to 0
|
// Value passed is invalid; reset it to 0
|
||||||
value = 0;
|
value = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle special cases
|
// Handle special cases
|
||||||
if (name === 'pgen') {
|
if (name === 'pgen') {
|
||||||
// Power generation
|
// Power generation
|
||||||
@@ -569,6 +581,7 @@ export default class Ship {
|
|||||||
// Reset Cumulative stats
|
// Reset Cumulative stats
|
||||||
this.fuelCapacity = 0;
|
this.fuelCapacity = 0;
|
||||||
this.cargoCapacity = 0;
|
this.cargoCapacity = 0;
|
||||||
|
this.passengerCapacity = 0;
|
||||||
this.ladenMass = 0;
|
this.ladenMass = 0;
|
||||||
this.armour = this.baseArmour;
|
this.armour = this.baseArmour;
|
||||||
this.shield = this.baseShieldStrength;
|
this.shield = this.baseShieldStrength;
|
||||||
@@ -616,7 +629,7 @@ export default class Ship {
|
|||||||
standard[i].cat = 0;
|
standard[i].cat = 0;
|
||||||
standard[i].priority = priorities && priorities[i + 1] ? priorities[i + 1] * 1 : 0;
|
standard[i].priority = priorities && priorities[i + 1] ? priorities[i + 1] * 1 : 0;
|
||||||
standard[i].type = 'SYS';
|
standard[i].type = 'SYS';
|
||||||
standard[i].m = null; // Resetting 'old' modul if there was one
|
standard[i].m = null; // Resetting 'old' module if there was one
|
||||||
standard[i].discountedCost = 0;
|
standard[i].discountedCost = 0;
|
||||||
if (comps) {
|
if (comps) {
|
||||||
let module = ModuleUtils.standard(i, comps.standard[i]);
|
let module = ModuleUtils.standard(i, comps.standard[i]);
|
||||||
@@ -1188,6 +1201,7 @@ export default class Ship {
|
|||||||
let unladenMass = this.hullMass;
|
let unladenMass = this.hullMass;
|
||||||
let cargoCapacity = 0;
|
let cargoCapacity = 0;
|
||||||
let fuelCapacity = 0;
|
let fuelCapacity = 0;
|
||||||
|
let passengerCapacity = 0;
|
||||||
|
|
||||||
unladenMass += this.bulkheads.m.getMass();
|
unladenMass += this.bulkheads.m.getMass();
|
||||||
|
|
||||||
@@ -1209,6 +1223,10 @@ export default class Ship {
|
|||||||
fuelCapacity += slot.m.fuel;
|
fuelCapacity += slot.m.fuel;
|
||||||
} else if (slot.m.grp === 'cr') {
|
} else if (slot.m.grp === 'cr') {
|
||||||
cargoCapacity += slot.m.cargo;
|
cargoCapacity += slot.m.cargo;
|
||||||
|
} else if (slot.m.grp.slice(0,2) === 'pc') {
|
||||||
|
if (slot.m.passengers) {
|
||||||
|
passengerCapacity += slot.m.passengers;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1224,6 +1242,7 @@ export default class Ship {
|
|||||||
this.unladenMass = unladenMass;
|
this.unladenMass = unladenMass;
|
||||||
this.cargoCapacity = cargoCapacity;
|
this.cargoCapacity = cargoCapacity;
|
||||||
this.fuelCapacity = fuelCapacity;
|
this.fuelCapacity = fuelCapacity;
|
||||||
|
this.passengerCapacity = passengerCapacity;
|
||||||
this.ladenMass = unladenMass + fuelCapacity + cargoCapacity;
|
this.ladenMass = unladenMass + fuelCapacity + cargoCapacity;
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import * as ModuleUtils from './ModuleUtils';
|
import * as ModuleUtils from './ModuleUtils'
|
||||||
import { canMount } from '../utils/SlotFunctions';
|
import { canMount } from '../utils/SlotFunctions'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Standard / typical role for multi-purpose or combat (if shielded with better bulkheads)
|
* Standard / typical role for multi-purpose or combat (if shielded with better bulkheads)
|
||||||
@@ -7,20 +7,20 @@ import { canMount } from '../utils/SlotFunctions';
|
|||||||
* @param {Boolean} shielded True if shield generator should be included
|
* @param {Boolean} shielded True if shield generator should be included
|
||||||
* @param {integer} bulkheadIndex Bulkhead to use see Constants.BulkheadNames
|
* @param {integer} bulkheadIndex Bulkhead to use see Constants.BulkheadNames
|
||||||
*/
|
*/
|
||||||
export function multiPurpose(ship, shielded, bulkheadIndex) {
|
export function multiPurpose (ship, shielded, bulkheadIndex) {
|
||||||
ship.useStandard('A')
|
ship.useStandard('A')
|
||||||
.use(ship.standard[3], ModuleUtils.standard(3, ship.standard[3].maxClass + 'D')) // D Life Support
|
.use(ship.standard[3], ModuleUtils.standard(3, ship.standard[3].maxClass + 'D')) // D Life Support
|
||||||
.use(ship.standard[5], ModuleUtils.standard(5, ship.standard[5].maxClass + 'D')) // D Sensors
|
.use(ship.standard[5], ModuleUtils.standard(5, ship.standard[5].maxClass + 'D')) // D Sensors
|
||||||
.useBulkhead(bulkheadIndex);
|
.useBulkhead(bulkheadIndex)
|
||||||
|
|
||||||
if (shielded) {
|
if (shielded) {
|
||||||
ship.internal.some(function(slot) {
|
ship.internal.some(function (slot) {
|
||||||
if (canMount(ship, slot, 'sg')) { // Assuming largest slot can hold an eligible shield
|
if (canMount(ship, slot, 'sg')) { // Assuming largest slot can hold an eligible shield
|
||||||
ship.use(slot, ModuleUtils.findInternal('sg', slot.maxClass, 'A'));
|
ship.use(slot, ModuleUtils.findInternal('sg', slot.maxClass, 'A'))
|
||||||
ship.setSlotEnabled(slot, true);
|
ship.setSlotEnabled(slot, true)
|
||||||
return true;
|
return true
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -30,40 +30,54 @@ export function multiPurpose(ship, shielded, bulkheadIndex) {
|
|||||||
* @param {Boolean} shielded True if shield generator should be included
|
* @param {Boolean} shielded True if shield generator should be included
|
||||||
* @param {Object} standardOpts [Optional] Standard module optional overrides
|
* @param {Object} standardOpts [Optional] Standard module optional overrides
|
||||||
*/
|
*/
|
||||||
export function trader(ship, shielded, standardOpts) {
|
export function trader (ship, shielded, standardOpts) {
|
||||||
let usedSlots = [],
|
let usedSlots = []
|
||||||
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass);
|
let bstCount = 2
|
||||||
|
let sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass)
|
||||||
|
ship.useStandard('A')
|
||||||
|
.use(ship.standard[3], ModuleUtils.standard(3, ship.standard[3].maxClass + 'D')) // D Life Support
|
||||||
|
.use(ship.standard[1], ModuleUtils.standard(1, ship.standard[1].maxClass + 'D')) // D Life Support
|
||||||
|
.use(ship.standard[4], ModuleUtils.standard(4, ship.standard[4].maxClass + 'D')) // D Life Support
|
||||||
|
.use(ship.standard[5], ModuleUtils.standard(5, ship.standard[5].maxClass + 'D')) // D Sensors
|
||||||
|
|
||||||
// Shield generator if required
|
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
if (shielded) {
|
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
.filter(a => (!a.eligible) || a.eligible.sg)
|
||||||
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
.filter(a => a.maxClass >= sg.class)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sg)
|
.sort((a, b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass))
|
||||||
.filter(a => a.maxClass >= sg.class)
|
shieldInternals.some(function (slot) {
|
||||||
.sort((a,b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass));
|
if (canMount(ship, slot, 'sg')) { // Assuming largest slot can hold an eligible shield
|
||||||
for (let i = 0; i < shieldInternals.length; i++) {
|
const shield = ModuleUtils.findInternal('sg', slot.maxClass, 'A')
|
||||||
if (canMount(ship, shieldInternals[i], 'sg')) {
|
if (shield && shield.maxmass > ship.hullMass) {
|
||||||
ship.use(shieldInternals[i], sg);
|
ship.use(slot, shield)
|
||||||
usedSlots.push(shieldInternals[i]);
|
ship.setSlotEnabled(slot, true)
|
||||||
break;
|
usedSlots.push(slot)
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
})
|
||||||
|
|
||||||
// Fill the empty internals with cargo racks
|
// Fill the empty internals with cargo racks
|
||||||
for (let i = ship.internal.length; i--;) {
|
for (let i = ship.internal.length; i--;) {
|
||||||
let slot = ship.internal[i];
|
let slot = ship.internal[i]
|
||||||
if (usedSlots.indexOf(slot) == -1 && canMount(ship, slot, 'cr')) {
|
if (usedSlots.indexOf(slot) == -1 && canMount(ship, slot, 'cr')) {
|
||||||
ship.use(slot, ModuleUtils.findInternal('cr', slot.maxClass, 'E'));
|
ship.use(slot, ModuleUtils.findInternal('cr', slot.maxClass, 'E'))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Empty the hardpoints
|
// Empty the hardpoints
|
||||||
for (let s of ship.hardpoints) {
|
for (let s of ship.hardpoints) {
|
||||||
ship.use(s, null);
|
ship.use(s, null)
|
||||||
}
|
}
|
||||||
|
for (let s of ship.hardpoints) {
|
||||||
ship.useLightestStandard(standardOpts);
|
if (s.maxClass == 0 && bstCount) { // Mount up to 2 boosters
|
||||||
|
ship.use(s, ModuleUtils.hardpoints('04'))
|
||||||
|
bstCount--
|
||||||
|
} else {
|
||||||
|
ship.use(s, null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ship.useLightestStandard(standardOpts);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -71,127 +85,127 @@ export function trader(ship, shielded, standardOpts) {
|
|||||||
* @param {Ship} ship Ship instance
|
* @param {Ship} ship Ship instance
|
||||||
* @param {Boolean} planetary True if Planetary Vehicle Hangar (PVH) should be included
|
* @param {Boolean} planetary True if Planetary Vehicle Hangar (PVH) should be included
|
||||||
*/
|
*/
|
||||||
export function explorer(ship, planetary) {
|
export function explorer (ship, planetary) {
|
||||||
let standardOpts = { ppRating: 'A' },
|
let standardOpts = {ppRating: 'A'},
|
||||||
heatSinkCount = 2, // Fit 2 heat sinks if possible
|
heatSinkCount = 2, // Fit 2 heat sinks if possible
|
||||||
usedSlots = [],
|
usedSlots = [],
|
||||||
sgSlot,
|
sgSlot,
|
||||||
fuelScoopSlot,
|
fuelScoopSlot,
|
||||||
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass);
|
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass)
|
||||||
|
|
||||||
if (!planetary) { // Non-planetary explorers don't really need to boost
|
if (!planetary) { // Non-planetary explorers don't really need to boost
|
||||||
standardOpts.pd = '1D';
|
standardOpts.pd = '1D'
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cargo hatch can be disabled
|
// Cargo hatch can be disabled
|
||||||
ship.setSlotEnabled(ship.cargoHatch, false);
|
ship.setSlotEnabled(ship.cargoHatch, false)
|
||||||
|
|
||||||
// Advanced Discovery Scanner - class 1 or higher
|
// Advanced Discovery Scanner - class 1 or higher
|
||||||
const adsOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const adsOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const adsInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const adsInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sc)
|
.filter(a => (!a.eligible) || a.eligible.sc)
|
||||||
.sort((a,b) => adsOrder.indexOf(a.maxClass) - adsOrder.indexOf(b.maxClass));
|
.sort((a, b) => adsOrder.indexOf(a.maxClass) - adsOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < adsInternals.length; i++) {
|
for (let i = 0; i < adsInternals.length; i++) {
|
||||||
if (canMount(ship, adsInternals[i], 'sc')) {
|
if (canMount(ship, adsInternals[i], 'sc')) {
|
||||||
ship.use(adsInternals[i], ModuleUtils.internal('2f'));
|
ship.use(adsInternals[i], ModuleUtils.internal('2f'))
|
||||||
usedSlots.push(adsInternals[i]);
|
usedSlots.push(adsInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (planetary) {
|
if (planetary) {
|
||||||
// Planetary Vehicle Hangar - class 2 or higher
|
// Planetary Vehicle Hangar - class 2 or higher
|
||||||
const pvhOrder = [2, 3, 4, 5, 6, 7, 8, 1];
|
const pvhOrder = [2, 3, 4, 5, 6, 7, 8, 1]
|
||||||
const pvhInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const pvhInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.pv)
|
.filter(a => (!a.eligible) || a.eligible.pv)
|
||||||
.sort((a,b) => pvhOrder.indexOf(a.maxClass) - pvhOrder.indexOf(b.maxClass));
|
.sort((a, b) => pvhOrder.indexOf(a.maxClass) - pvhOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < pvhInternals.length; i++) {
|
for (let i = 0; i < pvhInternals.length; i++) {
|
||||||
if (canMount(ship, pvhInternals[i], 'pv')) {
|
if (canMount(ship, pvhInternals[i], 'pv')) {
|
||||||
// Planetary Vehical Hangar only has even classes
|
// Planetary Vehical Hangar only has even classes
|
||||||
const pvhClass = pvhInternals[i].maxClass % 2 === 1 ? pvhInternals[i].maxClass - 1 : pvhInternals[i].maxClass;
|
const pvhClass = pvhInternals[i].maxClass % 2 === 1 ? pvhInternals[i].maxClass - 1 : pvhInternals[i].maxClass
|
||||||
ship.use(pvhInternals[i], ModuleUtils.findInternal('pv', pvhClass, 'G')); // G is lower mass
|
ship.use(pvhInternals[i], ModuleUtils.findInternal('pv', pvhClass, 'G')) // G is lower mass
|
||||||
ship.setSlotEnabled(pvhInternals[i], false); // Disable power for Planetary Vehical Hangar
|
ship.setSlotEnabled(pvhInternals[i], false) // Disable power for Planetary Vehical Hangar
|
||||||
usedSlots.push(pvhInternals[i]);
|
usedSlots.push(pvhInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shield generator
|
// Shield generator
|
||||||
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sg)
|
.filter(a => (!a.eligible) || a.eligible.sg)
|
||||||
.filter(a => a.maxClass >= sg.class)
|
.filter(a => a.maxClass >= sg.class)
|
||||||
.sort((a,b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass));
|
.sort((a, b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < shieldInternals.length; i++) {
|
for (let i = 0; i < shieldInternals.length; i++) {
|
||||||
if (canMount(ship, shieldInternals[i], 'sg')) {
|
if (canMount(ship, shieldInternals[i], 'sg')) {
|
||||||
ship.use(shieldInternals[i], sg);
|
ship.use(shieldInternals[i], sg)
|
||||||
usedSlots.push(shieldInternals[i]);
|
usedSlots.push(shieldInternals[i])
|
||||||
sgSlot = shieldInternals[i];
|
sgSlot = shieldInternals[i]
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Detailed Surface Scanner
|
// Detailed Surface Scanner
|
||||||
const dssOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const dssOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const dssInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const dssInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sc)
|
.filter(a => (!a.eligible) || a.eligible.sc)
|
||||||
.sort((a,b) => dssOrder.indexOf(a.maxClass) - dssOrder.indexOf(b.maxClass));
|
.sort((a, b) => dssOrder.indexOf(a.maxClass) - dssOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < dssInternals.length; i++) {
|
for (let i = 0; i < dssInternals.length; i++) {
|
||||||
if (canMount(ship, dssInternals[i], 'sc')) {
|
if (canMount(ship, dssInternals[i], 'sc')) {
|
||||||
ship.use(dssInternals[i], ModuleUtils.internal('2i'));
|
ship.use(dssInternals[i], ModuleUtils.internal('2i'))
|
||||||
usedSlots.push(dssInternals[i]);
|
usedSlots.push(dssInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fuel scoop - best possible
|
// Fuel scoop - best possible
|
||||||
const fuelScoopOrder = [8, 7, 6, 5, 4, 3, 2, 1];
|
const fuelScoopOrder = [8, 7, 6, 5, 4, 3, 2, 1]
|
||||||
const fuelScoopInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const fuelScoopInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.fs)
|
.filter(a => (!a.eligible) || a.eligible.fs)
|
||||||
.sort((a,b) => fuelScoopOrder.indexOf(a.maxClass) - fuelScoopOrder.indexOf(b.maxClass));
|
.sort((a, b) => fuelScoopOrder.indexOf(a.maxClass) - fuelScoopOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < fuelScoopInternals.length; i++) {
|
for (let i = 0; i < fuelScoopInternals.length; i++) {
|
||||||
if (canMount(ship, fuelScoopInternals[i], 'fs')) {
|
if (canMount(ship, fuelScoopInternals[i], 'fs')) {
|
||||||
ship.use(fuelScoopInternals[i], ModuleUtils.findInternal('fs', fuelScoopInternals[i].maxClass, 'A'));
|
ship.use(fuelScoopInternals[i], ModuleUtils.findInternal('fs', fuelScoopInternals[i].maxClass, 'A'))
|
||||||
usedSlots.push(fuelScoopInternals[i]);
|
usedSlots.push(fuelScoopInternals[i])
|
||||||
fuelScoopSlot = fuelScoopInternals[i];
|
fuelScoopSlot = fuelScoopInternals[i]
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// AFMUs - fill as they are 0-weight
|
// AFMUs - fill as they are 0-weight
|
||||||
const afmuOrder = [8, 7, 6, 5, 4, 3, 2, 1];
|
const afmuOrder = [8, 7, 6, 5, 4, 3, 2, 1]
|
||||||
const afmuInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const afmuInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.pc)
|
.filter(a => (!a.eligible) || a.eligible.pc)
|
||||||
.sort((a,b) => afmuOrder.indexOf(a.maxClass) - afmuOrder.indexOf(b.maxClass));
|
.sort((a, b) => afmuOrder.indexOf(a.maxClass) - afmuOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < afmuInternals.length; i++) {
|
for (let i = 0; i < afmuInternals.length; i++) {
|
||||||
if (canMount(ship, afmuInternals[i], 'am')) {
|
if (canMount(ship, afmuInternals[i], 'am')) {
|
||||||
ship.use(afmuInternals[i], ModuleUtils.findInternal('am', afmuInternals[i].maxClass, 'A'));
|
ship.use(afmuInternals[i], ModuleUtils.findInternal('am', afmuInternals[i].maxClass, 'A'))
|
||||||
usedSlots.push(afmuInternals[i]);
|
usedSlots.push(afmuInternals[i])
|
||||||
ship.setSlotEnabled(afmuInternals[i], false); // Disable power for AFM Unit
|
ship.setSlotEnabled(afmuInternals[i], false) // Disable power for AFM Unit
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let s of ship.hardpoints) {
|
for (let s of ship.hardpoints) {
|
||||||
if (s.maxClass == 0 && heatSinkCount) { // Mount up to 2 heatsinks
|
if (s.maxClass == 0 && heatSinkCount) { // Mount up to 2 heatsinks
|
||||||
ship.use(s, ModuleUtils.hardpoints('02'));
|
ship.use(s, ModuleUtils.hardpoints('02'))
|
||||||
ship.setSlotEnabled(s, heatSinkCount == 2); // Only enable a single Heatsink
|
ship.setSlotEnabled(s, heatSinkCount == 2) // Only enable a single Heatsink
|
||||||
heatSinkCount--;
|
heatSinkCount--
|
||||||
} else {
|
} else {
|
||||||
ship.use(s, null);
|
ship.use(s, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sgSlot && fuelScoopSlot) {
|
if (sgSlot && fuelScoopSlot) {
|
||||||
// The SG and Fuel scoop to not need to be powered at the same time
|
// The SG and Fuel scoop to not need to be powered at the same time
|
||||||
if (sgSlot.m.getPowerUsage() > fuelScoopSlot.m.getPowerUsage()) { // The Shield generator uses the most power
|
if (sgSlot.m.getPowerUsage() > fuelScoopSlot.m.getPowerUsage()) { // The Shield generator uses the most power
|
||||||
ship.setSlotEnabled(fuelScoopSlot, false);
|
ship.setSlotEnabled(fuelScoopSlot, false)
|
||||||
} else { // The Fuel scoop uses the most power
|
} else { // The Fuel scoop uses the most power
|
||||||
ship.setSlotEnabled(sgSlot, false);
|
ship.setSlotEnabled(sgSlot, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ship.useLightestStandard(standardOpts);
|
ship.useLightestStandard(standardOpts)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -199,187 +213,188 @@ export function explorer(ship, planetary) {
|
|||||||
* @param {Ship} ship Ship instance
|
* @param {Ship} ship Ship instance
|
||||||
* @param {Boolean} shielded True if shield generator should be included
|
* @param {Boolean} shielded True if shield generator should be included
|
||||||
*/
|
*/
|
||||||
export function miner(ship, shielded) {
|
export function miner (ship, shielded) {
|
||||||
let standardOpts = { ppRating: 'A' },
|
shielded = true
|
||||||
miningLaserCount = 2,
|
let standardOpts = {ppRating: 'A'},
|
||||||
usedSlots = [],
|
miningLaserCount = 2,
|
||||||
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass);
|
usedSlots = [],
|
||||||
|
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass)
|
||||||
|
|
||||||
// Cargo hatch should be enabled
|
// Cargo hatch should be enabled
|
||||||
ship.setSlotEnabled(ship.cargoHatch, true);
|
ship.setSlotEnabled(ship.cargoHatch, true)
|
||||||
|
|
||||||
// Largest possible refinery
|
// Largest possible refinery
|
||||||
const refineryOrder = [4, 5, 6, 7, 8, 3, 2, 1];
|
const refineryOrder = [4, 5, 6, 7, 8, 3, 2, 1]
|
||||||
const refineryInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const refineryInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.rf)
|
.filter(a => (!a.eligible) || a.eligible.rf)
|
||||||
.sort((a,b) => refineryOrder.indexOf(a.maxClass) - refineryOrder.indexOf(b.maxClass));
|
.sort((a, b) => refineryOrder.indexOf(a.maxClass) - refineryOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < refineryInternals.length; i++) {
|
for (let i = 0; i < refineryInternals.length; i++) {
|
||||||
if (canMount(ship, refineryInternals[i], 'rf')) {
|
if (canMount(ship, refineryInternals[i], 'rf')) {
|
||||||
ship.use(refineryInternals[i], ModuleUtils.findInternal('rf', Math.min(refineryInternals[i].maxClass, 4), 'A'));
|
ship.use(refineryInternals[i], ModuleUtils.findInternal('rf', Math.min(refineryInternals[i].maxClass, 4), 'A'))
|
||||||
usedSlots.push(refineryInternals[i]);
|
usedSlots.push(refineryInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prospector limpet controller - 3A if possible
|
// Prospector limpet controller - 3A if possible
|
||||||
const prospectorOrder = [3, 4, 5, 6, 7, 8, 2, 1];
|
const prospectorOrder = [3, 4, 5, 6, 7, 8, 2, 1]
|
||||||
const prospectorInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const prospectorInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.pc)
|
.filter(a => (!a.eligible) || a.eligible.pc)
|
||||||
.sort((a,b) => prospectorOrder.indexOf(a.maxClass) - prospectorOrder.indexOf(b.maxClass));
|
.sort((a, b) => prospectorOrder.indexOf(a.maxClass) - prospectorOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < prospectorInternals.length; i++) {
|
for (let i = 0; i < prospectorInternals.length; i++) {
|
||||||
if (canMount(ship, prospectorInternals[i], 'pc')) {
|
if (canMount(ship, prospectorInternals[i], 'pc')) {
|
||||||
// Prospector only has odd classes
|
// Prospector only has odd classes
|
||||||
const prospectorClass = prospectorInternals[i].maxClass % 2 === 0 ? prospectorInternals[i].maxClass - 1 : prospectorInternals[i].maxClass;
|
const prospectorClass = prospectorInternals[i].maxClass % 2 === 0 ? prospectorInternals[i].maxClass - 1 : prospectorInternals[i].maxClass
|
||||||
ship.use(prospectorInternals[i], ModuleUtils.findInternal('pc', prospectorClass, 'A'));
|
ship.use(prospectorInternals[i], ModuleUtils.findInternal('pc', prospectorClass, 'A'))
|
||||||
usedSlots.push(prospectorInternals[i]);
|
usedSlots.push(prospectorInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shield generator if required
|
// Shield generator if required
|
||||||
if (shielded) {
|
if (shielded) {
|
||||||
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sg)
|
.filter(a => (!a.eligible) || a.eligible.sg)
|
||||||
.filter(a => a.maxClass >= sg.class)
|
.filter(a => a.maxClass >= sg.class)
|
||||||
.sort((a,b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass));
|
.sort((a, b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < shieldInternals.length; i++) {
|
for (let i = 0; i < shieldInternals.length; i++) {
|
||||||
if (canMount(ship, shieldInternals[i], 'sg')) {
|
if (canMount(ship, shieldInternals[i], 'sg')) {
|
||||||
ship.use(shieldInternals[i], sg);
|
ship.use(shieldInternals[i], sg)
|
||||||
usedSlots.push(shieldInternals[i]);
|
usedSlots.push(shieldInternals[i])
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dual mining lasers of highest possible class; remove anything else
|
// Dual mining lasers of highest possible class; remove anything else
|
||||||
const miningLaserOrder = [2, 3, 4, 1, 0];
|
const miningLaserOrder = [2, 3, 4, 1, 0]
|
||||||
const miningLaserHardpoints = ship.hardpoints.concat().sort(function(a,b) {
|
const miningLaserHardpoints = ship.hardpoints.concat().sort(function (a, b) {
|
||||||
return miningLaserOrder.indexOf(a.maxClass) - miningLaserOrder.indexOf(b.maxClass);
|
return miningLaserOrder.indexOf(a.maxClass) - miningLaserOrder.indexOf(b.maxClass)
|
||||||
});
|
})
|
||||||
for (let s of miningLaserHardpoints) {
|
for (let s of miningLaserHardpoints) {
|
||||||
if (s.maxClass >= 1 && miningLaserCount) {
|
if (s.maxClass >= 1 && miningLaserCount) {
|
||||||
ship.use(s, ModuleUtils.hardpoints(s.maxClass >= 2 ? '2m' : '2l'));
|
ship.use(s, ModuleUtils.hardpoints(s.maxClass >= 2 ? '2m' : '2l'))
|
||||||
miningLaserCount--;
|
miningLaserCount--
|
||||||
} else {
|
} else {
|
||||||
ship.use(s, null);
|
ship.use(s, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Number of collector limpets required to be active is a function of the size of the ship and the power of the lasers
|
// Number of collector limpets required to be active is a function of the size of the ship and the power of the lasers
|
||||||
const miningLaserDps = ship.hardpoints.filter(h => h.m != null)
|
const miningLaserDps = ship.hardpoints.filter(h => h.m != null)
|
||||||
.reduce(function(a, b) {
|
.reduce(function (a, b) {
|
||||||
return a + b.m.getDps();
|
return a + b.m.getDps()
|
||||||
}, 0);
|
}, 0)
|
||||||
// Find out how many internal slots we have, and their potential cargo size
|
// Find out how many internal slots we have, and their potential cargo size
|
||||||
const potentialCargo = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const potentialCargo = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.cr)
|
.filter(a => (!a.eligible) || a.eligible.cr)
|
||||||
.map(b => Math.pow(2, b.maxClass));
|
.map(b => Math.pow(2, b.maxClass))
|
||||||
// One collector for each 1.25 DPS, multiply by 1.25 for medium ships and 1.5 for large ships as they have further to travel
|
// One collector for each 1.25 DPS, multiply by 1.25 for medium ships and 1.5 for large ships as they have further to travel
|
||||||
// 0 if we only have 1 cargo slot, otherwise minium of 1 and maximum of 6 (excluding size modifier)
|
// 0 if we only have 1 cargo slot, otherwise minium of 1 and maximum of 6 (excluding size modifier)
|
||||||
const sizeModifier = ship.class == 2 ? 1.2 : ship.class == 3 ? 1.5 : 1;
|
const sizeModifier = ship.class == 2 ? 1.2 : ship.class == 3 ? 1.5 : 1
|
||||||
let collectorLimpetsRequired = potentialCargo.length == 1 ? 0 : Math.ceil(sizeModifier * Math.min(6, Math.floor(miningLaserDps / 1.25)));
|
let collectorLimpetsRequired = potentialCargo.length == 1 ? 0 : Math.ceil(sizeModifier * Math.min(6, Math.floor(miningLaserDps / 1.25)))
|
||||||
|
|
||||||
if (collectorLimpetsRequired > 0) {
|
if (collectorLimpetsRequired > 0) {
|
||||||
const collectorOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const collectorOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const collectorInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const collectorInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.cc)
|
.filter(a => (!a.eligible) || a.eligible.cc)
|
||||||
.sort((a,b) => collectorOrder.indexOf(a.maxClass) - collectorOrder.indexOf(b.maxClass));
|
.sort((a, b) => collectorOrder.indexOf(a.maxClass) - collectorOrder.indexOf(b.maxClass))
|
||||||
// Always keep at least 2 slots free for cargo racks (1 for shielded)
|
// Always keep at least 2 slots free for cargo racks (1 for shielded)
|
||||||
for (let i = 0; i < collectorInternals.length - (shielded ? 1 : 2) && collectorLimpetsRequired > 0; i++) {
|
for (let i = 0; i < collectorInternals.length - (shielded ? 1 : 2) && collectorLimpetsRequired > 0; i++) {
|
||||||
if (canMount(ship, collectorInternals[i], 'cc')) {
|
if (canMount(ship, collectorInternals[i], 'cc')) {
|
||||||
// Collector only has odd classes
|
// Collector only has odd classes
|
||||||
const collectorClass = collectorInternals[i].maxClass % 2 === 0 ? collectorInternals[i].maxClass - 1 : collectorInternals[i].maxClass;
|
const collectorClass = collectorInternals[i].maxClass % 2 === 0 ? collectorInternals[i].maxClass - 1 : collectorInternals[i].maxClass
|
||||||
ship.use(collectorInternals[i], ModuleUtils.findInternal('cc', collectorClass, 'D'));
|
ship.use(collectorInternals[i], ModuleUtils.findInternal('cc', collectorClass, 'D'))
|
||||||
usedSlots.push(collectorInternals[i]);
|
usedSlots.push(collectorInternals[i])
|
||||||
collectorLimpetsRequired -= collectorInternals[i].m.maximum;
|
collectorLimpetsRequired -= collectorInternals[i].m.maximum
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Power distributor to power the mining lasers indefinitely
|
// Power distributor to power the mining lasers indefinitely
|
||||||
const wepRateRequired = ship.hardpoints.filter(h => h.m != null)
|
const wepRateRequired = ship.hardpoints.filter(h => h.m != null)
|
||||||
.reduce(function(a, b) {
|
.reduce(function (a, b) {
|
||||||
return a + b.m.getEps();
|
return a + b.m.getEps()
|
||||||
}, 0);
|
}, 0)
|
||||||
standardOpts.pd = ship.getAvailableModules().matchingPowerDist({ weprate: wepRateRequired }).id;
|
standardOpts.pd = ship.getAvailableModules().matchingPowerDist({weprate: wepRateRequired}).id
|
||||||
|
|
||||||
// Fill the empty internals with cargo racks
|
// Fill the empty internals with cargo racks
|
||||||
for (let i = ship.internal.length; i--;) {
|
for (let i = ship.internal.length; i--;) {
|
||||||
let slot = ship.internal[i];
|
let slot = ship.internal[i]
|
||||||
if (usedSlots.indexOf(slot) == -1 && canMount(ship, slot, 'cr')) {
|
if (usedSlots.indexOf(slot) == -1 && canMount(ship, slot, 'cr')) {
|
||||||
ship.use(slot, ModuleUtils.findInternal('cr', slot.maxClass, 'E'));
|
ship.use(slot, ModuleUtils.findInternal('cr', slot.maxClass, 'E'))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ship.useLightestStandard(standardOpts);
|
ship.useLightestStandard(standardOpts)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Racer Role
|
* Racer Role
|
||||||
* @param {Ship} ship Ship instance
|
* @param {Ship} ship Ship instance
|
||||||
*/
|
*/
|
||||||
export function racer(ship) {
|
export function racer (ship) {
|
||||||
let standardOpts = {},
|
let standardOpts = {},
|
||||||
usedSlots = [],
|
usedSlots = [],
|
||||||
sgSlot,
|
sgSlot,
|
||||||
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass);
|
sg = ship.getAvailableModules().lightestShieldGenerator(ship.hullMass)
|
||||||
|
|
||||||
// Cargo hatch can be disabled
|
// Cargo hatch can be disabled
|
||||||
ship.setSlotEnabled(ship.cargoHatch, false);
|
ship.setSlotEnabled(ship.cargoHatch, false)
|
||||||
|
|
||||||
// Shield generator
|
// Shield generator
|
||||||
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8];
|
const shieldOrder = [1, 2, 3, 4, 5, 6, 7, 8]
|
||||||
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
const shieldInternals = ship.internal.filter(a => usedSlots.indexOf(a) == -1)
|
||||||
.filter(a => (!a.eligible) || a.eligible.sg)
|
.filter(a => (!a.eligible) || a.eligible.sg)
|
||||||
.filter(a => a.maxClass >= sg.class)
|
.filter(a => a.maxClass >= sg.class)
|
||||||
.sort((a,b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass));
|
.sort((a, b) => shieldOrder.indexOf(a.maxClass) - shieldOrder.indexOf(b.maxClass))
|
||||||
for (let i = 0; i < shieldInternals.length; i++) {
|
for (let i = 0; i < shieldInternals.length; i++) {
|
||||||
if (canMount(ship, shieldInternals[i], 'sg')) {
|
if (canMount(ship, shieldInternals[i], 'sg')) {
|
||||||
ship.use(shieldInternals[i], sg);
|
ship.use(shieldInternals[i], sg)
|
||||||
usedSlots.push(shieldInternals[i]);
|
usedSlots.push(shieldInternals[i])
|
||||||
sgSlot = shieldInternals[i];
|
sgSlot = shieldInternals[i]
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Empty the hardpoints
|
// Empty the hardpoints
|
||||||
for (let s of ship.hardpoints) {
|
for (let s of ship.hardpoints) {
|
||||||
ship.use(s, null);
|
ship.use(s, null)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Empty the internals
|
// Empty the internals
|
||||||
for (let i = ship.internal.length; i--;) {
|
for (let i = ship.internal.length; i--;) {
|
||||||
let slot = ship.internal[i];
|
let slot = ship.internal[i]
|
||||||
if (usedSlots.indexOf(slot) == -1) {
|
if (usedSlots.indexOf(slot) == -1) {
|
||||||
ship.use(slot, null);
|
ship.use(slot, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Best thrusters
|
// Best thrusters
|
||||||
if (ship.standard[1].maxClass === 3) {
|
if (ship.standard[1].maxClass === 3) {
|
||||||
standardOpts.th = 'tz';
|
standardOpts.th = 'tz'
|
||||||
} else if (ship.standard[1].maxClass === 2) {
|
} else if (ship.standard[1].maxClass === 2) {
|
||||||
standardOpts.th = 'u0';
|
standardOpts.th = 'u0'
|
||||||
} else {
|
} else {
|
||||||
standardOpts.th = ship.standard[1].maxClass + 'A';
|
standardOpts.th = ship.standard[1].maxClass + 'A'
|
||||||
}
|
}
|
||||||
|
|
||||||
// Best power distributor for more boosting
|
// Best power distributor for more boosting
|
||||||
standardOpts.pd = ship.standard[4].maxClass + 'A';
|
standardOpts.pd = ship.standard[4].maxClass + 'A'
|
||||||
|
|
||||||
// Smallest possible FSD drive
|
// Smallest possible FSD drive
|
||||||
standardOpts.fsd = '2D';
|
standardOpts.fsd = '2D'
|
||||||
// Minimal fuel tank
|
// Minimal fuel tank
|
||||||
standardOpts.ft = '1C';
|
standardOpts.ft = '1C'
|
||||||
|
|
||||||
// Disable nearly everything
|
// Disable nearly everything
|
||||||
standardOpts.fsdDisabled = true;
|
standardOpts.fsdDisabled = true
|
||||||
standardOpts.sDisabled = true;
|
standardOpts.sDisabled = true
|
||||||
standardOpts.pdDisabled = true;
|
standardOpts.pdDisabled = true
|
||||||
standardOpts.lsDisabled = true;
|
standardOpts.lsDisabled = true
|
||||||
|
|
||||||
ship.useLightestStandard(standardOpts);
|
ship.useLightestStandard(standardOpts)
|
||||||
|
|
||||||
// Apply engineering to each module
|
// Apply engineering to each module
|
||||||
// ship.standard[1].m.blueprint = getBlueprint('Engine_Dirty', ship.standard[0]);
|
// ship.standard[1].m.blueprint = getBlueprint('Engine_Dirty', ship.standard[0]);
|
||||||
@@ -398,4 +413,3 @@ export function racer(ship) {
|
|||||||
// ship.standard[5].m.blueprint.grade = 5;
|
// ship.standard[5].m.blueprint.grade = 5;
|
||||||
// setBest(ship, ship.standard[5].m);
|
// setBest(ship, ship.standard[5].m);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,67 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Modifications } from 'coriolis-data/dist';
|
import { Modifications } from 'coriolis-data/dist';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate a tooltip with details of a blueprint's specials
|
||||||
|
* @param {Object} translate The translate object
|
||||||
|
* @param {Object} blueprint The blueprint at the required grade
|
||||||
|
* @param {string} grp The group of the module
|
||||||
|
* @param {Object} m The module to compare with
|
||||||
|
* @param specialName
|
||||||
|
* @returns {Object} The react components
|
||||||
|
*/
|
||||||
|
export function specialToolTip(translate, blueprint, grp, m, specialName) {
|
||||||
|
const effects = [];
|
||||||
|
if (!blueprint || !blueprint.features) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
if (m) {
|
||||||
|
// We also add in any benefits from specials that aren't covered above
|
||||||
|
if (m.blueprint) {
|
||||||
|
for (const feature in Modifications.modifierActions[specialName]) {
|
||||||
|
// if (!blueprint.features[feature] && !m.mods.feature) {
|
||||||
|
const featureDef = Modifications.modifications[feature];
|
||||||
|
if (featureDef && !featureDef.hidden) {
|
||||||
|
let symbol = '';
|
||||||
|
if (feature === 'jitter') {
|
||||||
|
symbol = '°';
|
||||||
|
} else if (featureDef.type === 'percentage') {
|
||||||
|
symbol = '%';
|
||||||
|
}
|
||||||
|
let current = m.getModValue(feature) - m.getModValue(feature, true);
|
||||||
|
if (featureDef.type === 'percentage') {
|
||||||
|
current = Math.round(current / 10) / 10;
|
||||||
|
} else if (featureDef.type === 'numeric') {
|
||||||
|
current /= 100;
|
||||||
|
}
|
||||||
|
const currentIsBeneficial = isValueBeneficial(feature, current);
|
||||||
|
|
||||||
|
effects.push(
|
||||||
|
<tr key={feature + '_specialTT'}>
|
||||||
|
<td style={{textAlign: 'left'}}>{translate(feature, grp)}</td>
|
||||||
|
<td> </td>
|
||||||
|
<td className={current === 0 ? '' : currentIsBeneficial ? 'secondary' : 'warning'}
|
||||||
|
style={{textAlign: 'right'}}>{current}{symbol}</td>
|
||||||
|
<td> </td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<table width='100%'>
|
||||||
|
<tbody>
|
||||||
|
{effects}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a tooltip with details of a blueprint's effects
|
* Generate a tooltip with details of a blueprint's effects
|
||||||
* @param {Object} translate The translate object
|
* @param {Object} translate The translate object
|
||||||
@@ -12,6 +73,9 @@ import { Modifications } from 'coriolis-data/dist';
|
|||||||
*/
|
*/
|
||||||
export function blueprintTooltip(translate, blueprint, engineers, grp, m) {
|
export function blueprintTooltip(translate, blueprint, engineers, grp, m) {
|
||||||
const effects = [];
|
const effects = [];
|
||||||
|
if (!blueprint || !blueprint.features) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
for (const feature in blueprint.features) {
|
for (const feature in blueprint.features) {
|
||||||
const featureIsBeneficial = isBeneficial(feature, blueprint.features[feature]);
|
const featureIsBeneficial = isBeneficial(feature, blueprint.features[feature]);
|
||||||
const featureDef = Modifications.modifications[feature];
|
const featureDef = Modifications.modifications[feature];
|
||||||
@@ -225,27 +289,13 @@ export function isValueBeneficial(feature, value) {
|
|||||||
*/
|
*/
|
||||||
export function getBlueprint(name, module) {
|
export function getBlueprint(name, module) {
|
||||||
// Start with a copy of the blueprint
|
// Start with a copy of the blueprint
|
||||||
const blueprint = JSON.parse(JSON.stringify(Modifications.blueprints[name]));
|
const findMod = val => Object.keys(Modifications.blueprints).find(elem => elem.toString().toLowerCase().search(val.toString().toLowerCase().replace(/(OutfittingFieldType_|persecond)/igm, '')) >= 0)
|
||||||
|
const found = Modifications.blueprints[findMod(name)];
|
||||||
|
if (!found || !found.fdname) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
const blueprint = JSON.parse(JSON.stringify(found));
|
||||||
if (module) {
|
if (module) {
|
||||||
if (module.grp === 'bh' || module.grp === 'hr' || module.grp === 'sg' || module.grp === 'psg' || module.grp === 'bsg') {
|
|
||||||
// Bulkheads, hull reinforcements and shield generators need to have their resistances altered by the base values
|
|
||||||
for (const grade in blueprint.grades) {
|
|
||||||
for (const feature in blueprint.grades[grade].features) {
|
|
||||||
if (feature === 'explres') {
|
|
||||||
blueprint.grades[grade].features[feature][0] *= (1 - module.explres);
|
|
||||||
blueprint.grades[grade].features[feature][1] *= (1 - module.explres);
|
|
||||||
}
|
|
||||||
if (feature === 'kinres') {
|
|
||||||
blueprint.grades[grade].features[feature][0] *= (1 - module.kinres);
|
|
||||||
blueprint.grades[grade].features[feature][1] *= (1 - module.kinres);
|
|
||||||
}
|
|
||||||
if (feature === 'thermres') {
|
|
||||||
blueprint.grades[grade].features[feature][0] *= (1 - module.thermres);
|
|
||||||
blueprint.grades[grade].features[feature][1] *= (1 - module.thermres);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (module.grp === 'sb') {
|
if (module.grp === 'sb') {
|
||||||
// Shield boosters are treated internally as straight modifiers, so rather than (for example)
|
// Shield boosters are treated internally as straight modifiers, so rather than (for example)
|
||||||
// being a 4% boost they are a 104% multiplier. We need to fix the values here so that they look
|
// being a 4% boost they are a 104% multiplier. We need to fix the values here so that they look
|
||||||
@@ -264,56 +314,31 @@ export function getBlueprint(name, module) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provide 'worst' primary modifications
|
* Provide 'percent' primary modifications
|
||||||
* @param {Object} ship The ship for which to perform the modifications
|
* @param {Object} ship The ship for which to perform the modifications
|
||||||
* @param {Object} m The module for which to perform the modifications
|
* @param {Object} m The module for which to perform the modifications
|
||||||
|
* @param {Number} percent The percent to set values to of full.
|
||||||
*/
|
*/
|
||||||
export function setWorst(ship, m) {
|
export function setPercent(ship, m, percent) {
|
||||||
ship.clearModifications(m);
|
|
||||||
const features = m.blueprint.grades[m.blueprint.grade].features;
|
|
||||||
for (const featureName in features) {
|
|
||||||
const value = features[featureName][0];
|
|
||||||
_setValue(ship, m, featureName, value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provide 'best' primary modifications
|
|
||||||
* @param {Object} ship The ship for which to perform the modifications
|
|
||||||
* @param {Object} m The module for which to perform the modifications
|
|
||||||
*/
|
|
||||||
export function setBest(ship, m) {
|
|
||||||
ship.clearModifications(m);
|
|
||||||
const features = m.blueprint.grades[m.blueprint.grade].features;
|
|
||||||
for (const featureName in features) {
|
|
||||||
const value = features[featureName][1];
|
|
||||||
_setValue(ship, m, featureName, value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provide 'extreme' primary modifications
|
|
||||||
* @param {Object} ship The ship for which to perform the modifications
|
|
||||||
* @param {Object} m The module for which to perform the modifications
|
|
||||||
*/
|
|
||||||
export function setExtreme(ship, m) {
|
|
||||||
ship.clearModifications(m);
|
ship.clearModifications(m);
|
||||||
|
// Pick given value as multiplier
|
||||||
|
const mult = percent / 100;
|
||||||
const features = m.blueprint.grades[m.blueprint.grade].features;
|
const features = m.blueprint.grades[m.blueprint.grade].features;
|
||||||
for (const featureName in features) {
|
for (const featureName in features) {
|
||||||
let value;
|
let value;
|
||||||
if (Modifications.modifications[featureName].higherbetter) {
|
if (Modifications.modifications[featureName].higherbetter) {
|
||||||
// Higher is better, but is this making it better or worse?
|
// Higher is better, but is this making it better or worse?
|
||||||
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
||||||
value = features[featureName][0];
|
value = features[featureName][1] + ((features[featureName][0] - features[featureName][1]) * mult);
|
||||||
} else {
|
} else {
|
||||||
value = features[featureName][1];
|
value = features[featureName][0] + ((features[featureName][1] - features[featureName][0]) * mult);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Higher is worse, but is this making it better or worse?
|
// Higher is worse, but is this making it better or worse?
|
||||||
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
||||||
value = features[featureName][1];
|
value = features[featureName][0] + ((features[featureName][1] - features[featureName][0]) * mult);
|
||||||
} else {
|
} else {
|
||||||
value = features[featureName][0];
|
value = features[featureName][1] + ((features[featureName][0] - features[featureName][1]) * mult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -327,30 +352,8 @@ export function setExtreme(ship, m) {
|
|||||||
* @param {Object} m The module for which to perform the modifications
|
* @param {Object} m The module for which to perform the modifications
|
||||||
*/
|
*/
|
||||||
export function setRandom(ship, m) {
|
export function setRandom(ship, m) {
|
||||||
ship.clearModifications(m);
|
|
||||||
// Pick a single value for our randomness
|
// Pick a single value for our randomness
|
||||||
const mult = Math.random();
|
setPercent(ship, m, Math.random() * 100);
|
||||||
const features = m.blueprint.grades[m.blueprint.grade].features;
|
|
||||||
for (const featureName in features) {
|
|
||||||
let value;
|
|
||||||
if (Modifications.modifications[featureName].higherbetter) {
|
|
||||||
// Higher is better, but is this making it better or worse?
|
|
||||||
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
|
||||||
value = features[featureName][1] + ((features[featureName][0] - features[featureName][1]) * mult);
|
|
||||||
} else {
|
|
||||||
value = features[featureName][0] + ((features[featureName][1] - features[featureName][0]) * mult);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Higher is worse, but is this making it better or worse?
|
|
||||||
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
|
||||||
value = features[featureName][0] + ((features[featureName][1] - features[featureName][0]) * mult);
|
|
||||||
} else {
|
|
||||||
value = features[featureName][1] + ((features[featureName][0] - features[featureName][1]) * mult);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_setValue(ship, m, featureName, value);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -369,3 +372,60 @@ function _setValue(ship, m, featureName, value) {
|
|||||||
ship.setModification(m, featureName, value);
|
ship.setModification(m, featureName, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide 'percent' primary query
|
||||||
|
* @param {Object} m The module for which to perform the query
|
||||||
|
* @returns {Number} percent The percentage indicator of current applied values.
|
||||||
|
*/
|
||||||
|
export function getPercent(m) {
|
||||||
|
let result = null;
|
||||||
|
const features = m.blueprint.grades[m.blueprint.grade].features;
|
||||||
|
for (const featureName in features) {
|
||||||
|
|
||||||
|
if (features[featureName][0] === features[featureName][1]) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let value = _getValue(m, featureName);
|
||||||
|
let mult;
|
||||||
|
if (Modifications.modifications[featureName].higherbetter) {
|
||||||
|
// Higher is better, but is this making it better or worse?
|
||||||
|
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
||||||
|
mult = Math.round((value - features[featureName][1]) / (features[featureName][0] - features[featureName][1]) * 100);
|
||||||
|
} else {
|
||||||
|
mult = Math.round((value - features[featureName][0]) / (features[featureName][1] - features[featureName][0]) * 100);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Higher is worse, but is this making it better or worse?
|
||||||
|
if (features[featureName][0] < 0 || (features[featureName][0] === 0 && features[featureName][1] < 0)) {
|
||||||
|
mult = Math.round((value - features[featureName][0]) / (features[featureName][1] - features[featureName][0]) * 100);
|
||||||
|
} else {
|
||||||
|
mult = Math.round((value - features[featureName][1]) / (features[featureName][0] - features[featureName][1]) * 100);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result && result != mult) {
|
||||||
|
return null;
|
||||||
|
} else if (result != mult) {
|
||||||
|
result = mult;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Query a feature value
|
||||||
|
* @param {Object} m The module for which to perform the query
|
||||||
|
* @param {string} featureName The feature being queried
|
||||||
|
*/
|
||||||
|
function _getValue(m, featureName) {
|
||||||
|
if (Modifications.modifications[featureName].type == 'percentage') {
|
||||||
|
return m.getModValue(featureName, true) / 10000;
|
||||||
|
} else if (Modifications.modifications[featureName].type == 'numeric') {
|
||||||
|
return m.getModValue(featureName, true) / 100;
|
||||||
|
} else {
|
||||||
|
return m.getModValue(featureName, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -35,13 +35,15 @@ const SHIP_FD_NAME_TO_CORIOLIS_NAME = {
|
|||||||
'Type6': 'type_6_transporter',
|
'Type6': 'type_6_transporter',
|
||||||
'Type7': 'type_7_transport',
|
'Type7': 'type_7_transport',
|
||||||
'Type9': 'type_9_heavy',
|
'Type9': 'type_9_heavy',
|
||||||
|
'Type9_Military': 'type_10_defender',
|
||||||
|
'TypeX': 'alliance_chieftain',
|
||||||
'Viper': 'viper',
|
'Viper': 'viper',
|
||||||
'Viper_MkIV': 'viper_mk_iv',
|
'Viper_MkIV': 'viper_mk_iv',
|
||||||
'Vulture': 'vulture'
|
'Vulture': 'vulture'
|
||||||
};
|
};
|
||||||
|
|
||||||
// Mapping from hardpoint class to name in companion API
|
// Mapping from hardpoint class to name in companion API
|
||||||
const HARDPOINT_NUM_TO_CLASS = {
|
export const HARDPOINT_NUM_TO_CLASS = {
|
||||||
0: 'Tiny',
|
0: 'Tiny',
|
||||||
1: 'Small',
|
1: 'Small',
|
||||||
2: 'Medium',
|
2: 'Medium',
|
||||||
@@ -104,7 +106,7 @@ function _moduleFromEdId(edId) {
|
|||||||
* @return {string} the Coriolis model of the ship
|
* @return {string} the Coriolis model of the ship
|
||||||
*/
|
*/
|
||||||
function _shipModelFromEDName(edName) {
|
function _shipModelFromEDName(edName) {
|
||||||
return SHIP_FD_NAME_TO_CORIOLIS_NAME[edName];
|
return SHIP_FD_NAME_TO_CORIOLIS_NAME[Object.keys(SHIP_FD_NAME_TO_CORIOLIS_NAME).find(elem => elem.toLowerCase() === edName.toLowerCase())];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -113,7 +115,7 @@ function _shipModelFromEDName(edName) {
|
|||||||
* @return {string} the Coriolis model of the ship
|
* @return {string} the Coriolis model of the ship
|
||||||
*/
|
*/
|
||||||
export function shipModelFromJson(json) {
|
export function shipModelFromJson(json) {
|
||||||
return _shipModelFromEDName(json.name);
|
return _shipModelFromEDName(json.name || json.Ship);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -142,29 +144,33 @@ export function shipFromJson(json) {
|
|||||||
ship.cargoHatch.priority = 4;
|
ship.cargoHatch.priority = 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let rootModule;
|
||||||
|
|
||||||
// Add the bulkheads
|
// Add the bulkheads
|
||||||
const armourJson = json.modules.Armour.module;
|
const armourJson = json.modules.Armour.module;
|
||||||
if (armourJson.name.endsWith('_Armour_Grade1')) {
|
if (armourJson.name.toLowerCase().endsWith('_armour_grade1')) {
|
||||||
ship.useBulkhead(0, true);
|
ship.useBulkhead(0, true);
|
||||||
} else if (armourJson.name.endsWith('_Armour_Grade2')) {
|
} else if (armourJson.name.toLowerCase().endsWith('_armour_grade2')) {
|
||||||
ship.useBulkhead(1, true);
|
ship.useBulkhead(1, true);
|
||||||
} else if (armourJson.name.endsWith('_Armour_Grade3')) {
|
} else if (armourJson.name.toLowerCase().endsWith('_armour_grade3')) {
|
||||||
ship.useBulkhead(2, true);
|
ship.useBulkhead(2, true);
|
||||||
} else if (armourJson.name.endsWith('_Armour_Mirrored')) {
|
} else if (armourJson.name.toLowerCase().endsWith('_armour_mirrored')) {
|
||||||
ship.useBulkhead(3, true);
|
ship.useBulkhead(3, true);
|
||||||
} else if (armourJson.name.endsWith('_Armour_Reactive')) {
|
} else if (armourJson.name.toLowerCase().endsWith('_armour_reactive')) {
|
||||||
ship.useBulkhead(4, true);
|
ship.useBulkhead(4, true);
|
||||||
} else {
|
} else {
|
||||||
throw 'Unknown bulkheads "' + armourJson.name + '"';
|
throw 'Unknown bulkheads "' + armourJson.name + '"';
|
||||||
}
|
}
|
||||||
ship.bulkheads.enabled = true;
|
ship.bulkheads.enabled = true;
|
||||||
if (armourJson.modifiers) _addModifications(ship.bulkheads.m, armourJson.modifiers, armourJson.recipeName, armourJson.recipeLevel);
|
rootModule = json.modules.Armour;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(ship.bulkheads.m, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
|
|
||||||
// Add the standard modules
|
// Add the standard modules
|
||||||
// Power plant
|
// Power plant
|
||||||
const powerplantJson = json.modules.PowerPlant.module;
|
const powerplantJson = json.modules.PowerPlant.module;
|
||||||
const powerplant = _moduleFromEdId(powerplantJson.id);
|
const powerplant = _moduleFromEdId(powerplantJson.id);
|
||||||
if (powerplantJson.modifiers) _addModifications(powerplant, powerplantJson.modifiers, powerplantJson.recipeName, powerplantJson.recipeLevel);
|
rootModule = json.modules.PowerPlant;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(powerplant, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[0], powerplant, true);
|
ship.use(ship.standard[0], powerplant, true);
|
||||||
ship.standard[0].enabled = powerplantJson.on === true;
|
ship.standard[0].enabled = powerplantJson.on === true;
|
||||||
ship.standard[0].priority = powerplantJson.priority;
|
ship.standard[0].priority = powerplantJson.priority;
|
||||||
@@ -172,7 +178,8 @@ export function shipFromJson(json) {
|
|||||||
// Thrusters
|
// Thrusters
|
||||||
const thrustersJson = json.modules.MainEngines.module;
|
const thrustersJson = json.modules.MainEngines.module;
|
||||||
const thrusters = _moduleFromEdId(thrustersJson.id);
|
const thrusters = _moduleFromEdId(thrustersJson.id);
|
||||||
if (thrustersJson.modifiers) _addModifications(thrusters, thrustersJson.modifiers, thrustersJson.recipeName, thrustersJson.recipeLevel);
|
rootModule = json.modules.MainEngines;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(thrusters, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[1], thrusters, true);
|
ship.use(ship.standard[1], thrusters, true);
|
||||||
ship.standard[1].enabled = thrustersJson.on === true;
|
ship.standard[1].enabled = thrustersJson.on === true;
|
||||||
ship.standard[1].priority = thrustersJson.priority;
|
ship.standard[1].priority = thrustersJson.priority;
|
||||||
@@ -180,7 +187,8 @@ export function shipFromJson(json) {
|
|||||||
// FSD
|
// FSD
|
||||||
const frameshiftdriveJson = json.modules.FrameShiftDrive.module;
|
const frameshiftdriveJson = json.modules.FrameShiftDrive.module;
|
||||||
const frameshiftdrive = _moduleFromEdId(frameshiftdriveJson.id);
|
const frameshiftdrive = _moduleFromEdId(frameshiftdriveJson.id);
|
||||||
if (frameshiftdriveJson.modifiers) _addModifications(frameshiftdrive, frameshiftdriveJson.modifiers, frameshiftdriveJson.recipeName, frameshiftdriveJson.recipeLevel);
|
rootModule = json.modules.FrameShiftDrive;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(frameshiftdrive, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[2], frameshiftdrive, true);
|
ship.use(ship.standard[2], frameshiftdrive, true);
|
||||||
ship.standard[2].enabled = frameshiftdriveJson.on === true;
|
ship.standard[2].enabled = frameshiftdriveJson.on === true;
|
||||||
ship.standard[2].priority = frameshiftdriveJson.priority;
|
ship.standard[2].priority = frameshiftdriveJson.priority;
|
||||||
@@ -188,7 +196,8 @@ export function shipFromJson(json) {
|
|||||||
// Life support
|
// Life support
|
||||||
const lifesupportJson = json.modules.LifeSupport.module;
|
const lifesupportJson = json.modules.LifeSupport.module;
|
||||||
const lifesupport = _moduleFromEdId(lifesupportJson.id);
|
const lifesupport = _moduleFromEdId(lifesupportJson.id);
|
||||||
if (lifesupportJson.modifiers)_addModifications(lifesupport, lifesupportJson.modifiers, lifesupportJson.recipeName, lifesupportJson.recipeLevel);
|
rootModule = json.modules.LifeSupport;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(lifesupport, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[3], lifesupport, true);
|
ship.use(ship.standard[3], lifesupport, true);
|
||||||
ship.standard[3].enabled = lifesupportJson.on === true;
|
ship.standard[3].enabled = lifesupportJson.on === true;
|
||||||
ship.standard[3].priority = lifesupportJson.priority;
|
ship.standard[3].priority = lifesupportJson.priority;
|
||||||
@@ -196,7 +205,8 @@ export function shipFromJson(json) {
|
|||||||
// Power distributor
|
// Power distributor
|
||||||
const powerdistributorJson = json.modules.PowerDistributor.module;
|
const powerdistributorJson = json.modules.PowerDistributor.module;
|
||||||
const powerdistributor = _moduleFromEdId(powerdistributorJson.id);
|
const powerdistributor = _moduleFromEdId(powerdistributorJson.id);
|
||||||
if (powerdistributorJson.modifiers) _addModifications(powerdistributor, powerdistributorJson.modifiers, powerdistributorJson.recipeName, powerdistributorJson.recipeLevel);
|
rootModule = json.modules.PowerDistributor;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(powerdistributor, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[4], powerdistributor, true);
|
ship.use(ship.standard[4], powerdistributor, true);
|
||||||
ship.standard[4].enabled = powerdistributorJson.on === true;
|
ship.standard[4].enabled = powerdistributorJson.on === true;
|
||||||
ship.standard[4].priority = powerdistributorJson.priority;
|
ship.standard[4].priority = powerdistributorJson.priority;
|
||||||
@@ -204,7 +214,8 @@ export function shipFromJson(json) {
|
|||||||
// Sensors
|
// Sensors
|
||||||
const sensorsJson = json.modules.Radar.module;
|
const sensorsJson = json.modules.Radar.module;
|
||||||
const sensors = _moduleFromEdId(sensorsJson.id);
|
const sensors = _moduleFromEdId(sensorsJson.id);
|
||||||
if (sensorsJson.modifiers) _addModifications(sensors, sensorsJson.modifiers, sensorsJson.recipeName, sensorsJson.recipeLevel);
|
rootModule = json.modules.Radar;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(sensors, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.standard[5], sensors, true);
|
ship.use(ship.standard[5], sensors, true);
|
||||||
ship.standard[5].enabled = sensorsJson.on === true;
|
ship.standard[5].enabled = sensorsJson.on === true;
|
||||||
ship.standard[5].priority = sensorsJson.priority;
|
ship.standard[5].priority = sensorsJson.priority;
|
||||||
@@ -240,7 +251,8 @@ export function shipFromJson(json) {
|
|||||||
} else {
|
} else {
|
||||||
const hardpointJson = hardpointSlot.module;
|
const hardpointJson = hardpointSlot.module;
|
||||||
const hardpoint = _moduleFromEdId(hardpointJson.id);
|
const hardpoint = _moduleFromEdId(hardpointJson.id);
|
||||||
if (hardpointJson.modifiers) _addModifications(hardpoint, hardpointJson.modifiers, hardpointJson.recipeName, hardpointJson.recipeLevel);
|
rootModule = hardpointSlot;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(hardpoint, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel, rootModule.specialModifications);
|
||||||
ship.use(ship.hardpoints[hardpointArrayNum], hardpoint, true);
|
ship.use(ship.hardpoints[hardpointArrayNum], hardpoint, true);
|
||||||
ship.hardpoints[hardpointArrayNum].enabled = hardpointJson.on === true;
|
ship.hardpoints[hardpointArrayNum].enabled = hardpointJson.on === true;
|
||||||
ship.hardpoints[hardpointArrayNum].priority = hardpointJson.priority;
|
ship.hardpoints[hardpointArrayNum].priority = hardpointJson.priority;
|
||||||
@@ -282,7 +294,8 @@ export function shipFromJson(json) {
|
|||||||
} else {
|
} else {
|
||||||
const internalJson = internalSlot.module;
|
const internalJson = internalSlot.module;
|
||||||
const internal = _moduleFromEdId(internalJson.id);
|
const internal = _moduleFromEdId(internalJson.id);
|
||||||
if (internalJson.modifiers) _addModifications(internal, internalJson.modifiers, internalJson.recipeName, internalJson.recipeLevel);
|
rootModule = internalSlot;
|
||||||
|
if (rootModule.WorkInProgress_modifications) _addModifications(internal, rootModule.WorkInProgress_modifications, rootModule.engineer.recipeName, rootModule.engineer.recipeLevel);
|
||||||
ship.use(ship.internal[i], internal, true);
|
ship.use(ship.internal[i], internal, true);
|
||||||
ship.internal[i].enabled = internalJson.on === true;
|
ship.internal[i].enabled = internalJson.on === true;
|
||||||
ship.internal[i].priority = internalJson.priority;
|
ship.internal[i].priority = internalJson.priority;
|
||||||
@@ -299,35 +312,48 @@ export function shipFromJson(json) {
|
|||||||
* @param {Object} modifiers the modifiers
|
* @param {Object} modifiers the modifiers
|
||||||
* @param {Object} blueprint the blueprint of the modification
|
* @param {Object} blueprint the blueprint of the modification
|
||||||
* @param {Object} grade the grade of the modification
|
* @param {Object} grade the grade of the modification
|
||||||
|
* @param {Object} specialModifications special modification
|
||||||
*/
|
*/
|
||||||
function _addModifications(module, modifiers, blueprint, grade) {
|
function _addModifications(module, modifiers, blueprint, grade, specialModifications) {
|
||||||
if (!modifiers || !modifiers.modifiers) return;
|
if (!modifiers) return;
|
||||||
|
|
||||||
let special;
|
let special;
|
||||||
for (const i in modifiers.modifiers) {
|
if (specialModifications) {
|
||||||
|
special = Modifications.specials[Object.keys(specialModifications)[0]]
|
||||||
|
}
|
||||||
|
for (const i in modifiers) {
|
||||||
// Some special modifications
|
// Some special modifications
|
||||||
if (modifiers.modifiers[i].name === 'mod_weapon_clip_size_override') {
|
if (modifiers[i].name === 'mod_weapon_clip_size_override') {
|
||||||
// This is a numeric addition to the clip size, but we need to work it out in terms of being a percentage so
|
// This is a numeric addition to the clip size, but we need to work it out in terms of being a percentage so
|
||||||
// that it works the same as other modifications
|
// that it works the same as other modifications
|
||||||
const origClip = module.clip || 1;
|
const origClip = module.clip || 1;
|
||||||
module.setModValue('clip', ((modifiers.modifiers[i].value - origClip) / origClip) * 10000);
|
module.setModValue('clip', ((modifiers[i].value - origClip) / origClip) * 10000);
|
||||||
} else if (modifiers.modifiers[i].name === 'mod_weapon_burst_size') {
|
} else if (modifiers[i].name === 'mod_weapon_burst_size') {
|
||||||
// This is an absolute number that acts as an override
|
// This is an absolute number that acts as an override
|
||||||
module.setModValue('burst', modifiers.modifiers[i].value * 100);
|
module.setModValue('burst', modifiers[i].value * 100);
|
||||||
} else if (modifiers.modifiers[i].name === 'mod_weapon_burst_rof') {
|
} else if (modifiers[i].name === 'mod_weapon_burst_rof') {
|
||||||
// This is an absolute number that acts as an override
|
// This is an absolute number that acts as an override
|
||||||
module.setModValue('burstrof', modifiers.modifiers[i].value * 100);
|
module.setModValue('burstrof', modifiers[i].value * 100);
|
||||||
} else if (modifiers.modifiers[i].name === 'mod_weapon_falloffrange_from_range') {
|
} else if (modifiers[i].name === 'mod_weapon_falloffrange_from_range') {
|
||||||
// Obtain the falloff value directly from the range
|
// Obtain the falloff value directly from the range
|
||||||
module.setModValue('fallofffromrange', 1);
|
module.setModValue('fallofffromrange', 1);
|
||||||
} else if (modifiers.modifiers[i].name && modifiers.modifiers[i].name.startsWith('special_')) {
|
} else if (modifiers[i].name && modifiers[i].name.startsWith('special_')) {
|
||||||
// We don't add special effects directly, but keep a note of them so they can be added when fetching values
|
// We don't add special effects directly, but keep a note of them so they can be added when fetching values
|
||||||
special = Modifications.specials[modifiers.modifiers[i].name];
|
special = Modifications.specials[modifiers[i].name];
|
||||||
} else {
|
} else {
|
||||||
// Look up the modifiers to find what we need to do
|
// Look up the modifiers to find what we need to do
|
||||||
const modifierActions = Modifications.modifierActions[modifiers.modifiers[i].name];
|
const modifierActions = Modifications.modifierActions[i];
|
||||||
const value = modifiers.modifiers[i].value;
|
let value;
|
||||||
|
if (i === 'OutfittingFieldType_DefenceModifierShieldMultiplier') {
|
||||||
|
value = modifiers[i].value - 1;
|
||||||
|
} else if (i === 'OutfittingFieldType_DefenceModifierHealthMultiplier' && blueprint.startsWith('Armour_')) {
|
||||||
|
value = (modifiers[i].value - module.hullboost) / module.hullboost;
|
||||||
|
} else if (i === 'OutfittingFieldType_DefenceModifierHealthMultiplier') {
|
||||||
|
value = modifiers[i].value / module.hullboost;
|
||||||
|
} else if (i === 'OutfittingFieldType_RateOfFire') {
|
||||||
|
value = (1 / Math.abs(modifiers[i].value));
|
||||||
|
} else {
|
||||||
|
value = modifiers[i].value - 1;
|
||||||
|
}
|
||||||
// Carry out the required changes
|
// Carry out the required changes
|
||||||
for (const action in modifierActions) {
|
for (const action in modifierActions) {
|
||||||
if (isNaN(modifierActions[action])) {
|
if (isNaN(modifierActions[action])) {
|
||||||
@@ -423,7 +449,7 @@ function _addModifications(module, modifiers, blueprint, grade) {
|
|||||||
// Bulkhead boost is based off the inherent boost of the module
|
// Bulkhead boost is based off the inherent boost of the module
|
||||||
if (module.grp == 'bh') {
|
if (module.grp == 'bh') {
|
||||||
const alteredBoost = (1 + module.hullboost) * (1 + module.getModValue('hullboost') / 10000) - 1;
|
const alteredBoost = (1 + module.hullboost) * (1 + module.getModValue('hullboost') / 10000) - 1;
|
||||||
module.setModValue('hullboost', (alteredBoost / module.hullboost - 1) * 10000);
|
module.setModValue('hullboost', (alteredBoost / module.hullboost - 1) * 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Jitter is an absolute number, so we need to divide it by 100
|
// Jitter is an absolute number, so we need to divide it by 100
|
||||||
@@ -431,11 +457,6 @@ function _addModifications(module, modifiers, blueprint, grade) {
|
|||||||
module.setModValue('jitter', module.getModValue('jitter') / 100);
|
module.setModValue('jitter', module.getModValue('jitter') / 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
// FD uses interval between bursts internally, so we need to translate this to a real rate of fire
|
|
||||||
if (module.getModValue('rof')) {
|
|
||||||
module.setModValue('rof', ((1 / (1 + module.getModValue('rof') / 10000)) - 1) * 10000);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clip size is rounded up so that the result is a whole number
|
// Clip size is rounded up so that the result is a whole number
|
||||||
if (module.getModValue('clip')) {
|
if (module.getModValue('clip')) {
|
||||||
const individual = 1 / (module.clip || 1);
|
const individual = 1 / (module.clip || 1);
|
||||||
|
|||||||
292
src/app/utils/JournalUtils.js
Normal file
292
src/app/utils/JournalUtils.js
Normal file
@@ -0,0 +1,292 @@
|
|||||||
|
import Ship from '../shipyard/Ship'
|
||||||
|
import { HARDPOINT_NUM_TO_CLASS, shipModelFromJson } from './CompanionApiUtils'
|
||||||
|
import { Ships } from 'coriolis-data/dist'
|
||||||
|
import Module from '../shipyard/Module'
|
||||||
|
import { Modules } from 'coriolis-data/dist'
|
||||||
|
import { Modifications } from 'coriolis-data/dist'
|
||||||
|
import { getBlueprint } from './BlueprintFunctions'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Obtain a module given its FD Name
|
||||||
|
* @param {string} fdname the FD Name of the module
|
||||||
|
* @return {Module} the module
|
||||||
|
*/
|
||||||
|
function _moduleFromFdName (fdname) {
|
||||||
|
if (!fdname) return null
|
||||||
|
fdname = fdname.toLowerCase()
|
||||||
|
// Check standard modules
|
||||||
|
for (const grp in Modules.standard) {
|
||||||
|
if (Modules.standard.hasOwnProperty(grp)) {
|
||||||
|
for (const i in Modules.standard[grp]) {
|
||||||
|
if (Modules.standard[grp][i].symbol && Modules.standard[grp][i].symbol.toLowerCase() === fdname) {
|
||||||
|
// Found it
|
||||||
|
return new Module({template: Modules.standard[grp][i]})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check hardpoint modules
|
||||||
|
for (const grp in Modules.hardpoints) {
|
||||||
|
if (Modules.hardpoints.hasOwnProperty(grp)) {
|
||||||
|
for (const i in Modules.hardpoints[grp]) {
|
||||||
|
if (Modules.hardpoints[grp][i].symbol && Modules.hardpoints[grp][i].symbol.toLowerCase() === fdname) {
|
||||||
|
// Found it
|
||||||
|
return new Module({template: Modules.hardpoints[grp][i]})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check internal modules
|
||||||
|
for (const grp in Modules.internal) {
|
||||||
|
if (Modules.internal.hasOwnProperty(grp)) {
|
||||||
|
for (const i in Modules.internal[grp]) {
|
||||||
|
if (Modules.internal[grp][i].symbol && Modules.internal[grp][i].symbol.toLowerCase() === fdname) {
|
||||||
|
// Found it
|
||||||
|
return new Module({template: Modules.internal[grp][i]})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not found
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a ship from the journal Loadout event JSON
|
||||||
|
* @param {object} json the Loadout event JSON
|
||||||
|
* @return {Ship} the built ship
|
||||||
|
*/
|
||||||
|
export function shipFromLoadoutJSON (json) {
|
||||||
|
// Start off building a basic ship
|
||||||
|
const shipModel = shipModelFromJson(json)
|
||||||
|
if (!shipModel) {
|
||||||
|
throw 'No such ship found: "' + json.Ship + '"'
|
||||||
|
}
|
||||||
|
const shipTemplate = Ships[shipModel]
|
||||||
|
|
||||||
|
let ship = new Ship(shipModel, shipTemplate.properties, shipTemplate.slots)
|
||||||
|
ship.buildWith(null)
|
||||||
|
// Initial Ship building, don't do engineering yet.
|
||||||
|
let opts = [];
|
||||||
|
|
||||||
|
for (const module of json.Modules) {
|
||||||
|
switch (module.Slot.toLowerCase()) {
|
||||||
|
// Cargo Hatch.
|
||||||
|
case 'cargohatch':
|
||||||
|
ship.cargoHatch.enabled = module.On
|
||||||
|
ship.cargoHatch.priority = module.Priority
|
||||||
|
break
|
||||||
|
// Add the bulkheads
|
||||||
|
case 'armour':
|
||||||
|
if (module.Item.toLowerCase().endsWith('_armour_grade1')) {
|
||||||
|
ship.useBulkhead(0, true)
|
||||||
|
} else if (module.Item.toLowerCase().endsWith('_armour_grade2')) {
|
||||||
|
ship.useBulkhead(1, true)
|
||||||
|
} else if (module.Item.toLowerCase().endsWith('_armour_grade3')) {
|
||||||
|
ship.useBulkhead(2, true)
|
||||||
|
} else if (module.Item.toLowerCase().endsWith('_armour_mirrored')) {
|
||||||
|
ship.useBulkhead(3, true)
|
||||||
|
} else if (module.Item.toLowerCase().endsWith('_armour_reactive')) {
|
||||||
|
ship.useBulkhead(4, true)
|
||||||
|
} else {
|
||||||
|
throw 'Unknown bulkheads "' + module.Item + '"'
|
||||||
|
}
|
||||||
|
ship.bulkheads.enabled = true
|
||||||
|
if (module.Engineering) _addModifications(ship.bulkheads.m, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'powerplant':
|
||||||
|
const powerplant = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[0], powerplant, true)
|
||||||
|
ship.standard[0].enabled = module.On
|
||||||
|
ship.standard[0].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(powerplant, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'mainengines':
|
||||||
|
const thrusters = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[1], thrusters, true)
|
||||||
|
ship.standard[1].enabled = module.On
|
||||||
|
ship.standard[1].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(thrusters, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'frameshiftdrive':
|
||||||
|
const frameshiftdrive = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[2], frameshiftdrive, true)
|
||||||
|
ship.standard[2].enabled = module.On
|
||||||
|
ship.standard[2].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(frameshiftdrive, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'lifesupport':
|
||||||
|
const lifesupport = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[3], lifesupport, true)
|
||||||
|
ship.standard[3].enabled = module.On === true
|
||||||
|
ship.standard[3].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(lifesupport, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'powerdistributor':
|
||||||
|
const powerdistributor = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[4], powerdistributor, true)
|
||||||
|
ship.standard[4].enabled = module.On
|
||||||
|
ship.standard[4].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(powerdistributor, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'radar':
|
||||||
|
const sensors = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[5], sensors, true)
|
||||||
|
ship.standard[5].enabled = module.On
|
||||||
|
ship.standard[5].priority = module.Priority
|
||||||
|
if (module.Engineering) _addModifications(sensors, module.Engineering.Modifiers, module.Engineering.BlueprintName, module.Engineering.Level, module.Engineering.ExperimentalEffect)
|
||||||
|
break
|
||||||
|
case 'fueltank':
|
||||||
|
const fueltank = _moduleFromFdName(module.Item)
|
||||||
|
ship.use(ship.standard[6], fueltank, true)
|
||||||
|
ship.standard[6].enabled = true
|
||||||
|
ship.standard[6].priority = 0
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
for (const module of json.Modules) {
|
||||||
|
if (module.Slot.toLowerCase().search(/hardpoint/) !== -1) {
|
||||||
|
// Add hardpoints
|
||||||
|
let hardpoint;
|
||||||
|
let hardpointClassNum = -1
|
||||||
|
let hardpointSlotNum = -1
|
||||||
|
let hardpointArrayNum = 0
|
||||||
|
for (let i in shipTemplate.slots.hardpoints) {
|
||||||
|
if (shipTemplate.slots.hardpoints[i] === hardpointClassNum) {
|
||||||
|
// Another slot of the same class
|
||||||
|
hardpointSlotNum++
|
||||||
|
} else {
|
||||||
|
// The first slot of a new class
|
||||||
|
hardpointClassNum = shipTemplate.slots.hardpoints[i]
|
||||||
|
hardpointSlotNum = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now that we know what we're looking for, find it
|
||||||
|
const hardpointName = HARDPOINT_NUM_TO_CLASS[hardpointClassNum] + 'Hardpoint' + hardpointSlotNum
|
||||||
|
const hardpointSlot = json.Modules.find(elem => elem.Slot.toLowerCase() === hardpointName.toLowerCase())
|
||||||
|
if (!hardpointSlot) {
|
||||||
|
// This can happen with old imports that don't contain new hardpoints
|
||||||
|
} else if (!hardpointSlot) {
|
||||||
|
// No module
|
||||||
|
} else {
|
||||||
|
hardpoint = _moduleFromFdName(hardpointSlot.Item)
|
||||||
|
ship.use(ship.hardpoints[hardpointArrayNum], hardpoint, true)
|
||||||
|
ship.hardpoints[hardpointArrayNum].enabled = hardpointSlot.On
|
||||||
|
ship.hardpoints[hardpointArrayNum].priority = hardpointSlot.Priority
|
||||||
|
opts.push({coriolisMod: hardpoint, json: hardpointSlot});
|
||||||
|
}
|
||||||
|
hardpointArrayNum++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (module.Slot.toLowerCase().search(/slot\d/) !== -1) {
|
||||||
|
let internalSlotNum = 1
|
||||||
|
let militarySlotNum = 1
|
||||||
|
for (let i in shipTemplate.slots.internal) {
|
||||||
|
const isMilitary = isNaN(shipTemplate.slots.internal[i]) ? shipTemplate.slots.internal[i].name = 'military' : false
|
||||||
|
|
||||||
|
// The internal slot might be a standard or a military slot. Military slots have a different naming system
|
||||||
|
let internalSlot = null
|
||||||
|
if (isMilitary) {
|
||||||
|
const internalName = 'Military0' + militarySlotNum
|
||||||
|
internalSlot = json.Modules.find(elem => elem.Slot.toLowerCase() === internalName.toLowerCase())
|
||||||
|
militarySlotNum++
|
||||||
|
} else {
|
||||||
|
// Slot numbers are not contiguous so handle skips.
|
||||||
|
while (internalSlot === null && internalSlotNum < 99) {
|
||||||
|
// Slot sizes have no relationship to the actual size, either, so check all possibilities
|
||||||
|
for (let slotsize = 0; slotsize < 9; slotsize++) {
|
||||||
|
const internalName = 'Slot' + (internalSlotNum <= 9 ? '0' : '') + internalSlotNum + '_Size' + slotsize
|
||||||
|
if (json.Modules.find(elem => elem.Slot.toLowerCase() === internalName.toLowerCase())) {
|
||||||
|
internalSlot = json.Modules.find(elem => elem.Slot.toLowerCase() === internalName.toLowerCase());
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
internalSlotNum++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!internalSlot) {
|
||||||
|
// This can happen with old imports that don't contain new slots
|
||||||
|
} else if (!internalSlot) {
|
||||||
|
// No module
|
||||||
|
} else {
|
||||||
|
const internalJson = internalSlot
|
||||||
|
const internal = _moduleFromFdName(internalJson.Item)
|
||||||
|
ship.use(ship.internal[i], internal, true)
|
||||||
|
ship.internal[i].enabled = internalJson.On === true
|
||||||
|
ship.internal[i].priority = internalJson.Priority
|
||||||
|
opts.push({coriolisMod: internal, json: internalSlot});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const i of opts) {
|
||||||
|
if (i.json.Engineering) _addModifications(i.coriolisMod, i.json.Engineering.Modifiers, i.json.Engineering.BlueprintName, i.json.Engineering.Level, i.json.Engineering.ExperimentalEffect)
|
||||||
|
}
|
||||||
|
// We don't have any information on it so guess it's priority 5 and disabled
|
||||||
|
if (!ship.cargoHatch) {
|
||||||
|
ship.cargoHatch.enabled = false
|
||||||
|
ship.cargoHatch.priority = 4
|
||||||
|
}
|
||||||
|
console.log(ship)
|
||||||
|
|
||||||
|
// Now update the ship's codes before returning it
|
||||||
|
return ship.updatePowerPrioritesString().updatePowerEnabledString().updateModificationsString()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add the modifications for a module
|
||||||
|
* @param {Module} module the module
|
||||||
|
* @param {Object} modifiers the modifiers
|
||||||
|
* @param {Object} blueprint the blueprint of the modification
|
||||||
|
* @param {Object} grade the grade of the modification
|
||||||
|
* @param {Object} specialModifications special modification
|
||||||
|
*/
|
||||||
|
function _addModifications (module, modifiers, blueprint, grade, specialModifications) {
|
||||||
|
if (!modifiers) return
|
||||||
|
let special
|
||||||
|
if (specialModifications) {
|
||||||
|
special = Modifications.specials[specialModifications]
|
||||||
|
}
|
||||||
|
for (const i in modifiers) {
|
||||||
|
// Some special modifications
|
||||||
|
// Look up the modifiers to find what we need to do
|
||||||
|
const findMod = val => Object.keys(Modifications.modifierActions).find(elem => elem.toString().toLowerCase().replace(/(outfittingfieldtype_|persecond)/igm, '') === val.toString().toLowerCase().replace(/(outfittingfieldtype_|persecond)/igm, ''))
|
||||||
|
const modifierActions = Modifications.modifierActions[findMod(modifiers[i].Label)]
|
||||||
|
//TODO: Figure out how to scale this value.
|
||||||
|
if (!!modifiers[i].LessIsGood) {
|
||||||
|
|
||||||
|
}
|
||||||
|
let value = (modifiers[i].Value / modifiers[i].OriginalValue * 100 - 100) * 100;
|
||||||
|
if (value === Infinity) {
|
||||||
|
value = modifiers[i].Value * 100;
|
||||||
|
}
|
||||||
|
if (modifiers[i].Label.search('Resistance') >= 0) {
|
||||||
|
value = (modifiers[i].Value * 100) - (modifiers[i].OriginalValue * 100)
|
||||||
|
}
|
||||||
|
// Carry out the required changes
|
||||||
|
for (const action in modifierActions) {
|
||||||
|
if (isNaN(modifierActions[action])) {
|
||||||
|
module.setModValue(action, modifierActions[action])
|
||||||
|
} else {
|
||||||
|
module.setModValue(action, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the blueprint definition, grade and special
|
||||||
|
if (blueprint) {
|
||||||
|
module.blueprint = getBlueprint(blueprint, module)
|
||||||
|
if (grade) {
|
||||||
|
module.blueprint.grade = Number(grade)
|
||||||
|
}
|
||||||
|
if (special) {
|
||||||
|
module.blueprint.special = special
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -220,7 +220,7 @@ export function diffDetails(language, m, mm) {
|
|||||||
|
|
||||||
let mCost = m.cost || 0;
|
let mCost = m.cost || 0;
|
||||||
let mmCost = mm ? mm.cost : 0;
|
let mmCost = mm ? mm.cost : 0;
|
||||||
if (mCost != mmCost) propDiffs.push(<div key='cost'>{translate('cost')}: <span className={diffClass(mCost, mmCost, true) }>{mCost ? Math.round(mCost * (1 - Persist.getModuleDiscount())) : 0}{units.CR}</span></div>);
|
if (mCost != mmCost) propDiffs.push(<div key='cost'>{translate('cost')}: <span className={diffClass(mCost, mmCost, true) }>{formats.int(mCost ? Math.round(mCost * (1 - Persist.getModuleDiscount())) : 0)}{units.CR}</span></div>);
|
||||||
|
|
||||||
let mMass = m.mass || 0;
|
let mMass = m.mass || 0;
|
||||||
let mmMass = mm ? mm.getMass() : 0;
|
let mmMass = mm ? mm.getMass() : 0;
|
||||||
|
|||||||
6
src/images/Empire.svg
Normal file
6
src/images/Empire.svg
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<svg width="1718.2px" height="999.8px" viewBox="0 0 1718.2 999.8" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<defs/>
|
||||||
|
<g>
|
||||||
|
<path stroke="none" fill="#FF8C0D" d="M1717.55 -0.05 L1717.65 -0.25 1718.2 -0.1 1718.6 0.35 1719.2 1.55 1719.3 2.25 1718.95 2.85 1709.75 11.35 1010.8 687.7 1008.05 690.5 1005.45 693.4 1003 696.35 1000.75 699.45 998.65 702.65 996.7 705.95 994.9 709.4 993.25 713.05 934.9 850.7 876.4 988.25 871.55 998.95 871.3 999.3 870.9 999.5 870.6 999.55 870 999.4 869.5 999.05 868.3 997.95 867.35 996.7 866.6 995.25 864.7 989.85 762.3 730.55 754.8 711.85 751.1 702.45 746.6 690.65 745.5 688.4 744.25 686.3 742.9 684.35 741.4 682.45 739.75 680.6 737.95 678.85 736 677.15 8.1 61.2 5.75 59.15 3.65 57.1 1.6 54.6 1.1 53.7 1.1 53.25 1.25 52.9 1.5 52.65 1.8 52.45 2.15 52.4 2.55 52.5 9.75 55.7 713.05 398.5 780.95 431.7 848.75 464.95 851.7 466.25 854.4 467.15 856.95 467.6 859.45 467.65 861.9 467.25 864.4 466.45 866.95 465.2 869.6 463.5 1416.95 79.15 1419.7 77.35 1422.45 75.7 1425.25 74.2 1428.1 72.85 1431 71.6 1434 70.5 1437.05 69.55 1440.2 68.7 1574.85 35.15 1709.5 1.5 1713.75 0.6 1717.55 -0.05 M903.9 129.8 L904 129.65 930.9 129.75 934.3 129.85 937.55 130.15 940.65 130.65 943.6 131.35 946.4 132.2 949.2 133.3 951.95 134.6 954.55 136.05 957 137.7 959.35 139.5 961.6 141.5 963.75 143.65 965.8 145.95 967.75 148.4 969.6 151 971.35 153.8 974.25 159 976.8 164.3 979.05 169.7 981 175.2 982.7 180.8 984.2 186.45 985.55 192.15 986.75 197.9 987.85 203.75 988.8 209.65 990.4 221.45 991.6 233.25 992.55 245.1 995 280.65 996 292.5 996.2 295.1 996.25 297.55 996.15 299.85 995.9 302 995.5 304.1 994.95 306.1 994.25 308 993.45 309.7 992.4 311.5 991.25 313.15 989.95 314.75 988.45 316.3 986.8 317.8 985 319.25 980.85 322.15 946.35 344.45 912.05 367.15 843.7 413.05 841.45 414.45 839.25 415.55 836.9 416.35 834.85 416.75 832.6 416.85 830.15 416.5 827.75 415.8 825.25 414.7 812.45 408.55 799.4 402.55 773.35 390.75 773 390.5 772.8 390.15 772.75 389.75 772.85 389.35 794.5 343.45 801.35 328.15 811 305.8 820.45 283.4 839 238.45 841 233 842.7 227.35 844.1 221.65 845.25 215.95 845.7 213.05 845.8 210.65 845.55 208.55 845.3 207.75 844.9 206.9 843.85 205.55 842.3 204.4 840.2 203.4 837.55 202.55 830.2 200.55 822.85 198.85 819.15 198.15 815.5 197.6 811.85 197.2 808.2 196.95 804.55 196.9 800.9 197.05 797.25 197.4 793.6 198 789.9 198.9 786.25 200.05 782.6 201.55 778.55 203.5 777 203.65 776.65 203.6 776.1 203.15 775.95 202.8 775.5 200.55 775.25 197.65 775.3 194.75 775.65 191.9 776.25 189.05 777.05 186.3 778 183.6 779.05 180.9 780.5 177.65 782.1 174.5 783.8 171.5 785.65 168.6 787.6 165.85 789.7 163.2 791.9 160.7 794.25 158.3 796.7 156 799.3 153.85 802 151.8 804.8 149.9 807.7 148.1 810.75 146.4 813.9 144.85 817.15 143.4 824.4 140.6 831.75 138.2 839.15 136.15 846.65 134.45 854.2 133.05 861.8 131.95 869.45 131.1 877.1 130.5 883.8 130.15 890.5 129.95 903.9 129.8"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.8 KiB |
6
src/images/Federation.svg
Normal file
6
src/images/Federation.svg
Normal file
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 30 KiB |
71
src/images/surface_port_pm.svg
Normal file
71
src/images/surface_port_pm.svg
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||||
|
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
|
||||||
|
width="631.072px" height="713.591px" viewBox="0 0 631.072 713.591" enable-background="new 0 0 631.072 713.591"
|
||||||
|
xml:space="preserve">
|
||||||
|
<g id="Layer_2">
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="460.344" y="668.509" transform="matrix(0.7399 0.6727 -0.6727 0.7399 586.8318 -155.8918)" fill="#FF8C0D" width="69.376" height="24.999"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="499.816" y="630.511" transform="matrix(0.8742 0.4856 -0.4856 0.8742 379.9255 -180.1477)" fill="#FF8C0D" width="75.501" height="24.999"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="537.078" y="590.363" transform="matrix(0.9477 0.319 -0.319 0.9477 222.2531 -151.1744)" fill="#FF8C0D" width="71.047" height="25"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="561.364" y="544.916" transform="matrix(0.9741 0.2263 -0.2263 0.9741 141.5892 -120.2454)" fill="#FF8C0D" width="67.758" height="25"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="113.454" y="633.977" transform="matrix(0.6727 0.7399 -0.7399 0.6727 536.0002 125.674)" fill="#FF8C0D" width="24.999" height="69.375"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="72.787" y="594.581" transform="matrix(0.4857 0.8741 -0.8741 0.4857 596.6155 250.6765)" fill="#FF8C0D" width="25" height="75.501"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="43.454" y="554.475" transform="matrix(0.319 0.9477 -0.9477 0.319 597.2715 348.7429)" fill="#FF8C0D" width="25" height="71.046"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
|
||||||
|
<rect x="23.329" y="507.12" transform="matrix(0.2263 0.9741 -0.9741 0.2263 554.6862 383.6775)" fill="#FF8C0D" width="25" height="67.758"/>
|
||||||
|
</g>
|
||||||
|
<path fill="#FF8C0D" d="M556.439,425.748c-13.466-24.188-32.583-45.79-56.821-64.205c-30.562-23.22-67.327-39.855-107.331-48.986
|
||||||
|
v25.737c93.31,23.32,160.681,90.053,160.681,168.752c0,98.096-104.668,177.619-233.785,177.619
|
||||||
|
c-129.115,0-233.784-79.523-233.784-177.619c0-75.918,62.696-140.699,150.889-166.118v-25.964
|
||||||
|
c-36.209,9.508-69.489,25.266-97.539,46.578c-24.238,18.415-43.355,40.017-56.821,64.205
|
||||||
|
c-14.286,25.66-21.529,53.014-21.529,81.299c0,28.284,7.243,55.637,21.529,81.298c13.465,24.188,32.583,45.789,56.821,64.204
|
||||||
|
c48.479,36.832,112.559,57.117,180.435,57.117c67.876,0,131.956-20.285,180.436-57.117c24.238-18.415,43.355-40.017,56.821-64.204
|
||||||
|
c14.285-25.661,21.528-53.014,21.528-81.298C577.968,478.761,570.725,451.408,556.439,425.748z"/>
|
||||||
|
<path fill="#FF8C0D" d="M473.97,427.082c-8.958-16.091-21.621-30.42-37.64-42.591c-13.129-9.975-27.996-18.079-44.043-24.152
|
||||||
|
v27.053c42.591,19.056,71.178,54.248,71.178,94.519c0,60.541-64.597,109.619-144.282,109.619
|
||||||
|
c-79.684,0-144.281-49.078-144.281-109.619c0-37.101,24.263-69.892,61.386-89.727v-27.819c-12.323,5.48-23.835,12.212-34.25,20.125
|
||||||
|
c-16.019,12.171-28.683,26.5-37.64,42.591c-9.618,17.276-14.495,35.724-14.495,54.829c0,19.104,4.877,37.552,14.495,54.829
|
||||||
|
c8.958,16.091,21.622,30.42,37.64,42.59c31.574,23.989,73.178,37.2,117.146,37.2c43.969,0,85.572-13.211,117.147-37.2
|
||||||
|
c16.019-12.17,28.682-26.499,37.64-42.59c9.618-17.277,14.495-35.725,14.495-54.829
|
||||||
|
C488.465,462.805,483.588,444.358,473.97,427.082z"/>
|
||||||
|
</g>
|
||||||
|
<g id="Layer_3">
|
||||||
|
<path fill="#FF8C0D" d="M253.551,232.333c-0.157-0.981-0.264-1.978-0.264-3c0-10.844,9.173-19.667,20.448-19.667
|
||||||
|
s20.448,8.822,20.448,19.667c0,1.022-0.106,2.019-0.264,3h25.15c0.067-0.993,0.113-1.991,0.113-3
|
||||||
|
c0-24.668-20.348-44.667-45.448-44.667s-45.448,19.998-45.448,44.667c0,1.009,0.046,2.007,0.113,3H253.551z"/>
|
||||||
|
<path fill="#FF8C0D" d="M302.831,48c-0.136-0.669-0.21-1.337-0.21-2c0-9.913,15.539-21,36.333-21s36.333,11.087,36.333,21
|
||||||
|
c0,0.663-0.074,1.331-0.21,2h25.143c0.038-0.664,0.067-1.329,0.067-2c0-25.405-27.46-46-61.333-46
|
||||||
|
c-33.874,0-61.333,20.595-61.333,46c0,0.671,0.03,1.336,0.067,2H302.831z"/>
|
||||||
|
<ellipse fill="#FF8C0D" cx="338.954" cy="68.499" rx="36.333" ry="6.75"/>
|
||||||
|
<path fill="#FF8C0D" d="M375.287,46c0,2.208-16.267,3.999-36.333,3.999S302.621,48.208,302.621,46h-25v158.499h25v-136h72.667
|
||||||
|
V473.51c-4.165,3.092-16.463,8.535-36.333,8.535c-7.763,0-14.353-0.836-19.771-2.033l-10.749,23.051
|
||||||
|
c8.99,2.529,19.405,3.982,30.52,3.982c33.873,0,61.333-13.432,61.333-30V46H375.287z"/>
|
||||||
|
<ellipse fill="#FF8C0D" cx="273.735" cy="253.132" rx="20.448" ry="3.916"/>
|
||||||
|
<path fill="#FF8C0D" d="M294.184,229.333c0,1.151-9.155,2.084-20.448,2.084s-20.448-0.933-20.448-2.084h-25v257.666h0.128
|
||||||
|
c1.667,14.529,21.313,26,45.32,26s43.653-11.471,45.32-26h0.128V229.333H294.184z M294.184,254.333v228.201
|
||||||
|
c-2.473,2.008-9.464,5.465-20.448,5.465s-17.975-3.456-20.448-5.465V253.132h40.896V254.333z"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 4.6 KiB |
@@ -55,10 +55,12 @@
|
|||||||
<!-- End Piwik Code -->
|
<!-- End Piwik Code -->
|
||||||
|
|
||||||
<!-- Bugsnag -->
|
<!-- Bugsnag -->
|
||||||
<script
|
<script src="//d2wy8f7a9ursnm.cloudfront.net/v4/bugsnag.min.js"></script>
|
||||||
src="//d2wy8f7a9ursnm.cloudfront.net/bugsnag-3.min.js"
|
<script src="//d2wy8f7a9ursnm.cloudfront.net/bugsnag-plugins/v1/bugsnag-react.min.js"></script>
|
||||||
data-apikey="2382691c622937f28f8fa82a1bfd797a"></script>
|
<script>
|
||||||
|
window.bugsnagClient = bugsnag('ba9fae819372850fb660755341fa6ef5', {appVersion: window.CORIOLIS_VERSION || undefined})
|
||||||
|
window.Bugsnag = window.bugsnagClient
|
||||||
|
</script>
|
||||||
</head>
|
</head>
|
||||||
<body style="background-color:#000;">
|
<body style="background-color:#000;">
|
||||||
<section id="coriolis"></section>
|
<section id="coriolis"></section>
|
||||||
|
|||||||
@@ -31,6 +31,50 @@ button {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.button-inline-menu {
|
||||||
|
white-space: nowrap;
|
||||||
|
line-height: 1.5em;
|
||||||
|
text-align: center;
|
||||||
|
margin: 0.5em 0;
|
||||||
|
padding-left: 5px;
|
||||||
|
border-top: 1px solid @primary-disabled;
|
||||||
|
border-bottom: 1px solid @primary-disabled;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
background: @primary-bg;
|
||||||
|
|
||||||
|
&.warning {
|
||||||
|
border-color: @warning-disabled;
|
||||||
|
color: @warning-disabled;
|
||||||
|
stroke: @warning-disabled;
|
||||||
|
|
||||||
|
.no-touch &:hover {
|
||||||
|
border-color: @warning;
|
||||||
|
color: @warning;
|
||||||
|
stroke: @warning;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.disabled, &.disabled:hover {
|
||||||
|
cursor: not-allowed;
|
||||||
|
border-color: @disabled;
|
||||||
|
color: @disabled;
|
||||||
|
stroke: @disabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
border-color: @secondary;
|
||||||
|
color: @secondary;
|
||||||
|
stroke: @secondary;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
border-color: @primary;
|
||||||
|
color: @primary;
|
||||||
|
stroke: @primary;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.button-lbl {
|
.button-lbl {
|
||||||
margin-left: 0.5em;
|
margin-left: 0.5em;
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,8 @@
|
|||||||
@secondary: #1FB0FF; // Light blue
|
@secondary: #1FB0FF; // Light blue
|
||||||
@warning: #FF3B00; // Dark Orange
|
@warning: #FF3B00; // Dark Orange
|
||||||
@disabled: #555; // Light grey
|
@disabled: #555; // Light grey
|
||||||
|
@success: #71a052; // Green
|
||||||
|
@purple: #800080; // Purple
|
||||||
@primary-disabled: darken(@primary, @disabledDarken);
|
@primary-disabled: darken(@primary, @disabledDarken);
|
||||||
@secondary-disabled: darken(@secondary, @disabledDarken);
|
@secondary-disabled: darken(@secondary, @disabledDarken);
|
||||||
@warning-disabled: darken(@warning, @disabledDarken);
|
@warning-disabled: darken(@warning, @disabledDarken);
|
||||||
@@ -18,6 +20,7 @@
|
|||||||
@bg: rgba(30,30,30,1);
|
@bg: rgba(30,30,30,1);
|
||||||
@bgBlack: #000;
|
@bgBlack: #000;
|
||||||
@primary-bg: fadeout(darken(@primary, 47%), 15%);
|
@primary-bg: fadeout(darken(@primary, 47%), 15%);
|
||||||
|
@alt-primary-bg: fadeout(darken(@primary, 42%), 15%); // Lighter brown background
|
||||||
@secondary-bg: fadeout(darken(@secondary, @bgDarken), @bgTransparency); // Brown background
|
@secondary-bg: fadeout(darken(@secondary, @bgDarken), @bgTransparency); // Brown background
|
||||||
@warning-bg: fadeout(darken(@warning, @bgDarken), @bgTransparency); // Dark Red
|
@warning-bg: fadeout(darken(@warning, @bgDarken), @bgTransparency); // Dark Red
|
||||||
|
|
||||||
|
|||||||
@@ -12,6 +12,20 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.view-changes {
|
||||||
|
position: fixed;
|
||||||
|
top: 3em;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 3em;
|
||||||
|
z-index: 3;
|
||||||
|
line-height: 3em;
|
||||||
|
text-align: center;
|
||||||
|
background-color: @bg;
|
||||||
|
color: @warning;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
header {
|
header {
|
||||||
background-color: @bg;
|
background-color: @bg;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ textarea {
|
|||||||
width:100%;
|
width:100%;
|
||||||
min-height: 10em;
|
min-height: 10em;
|
||||||
resize: vertical;
|
resize: vertical;
|
||||||
|
user-select: auto;
|
||||||
margin:2em 0;
|
margin:2em 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,13 +21,47 @@
|
|||||||
|
|
||||||
#overview {
|
#overview {
|
||||||
h1 {
|
h1 {
|
||||||
margin: 0;
|
margin: 0 1em 0.2em 0;
|
||||||
float: left;
|
float: left;
|
||||||
}
|
}
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
margin: 0.5em;
|
margin: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#requirements {
|
||||||
|
margin: 0;
|
||||||
|
float: left;
|
||||||
|
text-align: left;
|
||||||
|
|
||||||
|
div {
|
||||||
|
display: inline-block;
|
||||||
|
margin: 0 1em 0.5em 0;
|
||||||
|
height: 1.6em;
|
||||||
|
line-height: 1.6em;
|
||||||
|
text-align: left;
|
||||||
|
vertical-align: middle;
|
||||||
|
cursor: default;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: left center;
|
||||||
|
background-size: contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
.federation {
|
||||||
|
background-image: url('../images/Federation.svg');
|
||||||
|
padding-left: 1.985em + 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empire {
|
||||||
|
background-image: url('../images/Empire.svg');
|
||||||
|
padding-left: 2.750em + 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.horizons {
|
||||||
|
background-image: url('../images/surface_port_pm.svg');
|
||||||
|
padding-left: 1.415em + 0.5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#summary {
|
#summary {
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
-webkit-overflow-scrolling: touch;
|
-webkit-overflow-scrolling: touch;
|
||||||
@@ -36,10 +70,29 @@
|
|||||||
padding: 0.5em 0.2em;
|
padding: 0.5em 0.2em;
|
||||||
font-size: 0.9em;
|
font-size: 0.9em;
|
||||||
|
|
||||||
#summaryTable {
|
.summaryTable {
|
||||||
.user-select-none();
|
.user-select-none();
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
|
|
||||||
|
& > thead.blue {
|
||||||
|
background-color: @secondary;
|
||||||
|
border-left: 1px solid @primary-bg;
|
||||||
|
color: @primary-bg;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > thead.green {
|
||||||
|
background-color: @success;
|
||||||
|
border-left: 1px solid @primary-bg;
|
||||||
|
color: @primary-bg;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > thead.purple {
|
||||||
|
background-color: @purple;
|
||||||
|
border-left: 1px solid @primary-bg;
|
||||||
|
color: @primary-bg;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -41,6 +41,10 @@
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
input.cb:focus {
|
||||||
|
border-color:#fff;
|
||||||
|
}
|
||||||
|
|
||||||
.l {
|
.l {
|
||||||
text-transform: capitalize;
|
text-transform: capitalize;
|
||||||
margin-right: 0.8em;
|
margin-right: 0.8em;
|
||||||
|
|||||||
@@ -54,6 +54,10 @@ tbody tr {
|
|||||||
.no-touch &.highlight:hover, .no-touch &.highlighted {
|
.no-touch &.highlight:hover, .no-touch &.highlighted {
|
||||||
background-color: @warning-bg;
|
background-color: @warning-bg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.alt {
|
||||||
|
background-color: @alt-primary-bg;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
td {
|
td {
|
||||||
|
|||||||
@@ -1,28 +1,30 @@
|
|||||||
var path = require('path');
|
const path = require('path')
|
||||||
var exec = require('child_process').exec;
|
const exec = require('child_process').exec
|
||||||
var webpack = require('webpack');
|
const webpack = require('webpack')
|
||||||
var pkgJson = require('./package');
|
const HtmlWebpackPlugin = require('html-webpack-plugin')
|
||||||
var HtmlWebpackPlugin = require("html-webpack-plugin");
|
const ExtractTextPlugin = require('extract-text-webpack-plugin')
|
||||||
var ExtractTextPlugin = require("extract-text-webpack-plugin");
|
const WebpackNotifierPlugin = require('webpack-notifier')
|
||||||
|
const pkgJson = require('./package')
|
||||||
|
|
||||||
function CopyDirPlugin(source, destination) {
|
function CopyDirPlugin(source, destination) {
|
||||||
this.source = source;
|
this.source = source
|
||||||
this.destination = destination;
|
this.destination = destination
|
||||||
|
}
|
||||||
|
|
||||||
|
CopyDirPlugin.prototype.apply = function (compiler) {
|
||||||
|
compiler.plugin('done', () => {
|
||||||
|
console.log(compiler.outputPath, this.destination)
|
||||||
|
exec('cp -r ' + this.source + ' ' + path.join(compiler.outputPath, this.destination))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
CopyDirPlugin.prototype.apply = function(compiler) {
|
|
||||||
compiler.plugin('done', function() {
|
|
||||||
console.log(compiler.outputPath, this.destination);
|
|
||||||
exec('cp -r ' + this.source + ' ' + path.join(compiler.outputPath, this.destination));
|
|
||||||
}.bind(this));
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
devtool: 'eval',
|
devtool: 'source-map',
|
||||||
devServer: {
|
devServer: {
|
||||||
headers: { "Access-Control-Allow-Origin": "*" }
|
headers: {'Access-Control-Allow-Origin': '*'}
|
||||||
},
|
},
|
||||||
entry: {
|
entry: {
|
||||||
app: [ 'webpack-dev-server/client?http://0.0.0.0:3300', 'webpack/hot/only-dev-server', path.join(__dirname, "src/app/index.js") ],
|
app: ['webpack-dev-server/client?http://0.0.0.0:3300', 'webpack/hot/only-dev-server', path.join(__dirname, 'src/app/index.js')],
|
||||||
lib: ['d3', 'react', 'react-dom', 'classnames', 'fbemitter', 'lz-string']
|
lib: ['d3', 'react', 'react-dom', 'classnames', 'fbemitter', 'lz-string']
|
||||||
},
|
},
|
||||||
resolve: {
|
resolve: {
|
||||||
@@ -37,33 +39,34 @@ module.exports = {
|
|||||||
plugins: [
|
plugins: [
|
||||||
new CopyDirPlugin(path.join(__dirname, 'src/.htaccess'), ''),
|
new CopyDirPlugin(path.join(__dirname, 'src/.htaccess'), ''),
|
||||||
new webpack.optimize.CommonsChunkPlugin({
|
new webpack.optimize.CommonsChunkPlugin({
|
||||||
name: 'lib',
|
name: 'lib',
|
||||||
filename: 'lib.js'
|
filename: 'lib.js'
|
||||||
}),
|
}),
|
||||||
new HtmlWebpackPlugin({
|
new HtmlWebpackPlugin({
|
||||||
inject: false,
|
inject: false,
|
||||||
template: path.join(__dirname, "src/index.ejs"),
|
template: path.join(__dirname, 'src/index.ejs'),
|
||||||
version: pkgJson.version,
|
version: pkgJson.version,
|
||||||
gapiKey: process.env.CORIOLIS_GAPI_KEY || '',
|
gapiKey: process.env.CORIOLIS_GAPI_KEY || ''
|
||||||
}),
|
}),
|
||||||
new ExtractTextPlugin({
|
new ExtractTextPlugin({
|
||||||
filename: 'app.css',
|
filename: 'app.css',
|
||||||
disable: false,
|
disable: false,
|
||||||
allChunks: true
|
allChunks: true
|
||||||
}),
|
}),
|
||||||
|
new WebpackNotifierPlugin({alwaysNotify: true}),
|
||||||
new webpack.HotModuleReplacementPlugin(),
|
new webpack.HotModuleReplacementPlugin(),
|
||||||
new webpack.NoEmitOnErrorsPlugin()
|
new webpack.NoEmitOnErrorsPlugin()
|
||||||
],
|
],
|
||||||
module: {
|
module: {
|
||||||
rules: [
|
rules: [
|
||||||
{ test: /\.css$/, loader: ExtractTextPlugin.extract({ fallback: 'style-loader', use: 'css-loader'}) },
|
{test: /\.css$/, loader: ExtractTextPlugin.extract({fallback: 'style-loader', use: 'css-loader'})},
|
||||||
{ test: /\.less$/, loader: ExtractTextPlugin.extract({ fallback: 'style-loader', use: 'css-loader!less-loader'}) },
|
{test: /\.less$/, loader: ExtractTextPlugin.extract({fallback: 'style-loader', use: 'css-loader!less-loader'})},
|
||||||
{ test: /\.(js|jsx)$/, loaders: [ 'babel-loader' ], include: path.join(__dirname, 'src') },
|
{test: /\.(js|jsx)$/, loaders: ['babel-loader'], include: path.join(__dirname, 'src')},
|
||||||
{ test: /\.woff(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff' },
|
{test: /\.woff(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff'},
|
||||||
{ test: /\.woff2(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff' },
|
{test: /\.woff2(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff'},
|
||||||
{ test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/octet-stream' },
|
{test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/octet-stream'},
|
||||||
{ test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, loader: 'file-loader' },
|
{test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, loader: 'file-loader'},
|
||||||
{ test: /\.svg(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=image/svg+xml' }
|
{test: /\.svg(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=image/svg+xml'}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|||||||
@@ -1,24 +1,27 @@
|
|||||||
var path = require('path');
|
const path = require('path')
|
||||||
var exec = require('child_process').exec;
|
const exec = require('child_process').exec
|
||||||
var webpack = require('webpack');
|
const webpack = require('webpack')
|
||||||
var pkgJson = require('./package');
|
const HtmlWebpackPlugin = require('html-webpack-plugin')
|
||||||
var HtmlWebpackPlugin = require("html-webpack-plugin");
|
const ExtractTextPlugin = require('extract-text-webpack-plugin')
|
||||||
var ExtractTextPlugin = require("extract-text-webpack-plugin");
|
const AppCachePlugin = require('appcache-webpack-plugin')
|
||||||
var AppCachePlugin = require('appcache-webpack-plugin');
|
const {BugsnagSourceMapUploaderPlugin} = require('webpack-bugsnag-plugins')
|
||||||
|
const pkgJson = require('./package')
|
||||||
|
|
||||||
function CopyDirPlugin(source, destination) {
|
function CopyDirPlugin (source, destination) {
|
||||||
this.source = source;
|
this.source = source
|
||||||
this.destination = destination;
|
this.destination = destination
|
||||||
|
}
|
||||||
|
|
||||||
|
CopyDirPlugin.prototype.apply = function (compiler) {
|
||||||
|
compiler.plugin('done', () => {
|
||||||
|
console.log(compiler.outputPath, this.destination)
|
||||||
|
exec('cp -r ' + this.source + ' ' + path.join(compiler.outputPath, this.destination))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
CopyDirPlugin.prototype.apply = function(compiler) {
|
|
||||||
compiler.plugin('done', function() {
|
|
||||||
console.log(compiler.outputPath, this.destination);
|
|
||||||
exec('cp -r ' + this.source + ' ' + path.join(compiler.outputPath, this.destination));
|
|
||||||
}.bind(this));
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
cache: true,
|
cache: true,
|
||||||
|
devtool: 'source-map',
|
||||||
entry: {
|
entry: {
|
||||||
app: ['babel-polyfill', path.resolve(__dirname, 'src/app/index')],
|
app: ['babel-polyfill', path.resolve(__dirname, 'src/app/index')],
|
||||||
lib: ['d3', 'react', 'react-dom', 'classnames', 'fbemitter', 'lz-string']
|
lib: ['d3', 'react', 'react-dom', 'classnames', 'fbemitter', 'lz-string']
|
||||||
@@ -34,34 +37,39 @@ module.exports = {
|
|||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
new webpack.optimize.UglifyJsPlugin({
|
new webpack.optimize.UglifyJsPlugin({
|
||||||
'screw-ie8': true
|
'screw-ie8': true,
|
||||||
|
sourceMap: true
|
||||||
}),
|
}),
|
||||||
//new webpack.optimize.CommonsChunkPlugin({
|
//new webpack.optimize.CommonsChunkPlugin({
|
||||||
// name: 'lib',
|
// name: 'lib',
|
||||||
// filename: 'lib.[chunkhash:6].js'
|
// filename: 'lib.[chunkhash:6].js'
|
||||||
//}),
|
//}),
|
||||||
new HtmlWebpackPlugin({
|
new HtmlWebpackPlugin({
|
||||||
inject: false,
|
inject: false,
|
||||||
appCache: 'coriolis.appcache',
|
appCache: 'coriolis.appcache',
|
||||||
minify: {
|
minify: {
|
||||||
collapseBooleanAttributes: true,
|
collapseBooleanAttributes: true,
|
||||||
collapseWhitespace: true,
|
collapseWhitespace: true,
|
||||||
removeAttributeQuotes: true,
|
removeAttributeQuotes: true,
|
||||||
removeComments: true,
|
removeComments: true,
|
||||||
removeEmptyAttributes: true,
|
removeEmptyAttributes: true,
|
||||||
removeRedundantAttributes: true,
|
removeRedundantAttributes: true,
|
||||||
removeScriptTypeAttributes: true,
|
removeScriptTypeAttributes: true,
|
||||||
removeStyleLinkTypeAttributes: true
|
removeStyleLinkTypeAttributes: true
|
||||||
},
|
},
|
||||||
template: path.join(__dirname, "src/index.ejs"),
|
template: path.join(__dirname, 'src/index.ejs'),
|
||||||
uaTracking: process.env.CORIOLIS_UA_TRACKING || '',
|
uaTracking: process.env.CORIOLIS_UA_TRACKING || '',
|
||||||
gapiKey: process.env.CORIOLIS_GAPI_KEY || '',
|
gapiKey: process.env.CORIOLIS_GAPI_KEY || '',
|
||||||
version: pkgJson.version
|
version: pkgJson.version
|
||||||
}),
|
}),
|
||||||
new ExtractTextPlugin({
|
new ExtractTextPlugin({
|
||||||
filename: '[contenthash:6].css',
|
filename: '[contenthash:6].css',
|
||||||
disable: false,
|
disable: false,
|
||||||
allChunks: true
|
allChunks: true
|
||||||
|
}),
|
||||||
|
new BugsnagSourceMapUploaderPlugin({
|
||||||
|
apiKey: 'ba9fae819372850fb660755341fa6ef5',
|
||||||
|
appVersion: pkgJson.version
|
||||||
}),
|
}),
|
||||||
new CopyDirPlugin(path.join(__dirname, 'src/schemas'), 'schemas'),
|
new CopyDirPlugin(path.join(__dirname, 'src/schemas'), 'schemas'),
|
||||||
new CopyDirPlugin(path.join(__dirname, 'src/images/logo/*'), ''),
|
new CopyDirPlugin(path.join(__dirname, 'src/images/logo/*'), ''),
|
||||||
@@ -75,14 +83,14 @@ module.exports = {
|
|||||||
],
|
],
|
||||||
module: {
|
module: {
|
||||||
rules: [
|
rules: [
|
||||||
{ test: /\.css$/, loader: ExtractTextPlugin.extract({ fallback: 'style-loader', use: 'css-loader'}) },
|
{test: /\.css$/, loader: ExtractTextPlugin.extract({fallback: 'style-loader', use: 'css-loader'})},
|
||||||
{ test: /\.less$/, loader: ExtractTextPlugin.extract({ fallback: 'style-loader',use: 'css-loader!less-loader'}) },
|
{test: /\.less$/, loader: ExtractTextPlugin.extract({fallback: 'style-loader', use: 'css-loader!less-loader'})},
|
||||||
{ test: /\.(js|jsx)$/, loader: 'babel-loader?cacheDirectory=true', include: path.join(__dirname, 'src') },
|
{test: /\.(js|jsx)$/, loader: 'babel-loader?cacheDirectory=true', include: path.join(__dirname, 'src')},
|
||||||
{ test: /\.woff(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff' },
|
{test: /\.woff(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff'},
|
||||||
{ test: /\.woff2(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff' },
|
{test: /\.woff2(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/font-woff'},
|
||||||
{ test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/octet-stream' },
|
{test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=application/octet-stream'},
|
||||||
{ test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, loader: 'file-loader' },
|
{test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, loader: 'file-loader'},
|
||||||
{ test: /\.svg(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=image/svg+xml' }
|
{test: /\.svg(\?v=\d+\.\d+\.\d+)?$/, loader: 'url-loader?limit=10000&mimetype=image/svg+xml'}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user