render power data in table format
This commit is contained in:
@@ -1,18 +1,25 @@
|
|||||||
const ENDPOINT = '/data/v1/energy';
|
|
||||||
import extend from 'extend';
|
import extend from 'extend';
|
||||||
|
|
||||||
|
const ENDPOINT = '/data/v1/energy';
|
||||||
|
|
||||||
|
// send all date parameters as unix timestamps;
|
||||||
class EnergyDataApi {
|
class EnergyDataApi {
|
||||||
|
|
||||||
static index(params){
|
static index(params){
|
||||||
params = extend({
|
params = extend({
|
||||||
|
|
||||||
}, params);
|
}, params);
|
||||||
|
if (params.dates){
|
||||||
|
params.dates = params.dates.map((date_range)=>{
|
||||||
|
if (date_range[0]) date_range[0] = date_range[0].unix();
|
||||||
|
if (date_range[1]) date_range[1] = date_range[1].unix();
|
||||||
|
return [date_range[0], date_range[1]];
|
||||||
|
})
|
||||||
|
}
|
||||||
return jQuery.ajax({
|
return jQuery.ajax({
|
||||||
url: `${ENDPOINT}`,
|
url: ENDPOINT + '?' + jQuery.param(params),
|
||||||
type: 'GET',
|
type: 'GET',
|
||||||
params: params,
|
|
||||||
dataType: 'json'
|
dataType: 'json'
|
||||||
}).success((res)=>{
|
}).then((res)=>{
|
||||||
return res.data;
|
return res.data;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
const ENDPOINT = '/data/v1/houses';
|
||||||
|
import extend from 'extend';
|
||||||
|
|
||||||
|
class HousesApi {
|
||||||
|
|
||||||
|
static index(params){
|
||||||
|
return jQuery.ajax({
|
||||||
|
url: ENDPOINT + '?' + jQuery.param(params),
|
||||||
|
type: 'GET',
|
||||||
|
dataType: 'json'
|
||||||
|
}).then((res)=>{
|
||||||
|
return res.data;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default HousesApi;
|
||||||
|
|||||||
@@ -1 +1,29 @@
|
|||||||
|
const ENDPOINT = '/data/v1/power';
|
||||||
|
import extend from 'extend';
|
||||||
|
|
||||||
|
// send all date parameters as unix timestamps;
|
||||||
|
class PowerDataApi {
|
||||||
|
|
||||||
|
static index(params){
|
||||||
|
params = extend({
|
||||||
|
}, params);
|
||||||
|
if (params.dates){
|
||||||
|
params.dates = params.dates.map((date_range)=>{
|
||||||
|
if (date_range[0]) date_range[0] = date_range[0].unix();
|
||||||
|
if (date_range[1]) date_range[1] = date_range[1].unix();
|
||||||
|
return [date_range[0], date_range[1]];
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return jQuery.ajax({
|
||||||
|
url: ENDPOINT + '?' + jQuery.param(params),
|
||||||
|
type: 'GET',
|
||||||
|
dataType: 'json'
|
||||||
|
}).then((res)=>{
|
||||||
|
return res.data;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default PowerDataApi;
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
import 'babel-polyfill';
|
import 'babel-polyfill';
|
||||||
|
import 'bootstrap/dist/js/bootstrap.min';
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import ReactDOM from 'react-dom';
|
import ReactDOM from 'react-dom';
|
||||||
import Layout from './dashboard/layout/layout';
|
import Layout from './dashboard/layout/layout';
|
||||||
|
|
||||||
|
|
||||||
ReactDOM.render(
|
ReactDOM.render(
|
||||||
React.createElement(Layout),
|
React.createElement(Layout),
|
||||||
document.getElementById('root')
|
document.getElementById('root')
|
||||||
|
|||||||
1
client/config/api.js
Normal file
1
client/config/api.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
api.js
|
||||||
1
client/config/store.js
Normal file
1
client/config/store.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
store.js
|
||||||
@@ -1,11 +1,21 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import layoutRt from './layout.rt.js';
|
import layoutRt from './layout.rt.js';
|
||||||
|
import House from './../../models/house';
|
||||||
|
|
||||||
|
const VIEWS = [['power', 'Power Savings'], ['energy', 'Energy Production']];
|
||||||
|
|
||||||
var Layout = React.createClass({
|
var Layout = React.createClass({
|
||||||
|
|
||||||
getInitialState: function(){
|
getInitialState: function(){
|
||||||
return {view: "????"};
|
var layout = this;
|
||||||
|
layout.view_name = VIEWS[0][1];
|
||||||
|
return {
|
||||||
|
views: VIEWS,
|
||||||
|
houses: null,
|
||||||
|
house: null,
|
||||||
|
view: 'power',
|
||||||
|
requesting_data: true
|
||||||
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
handleResize: function(e) {
|
handleResize: function(e) {
|
||||||
@@ -13,13 +23,49 @@ var Layout = React.createClass({
|
|||||||
},
|
},
|
||||||
|
|
||||||
componentDidMount: function() {
|
componentDidMount: function() {
|
||||||
window.addEventListener('resize', this.handleResize);
|
var layout = this;
|
||||||
|
// window.addEventListener('resize', this.handleResize);
|
||||||
|
House.ensureHouses().then((houses)=>{
|
||||||
|
layout.setState({houses: houses, house: houses[0]});
|
||||||
|
layout.ensureHouseViewData();
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
setView: function(event) {
|
setView: function(event){
|
||||||
var layout = this;
|
var layout = this,
|
||||||
console.log(event.target.value)
|
view = event.target.value;
|
||||||
layout.setState({view: event.target.value});
|
layout.view_name = event.target.innerText;
|
||||||
|
layout.setState({view: view});
|
||||||
|
layout.ensureHouseViewData();
|
||||||
|
},
|
||||||
|
|
||||||
|
setHouse: function(event){
|
||||||
|
var layout = this,
|
||||||
|
house_id = event.target.value,
|
||||||
|
house = layout.state.houses.find((house)=>{ return house.data.id == house_id });
|
||||||
|
layout.setState({house: house}, function(){
|
||||||
|
layout.ensureHouseViewData();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
ensureHouseViewData: function(){
|
||||||
|
var layout = this,
|
||||||
|
house = layout.state.house,
|
||||||
|
view = layout.state.view,
|
||||||
|
request;
|
||||||
|
layout.setState({requesting_data: true}, ()=>{
|
||||||
|
if (view === 'power'){
|
||||||
|
request = house.ensurePowerData();
|
||||||
|
} else {
|
||||||
|
request = house.ensureEnergyData();
|
||||||
|
}
|
||||||
|
request.then(()=>{
|
||||||
|
console.log('data retrieved')
|
||||||
|
layout.setState({requesting_data: false}, ()=>{
|
||||||
|
console.log(layout.state.requesting_data);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
render: function() {
|
render: function() {
|
||||||
|
|||||||
@@ -1,8 +1,54 @@
|
|||||||
|
|
||||||
<div id="layout">
|
<div id="layout">
|
||||||
<h1>{this.state.view}</h1>
|
<div class="alert alert-warning" rt-if="this.state.requesting_data">Loading data...</div>
|
||||||
<select onChange="{this.setView}">
|
<h1 rt-if="this.state.house">{this.state.house.name}</h1>
|
||||||
<option value="savings">Savings</option>
|
<h3 rt-if="this.state.view">{this.view_name}</h3>
|
||||||
<option value="production">Production</option>
|
|
||||||
</select>
|
<div>
|
||||||
|
<select class="form-control" onChange="{this.setView}">
|
||||||
|
<option rt-repeat="view in this.state.views" value="{view[0]}" key="view-{view[0]}">{view[1]}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div rt-if="this.state.houses">
|
||||||
|
<select class="form-control" onChange="{this.setHouse}">
|
||||||
|
<option rt-repeat="house in this.state.houses" value="{house.data.id}" key="{house.react_key}">{house.data.name}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table class="table" rt-if="this.state.view === 'energy' && this.state.house">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th></th>
|
||||||
|
<th>Day</th>
|
||||||
|
<th>Consumption (Wh)</th>
|
||||||
|
<th>Production (Wh)</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr rt-repeat="energy_datum in this.state.house.energy_data" key="{energy_datum.react_key}">
|
||||||
|
<td></td>
|
||||||
|
<td>{energy_datum.day_to_s}</td>
|
||||||
|
<td>{energy_datum.consumption_to_s}</td>
|
||||||
|
<td>{energy_datum.production_to_s}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<table class="table" rt-if="this.state.view === 'power' && this.state.house">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th></th>
|
||||||
|
<th>Time</th>
|
||||||
|
<th>Consumption (W)</th>
|
||||||
|
<th>Production (W)</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr rt-repeat="power_datum in this.state.house.power_data" key="{power_datum.react_key}">
|
||||||
|
<td></td>
|
||||||
|
<td>{power_datum.time_to_s}</td>
|
||||||
|
<td>{power_datum.consumption_to_s}</td>
|
||||||
|
<td>{power_datum.production_to_s}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,5 +1,45 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
|
function repeatView1(view, viewIndex) {
|
||||||
|
return React.createElement('option', {
|
||||||
|
'value': view[0],
|
||||||
|
'key': 'view-' + view[0]
|
||||||
|
}, view[1]);
|
||||||
|
}
|
||||||
|
function repeatHouse2(house, houseIndex) {
|
||||||
|
return React.createElement('option', {
|
||||||
|
'value': house.data.id,
|
||||||
|
'key': house.react_key
|
||||||
|
}, house.data.name);
|
||||||
|
}
|
||||||
|
function repeatEnergy_datum3(energy_datum, energy_datumIndex) {
|
||||||
|
return React.createElement('tr', { 'key': energy_datum.react_key }, React.createElement('td', {}), React.createElement('td', {}, energy_datum.day_to_s), React.createElement('td', {}, energy_datum.consumption_to_s), React.createElement('td', {}, energy_datum.production_to_s));
|
||||||
|
}
|
||||||
|
function repeatPower_datum4(power_datum, power_datumIndex) {
|
||||||
|
return React.createElement('tr', { 'key': power_datum.react_key }, React.createElement('td', {}), React.createElement('td', {}, power_datum.time_to_s), React.createElement('td', {}, power_datum.consumption_to_s), React.createElement('td', {}, power_datum.production_to_s));
|
||||||
|
}
|
||||||
export default function () {
|
export default function () {
|
||||||
return React.createElement('div', { 'id': 'layout' }, React.createElement('h1', {}, this.state.view), React.createElement('select', { 'onChange': this.setView }, React.createElement('option', { 'value': 'savings' }, 'Savings'), React.createElement('option', { 'value': 'production' }, 'Production')));
|
return React.createElement('div', { 'id': 'layout' }, this.state.requesting_data ? React.createElement('div', { 'className': 'alert alert-warning' }, 'Loading data...') : null, this.state.house ? React.createElement('h1', {}, this.state.house.name) : null, this.state.view ? React.createElement('h3', {}, this.view_name) : null, React.createElement('div', {}, React.createElement.apply(this, [
|
||||||
|
'select',
|
||||||
|
{
|
||||||
|
'className': 'form-control',
|
||||||
|
'onChange': this.setView
|
||||||
|
},
|
||||||
|
_.map(this.state.views, repeatView1.bind(this))
|
||||||
|
])), this.state.houses ? React.createElement('div', {}, React.createElement.apply(this, [
|
||||||
|
'select',
|
||||||
|
{
|
||||||
|
'className': 'form-control',
|
||||||
|
'onChange': this.setHouse
|
||||||
|
},
|
||||||
|
_.map(this.state.houses, repeatHouse2.bind(this))
|
||||||
|
])) : null, this.state.view === 'energy' && this.state.house ? React.createElement('table', { 'className': 'table' }, React.createElement('thead', {}, React.createElement('tr', {}, React.createElement('th', {}), React.createElement('th', {}, 'Day'), React.createElement('th', {}, 'Consumption (Wh)'), React.createElement('th', {}, 'Production (Wh)'))), React.createElement.apply(this, [
|
||||||
|
'tbody',
|
||||||
|
{},
|
||||||
|
_.map(this.state.house.energy_data, repeatEnergy_datum3.bind(this))
|
||||||
|
])) : null, this.state.view === 'power' && this.state.house ? React.createElement('table', { 'className': 'table' }, React.createElement('thead', {}, React.createElement('tr', {}, React.createElement('th', {}), React.createElement('th', {}, 'Time'), React.createElement('th', {}, 'Consumption (W)'), React.createElement('th', {}, 'Production (W)'))), React.createElement.apply(this, [
|
||||||
|
'tbody',
|
||||||
|
{},
|
||||||
|
_.map(this.state.house.power_data, repeatPower_datum4.bind(this))
|
||||||
|
])) : null);
|
||||||
};
|
};
|
||||||
@@ -1,13 +1,17 @@
|
|||||||
import extend from 'extend';
|
import extend from 'extend';
|
||||||
|
import moment from 'moment-timezone';
|
||||||
|
|
||||||
class EnergyDatum {
|
class EnergyDatum {
|
||||||
__constructor(data, house){
|
constructor(data, house){
|
||||||
var energy_datum = this;
|
var energy_datum = this;
|
||||||
energy_datum.house = house;
|
energy_datum.house = house;
|
||||||
|
data.day = moment.tz(data.day, house.data.timezone);
|
||||||
energy_datum.data = data;
|
energy_datum.data = data;
|
||||||
moment.tz(data.day, house.data.timezone);
|
EnergyDatum.store.set(data.id, energy_datum);
|
||||||
EnergyDatum.store[data.id] energy_datum;
|
}
|
||||||
|
|
||||||
|
get react_key(){
|
||||||
|
return `energy-datum-${this.data.id}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
get day_to_date(){
|
get day_to_date(){
|
||||||
@@ -17,6 +21,21 @@ class EnergyDatum {
|
|||||||
return moment(moment_tz.toArray()).toDate();
|
return moment(moment_tz.toArray()).toDate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get day_to_s(){
|
||||||
|
var energy_datum = this;
|
||||||
|
return energy_datum.data.day.format('YYYY-MM-DD');
|
||||||
|
}
|
||||||
|
|
||||||
|
get consumption_to_s(){
|
||||||
|
var energy_datum = this;
|
||||||
|
return Math.round(energy_datum.data.consumption);
|
||||||
|
}
|
||||||
|
|
||||||
|
get production_to_s(){
|
||||||
|
var energy_datum = this;
|
||||||
|
return Math.round(energy_datum.data.production);
|
||||||
|
}
|
||||||
|
|
||||||
update(data){
|
update(data){
|
||||||
var energy_datum = this,
|
var energy_datum = this,
|
||||||
house = power_datum.house;
|
house = power_datum.house;
|
||||||
@@ -24,11 +43,13 @@ class EnergyDatum {
|
|||||||
extend(energy_datum.data, data);
|
extend(energy_datum.data, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static updateOrInitialize(id, data, house){
|
static updateOrInitialize(data, house){
|
||||||
var energy_datum = EnergyDatum.store.get(id);
|
var energy_datum = EnergyDatum.store.get(data.id);
|
||||||
if (energy_datum) energy_datum.update(data);
|
if (energy_datum) energy_datum.update(data);
|
||||||
return energy_datum || new EnergyDatum(data, house)
|
return energy_datum || new EnergyDatum(data, house)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
EnergyDatum.store = new Map();
|
EnergyDatum.store = new Map();
|
||||||
|
|
||||||
|
export default EnergyDatum;
|
||||||
|
|||||||
@@ -1,38 +1,56 @@
|
|||||||
import Api from './../api';
|
import extend from 'extend';
|
||||||
import Store from './../store';
|
|
||||||
import ArrayUtil from './../../shared/util/array'
|
|
||||||
|
|
||||||
class House extends Base {
|
import PowerDatum from './power_datum';
|
||||||
|
import EnergyDatum from './energy_datum';
|
||||||
|
import PowerDataApi from './../api/power_data';
|
||||||
|
import EnergyDataApi from './../api/energy_data';
|
||||||
|
import HousesApi from './../api/houses';
|
||||||
|
import ArrayUtil from './../../shared/utils/array'
|
||||||
|
import MathUtil from './../../shared/utils/math'
|
||||||
|
|
||||||
__constructor(data, house){
|
class House {
|
||||||
var energy_datum = this;
|
|
||||||
|
constructor(data){
|
||||||
|
var house = this;
|
||||||
House.store.set(data.id, house);
|
House.store.set(data.id, house);
|
||||||
house.data = data;
|
house.data = data;
|
||||||
house.energy_data = new Map();
|
house.energy_data = [];
|
||||||
house.power_data = new Map();
|
house.power_data = [];
|
||||||
|
house.energy_data_store = new Map();
|
||||||
|
house.power_data_store = new Map();
|
||||||
}
|
}
|
||||||
|
|
||||||
ensurePowerData(start_date, end_date){
|
get react_key(){
|
||||||
|
return `house-${this.data.id}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
ensurePowerData(opts){
|
||||||
|
opts = extend({
|
||||||
|
start_date: undefined,
|
||||||
|
end_date: undefined
|
||||||
|
}, opts || {});
|
||||||
var house = this,
|
var house = this,
|
||||||
date_range = Array.from(house.power_data.keys()),
|
date_range = Array.from(house.power_data_store.keys()),
|
||||||
min_date = Math.min(date_range),
|
min_date = Math.min(date_range),
|
||||||
max_date = Math.max(date_range),
|
max_date = Math.max(date_range),
|
||||||
query_ranges, cache;
|
query_ranges, cache;
|
||||||
|
|
||||||
if (date_range.length === 0) return house.getPowerData({dates: [[start_date, end_date]]})
|
if (date_range.length === 0){
|
||||||
|
return house.getPowerData({dates: [[opts.start_date, opts.end_date]]});
|
||||||
|
}
|
||||||
|
|
||||||
query_ranges = MathUtil.minusRange([start_date, end_date], [min_date, max_date]);
|
query_ranges = MathUtil.minusRange([opts.start_date, opts.end_date], [min_date, max_date]);
|
||||||
|
|
||||||
cache = ArrayUtil.selectMap(date_range, (datum_time)=>{
|
cache = ArrayUtil.selectMap(date_range, (datum_time)=>{
|
||||||
return ArrayUtil.all(query_ranges, (query_range)=>{
|
return ArrayUtil.all(query_ranges, (query_range)=>{
|
||||||
!MathUtil.inRange(datum_time, query_range);
|
!MathUtil.inRange(datum_time, query_range);
|
||||||
}));
|
});
|
||||||
}, (datum_time)=>{
|
}, (datum_time)=>{
|
||||||
return house.power_data.get(datum_time);
|
return house.power_data_store.get(datum_time);
|
||||||
});
|
});
|
||||||
|
|
||||||
if (query_ranges.length > 0){
|
if (query_ranges.length > 0){
|
||||||
return house.getPowerData({dates: dates}).then((new_power_data)=>{
|
return house.getPowerData({dates: query_ranges}).then((new_power_data)=>{
|
||||||
return new_power_data.concat(cache);
|
return new_power_data.concat(cache);
|
||||||
});
|
});
|
||||||
} else return Promise.resolve(cache);
|
} else return Promise.resolve(cache);
|
||||||
@@ -41,37 +59,42 @@ class House extends Base {
|
|||||||
getPowerData(params){
|
getPowerData(params){
|
||||||
var house = this;
|
var house = this;
|
||||||
params.house_id = house.data.id;
|
params.house_id = house.data.id;
|
||||||
return Api.PowerData.index(params).then((power_data)=>{
|
return PowerDataApi.index(params).then((power_data)=>{
|
||||||
return power_data.map((power_datum_data)=>{
|
return power_data.map((power_datum_data)=>{
|
||||||
var power_datum = Store.PowerDatum.updateOrInitialize(power_datum_data, house);
|
var power_datum = PowerDatum.updateOrInitialize(power_datum_data, house);
|
||||||
house.power_data.set(power_datum.data.time, power_datum);
|
house.power_data_store.set(power_datum.data.time, power_datum);
|
||||||
|
house.power_data.push(power_datum);
|
||||||
return power_datum;
|
return power_datum;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
ensureEnergyData(start_date, end_date){
|
ensureEnergyData(opts){
|
||||||
|
opts = extend({
|
||||||
|
start_date: undefined,
|
||||||
|
end_date: undefined
|
||||||
|
}, opts || {});
|
||||||
var house = this,
|
var house = this,
|
||||||
date_range = Array.from(house.energy_data.keys()),
|
date_range = Array.from(house.energy_data_store.keys()),
|
||||||
min_date = Math.min(date_range),
|
min_date = Math.min(date_range),
|
||||||
max_date = Math.max(date_range),
|
max_date = Math.max(date_range),
|
||||||
query_ranges, cache;
|
query_ranges, cache;
|
||||||
|
|
||||||
if (date_range.length === 0) return house.getEnergyData({dates: [[start_date, end_date]]})
|
if (date_range.length === 0) return house.getEnergyData({dates: [[opts.start_date, opts.end_date]]})
|
||||||
|
|
||||||
query_ranges = MathUtil.minusRange([start_date, end_date], [min_date, max_date]);
|
query_ranges = MathUtil.minusRange([start_date, end_date], [min_date, max_date]);
|
||||||
|
|
||||||
cache = ArrayUtil.selectMap(date_range, (datum_day)=>{
|
cache = ArrayUtil.selectMap(date_range, (datum_day)=>{
|
||||||
return ArrayUtil.all(query_ranges, (query_range)=>{
|
return ArrayUtil.all(query_ranges, (query_range)=>{
|
||||||
!MathUtil.inRange(datum_day, query_range);
|
return !MathUtil.inRange(datum_day, query_range);
|
||||||
}));
|
});
|
||||||
}, (datum_day)=>{
|
}, (datum_day)=>{
|
||||||
return house.energy_data.get(datum_day);
|
return house.energy_data_store.get(datum_day);
|
||||||
});
|
});
|
||||||
|
|
||||||
if (query_ranges.length > 0){
|
if (query_ranges.length > 0){
|
||||||
return house.getEnergyData({dates: dates}).then((new_energy_data)=>{
|
return house.getEnergyData({dates: query_ranges}).then((new_energy_data)=>{
|
||||||
return new_energy_data.concat(cache);
|
return new_energy_data_store.concat(cache);
|
||||||
});
|
});
|
||||||
} else return Promise.resolve(cache);
|
} else return Promise.resolve(cache);
|
||||||
}
|
}
|
||||||
@@ -79,10 +102,11 @@ class House extends Base {
|
|||||||
getEnergyData(params){
|
getEnergyData(params){
|
||||||
var house = this;
|
var house = this;
|
||||||
params.house_id = house.data.id;
|
params.house_id = house.data.id;
|
||||||
return Api.PowerData.index(params).then((energy_data)=>{
|
return EnergyDataApi.index(params).then((energy_data)=>{
|
||||||
return power_data.map((energy_datum_data)=>{
|
return energy_data.map((energy_datum_data)=>{
|
||||||
var energy_datum = Store.EnergyDatum.updateOrInitialize(energy_datum_data, house);
|
var energy_datum = EnergyDatum.updateOrInitialize(energy_datum_data, house);
|
||||||
house.energy_data.set(power_datum.time, energy_datum);
|
house.energy_data_store.set(power_datum.time, energy_datum);
|
||||||
|
house.energy_data.push(energy_datum);
|
||||||
return energy_datum;
|
return energy_datum;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -100,18 +124,21 @@ class House extends Base {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static ensureHouses(ids){
|
static ensureHouses(ids){
|
||||||
var required_ids = ArrayUtil.diff(ids, House.store.keys()),
|
var required_ids, cached_houses = [];
|
||||||
|
if (ids){
|
||||||
|
required_ids = ArrayUtil.diff(ids, House.store.keys());
|
||||||
cached_houses = ArrayUtil.diff(ids, required_ids).map((id)=>{ return House.store.get(id); });
|
cached_houses = ArrayUtil.diff(ids, required_ids).map((id)=>{ return House.store.get(id); });
|
||||||
if (required_ids.length == 0) return Promise.resolve([]);
|
}
|
||||||
|
if (required_ids && required_ids.length == 0) return Promise.resolve([]);
|
||||||
|
|
||||||
return House.getHouses(required_ids).then((new_houses){
|
return House.getHouses(required_ids).then((new_houses)=>{
|
||||||
// if these need to be ordered, then concatenation needs to be merged in order.
|
// if these need to be ordered, then concatenation needs to be merged in order.
|
||||||
return new_houses.concat(cached_houses);
|
return new_houses.concat(cached_houses);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
static getHouses(ids){
|
static getHouses(ids){
|
||||||
return Api.HousesApi.index({id: ids}).then((houses_data)=>{
|
return HousesApi.index({id: ids}).then((houses_data)=>{
|
||||||
return houses_data.map((house_data)=>{
|
return houses_data.map((house_data)=>{
|
||||||
return new House(house_data);
|
return new House(house_data);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,13 +1,17 @@
|
|||||||
import extend from 'extend';
|
import extend from 'extend';
|
||||||
|
import moment from 'moment-timezone';
|
||||||
|
|
||||||
class PowerDatum {
|
class PowerDatum {
|
||||||
__constructor(data, house){
|
constructor(data, house){
|
||||||
var power_datum = this;
|
var power_datum = this;
|
||||||
power_datum.house = house;
|
power_datum.house = house;
|
||||||
|
data.time = moment.tz(data.time, house.data.timezone);
|
||||||
power_datum.data = data;
|
power_datum.data = data;
|
||||||
moment.format(data.time);
|
PowerDatum.store.set(data.id, power_datum);
|
||||||
PowerDatum.store[data.id] power_datum;
|
}
|
||||||
|
|
||||||
|
get react_key(){
|
||||||
|
return `power-datum-${this.data.id}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
get time_to_date(){
|
get time_to_date(){
|
||||||
@@ -17,6 +21,19 @@ class PowerDatum {
|
|||||||
return moment(moment_tz.toArray()).toDate();
|
return moment(moment_tz.toArray()).toDate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get time_to_s(){
|
||||||
|
var power_datum = this;
|
||||||
|
return power_datum.data.time.format('YYYY-MM-DD HH:MM');
|
||||||
|
}
|
||||||
|
get consumption_to_s(){
|
||||||
|
var power_datum = this;
|
||||||
|
return Math.round(power_datum.data.consumption);
|
||||||
|
}
|
||||||
|
get production_to_s(){
|
||||||
|
var power_datum = this;
|
||||||
|
return Math.round(power_datum.data.production);
|
||||||
|
}
|
||||||
|
|
||||||
update(data){
|
update(data){
|
||||||
var power_datum = this,
|
var power_datum = this,
|
||||||
house = power_datum.house;
|
house = power_datum.house;
|
||||||
@@ -24,8 +41,8 @@ class PowerDatum {
|
|||||||
extend(power_datum.data, data);
|
extend(power_datum.data, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static updateOrInitialize(id, data, house){
|
static updateOrInitialize(data, house){
|
||||||
var power_datum = PowerDatum.store.get(id);
|
var power_datum = PowerDatum.store.get(data.id);
|
||||||
if (power_datum) power_datum.update(data);
|
if (power_datum) power_datum.update(data);
|
||||||
return power_datum || new PowerDatum(data, house)
|
return power_datum || new PowerDatum(data, house)
|
||||||
}
|
}
|
||||||
@@ -33,3 +50,5 @@ class PowerDatum {
|
|||||||
}
|
}
|
||||||
|
|
||||||
PowerDatum.store = new Map();
|
PowerDatum.store = new Map();
|
||||||
|
|
||||||
|
export default PowerDatum;
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
0 info it worked if it ends with ok
|
|
||||||
1 verbose cli [ '/usr/local/bin/node', '/usr/local/bin/npm', 'start' ]
|
|
||||||
2 info using npm@3.5.3
|
|
||||||
3 info using node@v5.4.1
|
|
||||||
4 verbose run-script [ 'prestart', 'start', 'poststart' ]
|
|
||||||
5 info lifecycle spike_proto@0.0.0~prestart: spike_proto@0.0.0
|
|
||||||
6 silly lifecycle spike_proto@0.0.0~prestart: no script for prestart, continuing
|
|
||||||
7 info lifecycle spike_proto@0.0.0~start: spike_proto@0.0.0
|
|
||||||
8 verbose lifecycle spike_proto@0.0.0~start: unsafe-perm in lifecycle true
|
|
||||||
9 verbose lifecycle spike_proto@0.0.0~start: PATH: /usr/local/lib/node_modules/npm/bin/node-gyp-bin:/home/eric/Code/spike2/node_modules/.bin:/home/eric/.rvm/gems/ruby-1.9.3-p484@oroeco_dev/bin:/home/eric/.rvm/gems/ruby-1.9.3-p484@global/bin:/home/eric/.rvm/rubies/ruby-1.9.3-p484/bin:/home/eric/.rvm/bin:/home/eric/bin:/usr/local/heroku/bin:/home/eric/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/opt/lampp/bin
|
|
||||||
10 verbose lifecycle spike_proto@0.0.0~start: CWD: /home/eric/Code/spike2
|
|
||||||
11 silly lifecycle spike_proto@0.0.0~start: Args: [ '-c', 'babel-node ./server/app.express.js' ]
|
|
||||||
12 silly lifecycle spike_proto@0.0.0~start: Returned: code: 1 signal: null
|
|
||||||
13 info lifecycle spike_proto@0.0.0~start: Failed to exec start script
|
|
||||||
14 verbose stack Error: spike_proto@0.0.0 start: `babel-node ./server/app.express.js`
|
|
||||||
14 verbose stack Exit status 1
|
|
||||||
14 verbose stack at EventEmitter.<anonymous> (/usr/local/lib/node_modules/npm/lib/utils/lifecycle.js:232:16)
|
|
||||||
14 verbose stack at emitTwo (events.js:87:13)
|
|
||||||
14 verbose stack at EventEmitter.emit (events.js:172:7)
|
|
||||||
14 verbose stack at ChildProcess.<anonymous> (/usr/local/lib/node_modules/npm/lib/utils/spawn.js:24:14)
|
|
||||||
14 verbose stack at emitTwo (events.js:87:13)
|
|
||||||
14 verbose stack at ChildProcess.emit (events.js:172:7)
|
|
||||||
14 verbose stack at maybeClose (internal/child_process.js:821:16)
|
|
||||||
14 verbose stack at Process.ChildProcess._handle.onexit (internal/child_process.js:211:5)
|
|
||||||
15 verbose pkgid spike_proto@0.0.0
|
|
||||||
16 verbose cwd /home/eric/Code/spike2
|
|
||||||
17 error Linux 3.19.0-49-generic
|
|
||||||
18 error argv "/usr/local/bin/node" "/usr/local/bin/npm" "start"
|
|
||||||
19 error node v5.4.1
|
|
||||||
20 error npm v3.5.3
|
|
||||||
21 error code ELIFECYCLE
|
|
||||||
22 error spike_proto@0.0.0 start: `babel-node ./server/app.express.js`
|
|
||||||
22 error Exit status 1
|
|
||||||
23 error Failed at the spike_proto@0.0.0 start script 'babel-node ./server/app.express.js'.
|
|
||||||
23 error Make sure you have the latest version of node.js and npm installed.
|
|
||||||
23 error If you do, this is most likely a problem with the spike_proto package,
|
|
||||||
23 error not with npm itself.
|
|
||||||
23 error Tell the author that this fails on your system:
|
|
||||||
23 error babel-node ./server/app.express.js
|
|
||||||
23 error You can get information on how to open an issue for this project with:
|
|
||||||
23 error npm bugs spike_proto
|
|
||||||
23 error Or if that isn't available, you can get their info via:
|
|
||||||
23 error npm owner ls spike_proto
|
|
||||||
23 error There is likely additional logging output above.
|
|
||||||
24 verbose exit [ 1, true ]
|
|
||||||
@@ -41,6 +41,7 @@
|
|||||||
"raw-loader": "0.5.1",
|
"raw-loader": "0.5.1",
|
||||||
"sass-loader": "3.1.2",
|
"sass-loader": "3.1.2",
|
||||||
"style-loader": "^0.12.3",
|
"style-loader": "^0.12.3",
|
||||||
|
"json-loader": "0.5.4",
|
||||||
"node-sass": "3.4.2",
|
"node-sass": "3.4.2",
|
||||||
"moment-timezone":"0.5.0",
|
"moment-timezone":"0.5.0",
|
||||||
"yargs": "3.32.0",
|
"yargs": "3.32.0",
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ var config = require('./config/webpack/development'),
|
|||||||
contentBase: __dirname + '/../client/build/development',
|
contentBase: __dirname + '/../client/build/development',
|
||||||
publicPath: "/assets/",
|
publicPath: "/assets/",
|
||||||
proxy: {
|
proxy: {
|
||||||
'/data': `http://localhost:${APP_PORT}`,
|
'/data*': `http://localhost:${API_PORT}`,
|
||||||
},
|
},
|
||||||
stats: {colors: true}
|
stats: {colors: true}
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -11,10 +11,6 @@ module.exports = {
|
|||||||
filename: '[name].js',
|
filename: '[name].js',
|
||||||
path: ROOT + 'client/build/development'
|
path: ROOT + 'client/build/development'
|
||||||
},
|
},
|
||||||
externals: {
|
|
||||||
jquery: "$",
|
|
||||||
d3: "d3"
|
|
||||||
},
|
|
||||||
module: {
|
module: {
|
||||||
loaders: [
|
loaders: [
|
||||||
{
|
{
|
||||||
@@ -26,6 +22,9 @@ module.exports = {
|
|||||||
}, {
|
}, {
|
||||||
test: /\.js$/,
|
test: /\.js$/,
|
||||||
loader: 'babel'
|
loader: 'babel'
|
||||||
|
}, {
|
||||||
|
test: /\.json$/,
|
||||||
|
loader: 'json-loader'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -37,6 +36,10 @@ module.exports = {
|
|||||||
$: "jquery",
|
$: "jquery",
|
||||||
jQuery: "jquery",
|
jQuery: "jquery",
|
||||||
"window.jQuery": "jquery"
|
"window.jQuery": "jquery"
|
||||||
|
}),
|
||||||
|
new webpack.ProvidePlugin({
|
||||||
|
d3: "d3",
|
||||||
|
"window.d3": "d3"
|
||||||
})
|
})
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,10 +5,8 @@ const NAME = 'EnergyController';
|
|||||||
class EnergyController{
|
class EnergyController{
|
||||||
|
|
||||||
static index(req, res){
|
static index(req, res){
|
||||||
DB.House.findOne({where: {name: req.housename}}).then((house)=>{
|
DB.EnergyDatum.exposeForHouseAtDates(req.query.house_id, req.query.dates).then((energy_data)=>{
|
||||||
house.getEnergyDataByTime(req.params.start_time, req.params.end_time).then((energy_data)=>{
|
req.json({data: energy_data});
|
||||||
req.json(energy_data);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
import DB from './../config/database.js';
|
import DB from './../config/database.js';
|
||||||
|
|
||||||
const NAME = HousesController;
|
const NAME = 'HousesController';
|
||||||
|
|
||||||
class HousesController{
|
class HousesController {
|
||||||
|
|
||||||
static index(req, res){
|
static index(req, res){
|
||||||
DB.House.findAll({attributes: ['id', 'name']}).then((houses)=>{
|
var params = {};
|
||||||
res.json(houses);
|
if (req.query.ids) query.id = ids;
|
||||||
|
DB.House.findAll({where: params, attributes: ['id', 'name', 'timezone']}).then((houses)=>{
|
||||||
|
res.json({data: houses.map((house)=>{ return house.dataValues; })});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,10 +5,9 @@ const NAME = 'PowerController';
|
|||||||
class PowerController{
|
class PowerController{
|
||||||
|
|
||||||
static index(req, res){
|
static index(req, res){
|
||||||
DB.House.findOne({where: {name: req.housename}}).then((house)=>{
|
console.log(req.query);
|
||||||
house.getPowerDataByTime(req.params.start_time, req.params.end_time).then((power_data)=>{
|
DB.PowerDatum.exposeForHouseAtDates(req.query.house_id, req.query.dates).then((power_data)=>{
|
||||||
res.json(power_data);
|
res.json({data: power_data});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
31
server/helpers/api_helper.js
Normal file
31
server/helpers/api_helper.js
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import moment from 'moment';
|
||||||
|
|
||||||
|
class ApiHelper {
|
||||||
|
|
||||||
|
// assume all dates from api coming as UNIX timestamps.
|
||||||
|
static datesParamToSequelize(dates, field_name){
|
||||||
|
if (!dates) return {};
|
||||||
|
var params = {};
|
||||||
|
|
||||||
|
if (dates.length > 1){
|
||||||
|
params['$or'] = [];
|
||||||
|
dates.forEach((min_max)=>{
|
||||||
|
var condition_n = {};
|
||||||
|
condition_n[field_name] = {};
|
||||||
|
if (min_max[0]) condition_n[field_name]['$gt'] = moment.unix(min_max[0]).toDate();
|
||||||
|
if (min_max[1]) condition_n[field_name]['$lt'] = moment.unix(min_max[1]).toDate();
|
||||||
|
if (Object.keys(condition_n).length) params['$or'].push(condition_n);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
var min_max = dates[0],
|
||||||
|
condition = {}
|
||||||
|
if (min_max[0]) params[field_name]['$gt'] = moment.unix(min_max[0]).toDate();
|
||||||
|
if (min_max[1]) params[field_name]['$lt'] = moment.unix(min_max[1]).toDate();
|
||||||
|
if (Object.keys(condition).length) params[field_name] = condition;
|
||||||
|
}
|
||||||
|
return params;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ApiHelper;
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
import DB from "./../config/database";
|
import DB from "./../config/database";
|
||||||
|
import extend from 'extend';
|
||||||
|
import ApiHelper from './../helpers/api_helper';
|
||||||
|
|
||||||
const NAME = 'EnergyDatum';
|
const NAME = 'EnergyDatum';
|
||||||
|
|
||||||
@@ -20,7 +22,12 @@ var EnergyDatum = DB.sequelize.define(NAME, {
|
|||||||
underscored: true,
|
underscored: true,
|
||||||
tableName: "energy_data",
|
tableName: "energy_data",
|
||||||
instanceMethods: {
|
instanceMethods: {
|
||||||
|
exposeToApi: ()=>{
|
||||||
|
var energy_datum = this,
|
||||||
|
values = this.dataValues;
|
||||||
|
values.energy_datum = energy_datum.day.getTime() / 1000;
|
||||||
|
return values;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
classMethods: {
|
classMethods: {
|
||||||
set: ()=>{
|
set: ()=>{
|
||||||
@@ -28,6 +35,18 @@ var EnergyDatum = DB.sequelize.define(NAME, {
|
|||||||
},
|
},
|
||||||
associate: ()=>{
|
associate: ()=>{
|
||||||
EnergyDatum.belongsTo(DB.House);
|
EnergyDatum.belongsTo(DB.House);
|
||||||
|
},
|
||||||
|
exposeForHouseAtDates: (house_id, dates)=>{
|
||||||
|
var params = {house_id: house_id};
|
||||||
|
extend(params, ApiHelper.datesParamToSequelize(dates, 'day'));
|
||||||
|
return EnergyDatum.findAll({
|
||||||
|
where: params,
|
||||||
|
attributes: ['id', 'production', 'consumption', 'day']
|
||||||
|
}).then((energy_data)=>{
|
||||||
|
return energy_data.map((energy_datum)=>{
|
||||||
|
return energy_datum.exposeToApi();
|
||||||
|
});
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -44,8 +44,6 @@ var House = DB.sequelize.define(NAME, {
|
|||||||
power_data.forEach((power_datum)=>{
|
power_data.forEach((power_datum)=>{
|
||||||
var day = house.timeToDateString(power_datum.time),
|
var day = house.timeToDateString(power_datum.time),
|
||||||
energy_datum = energy_data.get(day) || {production: 0, consumption: 0, day: day, house_id: house.id};
|
energy_datum = energy_data.get(day) || {production: 0, consumption: 0, day: day, house_id: house.id};
|
||||||
console.log(power_datum.time)
|
|
||||||
console.log(day)
|
|
||||||
energy_datum.production += power_datum.production;
|
energy_datum.production += power_datum.production;
|
||||||
energy_datum.consumption += power_datum.consumption;
|
energy_datum.consumption += power_datum.consumption;
|
||||||
energy_data.set(day, energy_datum);
|
energy_data.set(day, energy_datum);
|
||||||
@@ -62,10 +60,10 @@ var House = DB.sequelize.define(NAME, {
|
|||||||
associate: ()=>{
|
associate: ()=>{
|
||||||
House.hasMany(DB.PowerDatum, {as: 'PowerData'});
|
House.hasMany(DB.PowerDatum, {as: 'PowerData'});
|
||||||
},
|
},
|
||||||
getPowerDataByTime: (start_date, end_date)=>{
|
getPowerDataByTime: (dates)=>{
|
||||||
var params = {
|
var params = {
|
||||||
where: {time: {}},
|
where: {time: {}},
|
||||||
attributes: ['time', 'consumption', 'production']
|
attributes: ['id', 'time', 'consumption', 'production']
|
||||||
};
|
};
|
||||||
if (start_date) params.where.time.$gt = moment.utc(start_date).toDate();
|
if (start_date) params.where.time.$gt = moment.utc(start_date).toDate();
|
||||||
if (end_date) params.where.time.$lt = moment.utc(end_date).toDate();
|
if (end_date) params.where.time.$lt = moment.utc(end_date).toDate();
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
import DB from "./../config/database";
|
import DB from "./../config/database";
|
||||||
|
import extend from 'extend';
|
||||||
|
import ApiHelper from './../helpers/api_helper';
|
||||||
|
|
||||||
const NAME = 'PowerDatum';
|
const NAME = 'PowerDatum';
|
||||||
|
|
||||||
@@ -20,9 +22,28 @@ var PowerDatum = DB.sequelize.define(NAME, {
|
|||||||
underscored: true,
|
underscored: true,
|
||||||
tableName: "power_data",
|
tableName: "power_data",
|
||||||
instanceMethods: {
|
instanceMethods: {
|
||||||
|
exposeToApi: function(){
|
||||||
|
var power_datum = this,
|
||||||
|
data = power_datum.dataValues;
|
||||||
|
data.consumption = data.consumption * 4; // convert Wh / 15 minutes, to W
|
||||||
|
data.production = data.production * 4; // convert Wh / 15 minutes, to W
|
||||||
|
return data;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
classMethods: {
|
classMethods: {
|
||||||
|
exposeForHouseAtDates: (house_id, dates)=>{
|
||||||
|
var params = {house_id: house_id};
|
||||||
|
params = extend(params, ApiHelper.datesParamToSequelize(dates, 'time'));
|
||||||
|
console.log(params);
|
||||||
|
return PowerDatum.findAll({
|
||||||
|
where: params,
|
||||||
|
attributes: ['id', 'production', 'consumption', 'time']
|
||||||
|
}).then((power_data)=>{
|
||||||
|
return power_data.map((power_datum)=>{
|
||||||
|
return power_datum.exposeToApi();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
},
|
||||||
set: ()=>{
|
set: ()=>{
|
||||||
PowerDatum.associate();
|
PowerDatum.associate();
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ export default function(app){
|
|||||||
|
|
||||||
Controllers.sync();
|
Controllers.sync();
|
||||||
|
|
||||||
app.use('/data/v1/savings/:housename', Controllers.PowerController.index);
|
app.use('/data/v1/power', Controllers.PowerController.index);
|
||||||
app.use('/data/v1/production/:housename', Controllers.EnergyController.index);
|
app.use('/data/v1/energy', Controllers.EnergyController.index);
|
||||||
app.use('/data/v1/houses/', Controllers.EnergyController.index);
|
app.use('/data/v1/houses', Controllers.HousesController.index);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,15 +1,13 @@
|
|||||||
class ArrayUtil {
|
class ArrayUtil {
|
||||||
|
|
||||||
static diff(a1, a2){
|
static diff(a1, a2){
|
||||||
a1.filter((a1n)=>{ return a2.indexOf(a1n) < 0; });
|
return a1.filter((a1n)=>{ return a2.indexOf(a1n) < 0; });
|
||||||
}
|
}
|
||||||
|
|
||||||
static selectMap(a, fnSelect, fnMap){
|
static selectMap(a, fnSelect, fnMap){
|
||||||
var map = [];
|
var map = [];
|
||||||
for (var elem of a){
|
for (var elem of a){
|
||||||
if (fnSelect(elem)){
|
if (fnSelect(elem)) map.push(fnMap(elem));
|
||||||
map.push(fnMap(elem));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,7 +26,9 @@ export default class {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inRange(n, min_max){
|
static inRange(n, min_max){
|
||||||
return n >= min_max[0] && n =< min_max[1];
|
var min = min_max[0],
|
||||||
|
max = min_max[1];
|
||||||
|
return ((n >= min) && (n <= max));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user