-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathapi.js
153 lines (137 loc) · 5.12 KB
/
api.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
import moment from 'moment';
import queryString from 'query-string';
import isEqual from 'lodash/isEqual';
import { cartoLayerSource } from './app_config';
import { defaultState } from '../reducers/filter_by_area_reducer';
export const dateStringFormatModel = 'YYYY-MM';
export const dateStringFormatView = 'MMM, YYYY';
export const momentize = dateString => moment(dateString, dateStringFormatModel, true);
// Names for Filter by Boundary
export const geos = ['citywide', 'borough', 'community_board', 'city_council',
'neighborhood', 'assembly', 'senate', 'nypd_precinct', 'intersection', 'custom'];
// Borough Names mapped to array index position
export const boroughs = [undefined, 'Manhattan', 'Bronx', 'Brooklyn', 'Queens', 'Staten Island'];
// creates default app state using any available params from window.location.hash
export const makeDefaultState = () => {
const hash = window.location.hash;
const qString = hash.substring(3, hash.length);
const q = queryString.parse(qString);
const p = {};
const isJsonString = (str) => {
try {
JSON.parse(str);
} catch (e) {
return false;
}
return true;
};
const isBool = val => typeof val === 'boolean';
const isValidMomentObj = (dateString) => {
const m = dateString ? momentize(dateString) : undefined;
if (m && m.isValid()) {
return m;
}
// fallback to last month, which is more likely to have data then the current month
// e.g. if the current date is the first week of the month,
// then data may not exist for that month yet
const m2 = moment().subtract(1, 'month');
const defaultYearMonth = `${m2.year()}-${m2.format('MM')}`;
return momentize(defaultYearMonth);
};
const isValidGeo = (geo) => {
if (geos.indexOf(geo) !== -1) {
return geo;
}
return geos[0];
};
// parse query params, with compatibility hack issue 97 to accept Chart View identifier
// which is really the full string name with a borough prefix
Object.keys(q).forEach((key) => {
const decoded = decodeURIComponent(q[key]);
if (isJsonString(decoded)) {
p[key] = JSON.parse(decoded);
} else {
p[key] = decoded;
}
});
if (p.identifier && typeof p.identifier === 'string') {
if (p.identifier.indexOf(',') !== -1) {
p.identifier = p.identifier.split(',')[1];
}
p.identifier = p.identifier.trim();
const identifierEndsInNumber = p.identifier.match(/(\d+)$/);
if (identifierEndsInNumber) {
p.identifier = identifierEndsInNumber[1];
}
}
return {
filterDate: {
startDate: isValidMomentObj(p.startDate),
endDate: isValidMomentObj(p.endDate),
},
filterArea: {
...defaultState,
geo: isValidGeo(p.geo),
identifier: p.identifier || undefined,
lngLats: p.lngLats || [],
},
// filterTypes default to true for all injury & fatality
// except for noInjuryFatality which defaults to false
filterType: {
injury: {
cyclist: isBool(p.cinj) ? p.cinj : true,
motorist: isBool(p.minj) ? p.minj : true,
pedestrian: isBool(p.pinj) ? p.pinj : true,
},
fatality: {
cyclist: isBool(p.cfat) ? p.cfat : true,
motorist: isBool(p.mfat) ? p.mfat : true,
pedestrian: isBool(p.pfat) ? p.pfat : true,
},
noInjuryFatality: isBool(p.noInjFat) ? p.noInjFat : false,
},
filterVehicle: {
vehicle: {
car: isBool(p.vcar) ? p.vcar : true,
truck: isBool(p.vtruck) ? p.vtruck : true,
motorcycle: isBool(p.vmotorcycle) ? p.vmotorcycle : true,
bicycle: isBool(p.vbicycle) ? p.vbicycle : true,
suv: isBool(p.vsuv) ? p.vsuv : true,
busvan: isBool(p.vbusvan) ? p.vbusvan : true,
scooter: isBool(p.vscooter) ? p.vscooter : true,
other: isBool(p.vother) ? p.vother : true,
},
},
filterContributingFactor: p.contrFactor || 'ALL',
modal: {
showModal: false,
modalType: '',
}
};
};
// configures Carto crashes map layer's SQL
export const configureLayerSource = (sql) => {
cartoLayerSource.sublayers[0].sql = sql;
return cartoLayerSource;
};
// Should the component fetch new crash data?
// also used in App.js to determine whether to update URL params
// @param {object} curProps; the component's this.props
// @param {object} nextProps; the component's nextProps in componentWillReceiveProps
export const crashDataChanged = (curProps, nextProps) => {
const { endDate, startDate, filterType, identifier, geo, lngLats } = nextProps;
const { injury, fatality, noInjuryFatality } = filterType;
const { filterVehicle } = nextProps;
if (!startDate.isSame(curProps.startDate) ||
!endDate.isSame(curProps.endDate) ||
!isEqual(injury, curProps.filterType.injury) ||
!isEqual(fatality, curProps.filterType.fatality) ||
noInjuryFatality !== curProps.filterType.noInjuryFatality ||
!isEqual(filterVehicle.vehicle, curProps.filterVehicle.vehicle) ||
(geo === 'citywide' && curProps.geo !== 'citywide') ||
(identifier && identifier !== curProps.identifier) ||
(lngLats && lngLats.length && !isEqual(lngLats, curProps.lngLats))) {
return true;
}
return false;
};