@opentripplanner/vehicle-rental-overlay 5.0.0 → 6.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/styled.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"styled.js","names":["_styledComponents","_interopRequireDefault","require","_MapMarkerAlt","getPctIcon","percent","Math","floor","BaseBikeRentalIcon","exports","styled","div","withConfig","displayName","componentId","props","StationMarker","MapMarkerAlt","color"],"sources":["../src/styled.ts"],"sourcesContent":["import styled from \"styled-components\";\nimport { MapMarkerAlt } from \"@styled-icons/fa-solid/MapMarkerAlt\";\n\nconst getPctIcon = (percent: number) => {\n switch (Math.floor(percent * 10)) {\n case 0:\n return 'background-image: url(\"\");';\n case 1:\n return 'background-image: url(\"\");';\n case 2:\n return 'background-image: url(\"\")';\n case 3:\n return 'background-image: url(\"\");';\n case 4:\n return 'background-image: url(\"\");';\n case 5:\n return 'background-image: url(\"\");';\n case 6:\n return 'background-image: url(\"\");';\n case 7:\n return 'background-image: url(\"\");';\n case 8:\n return 'background-image: url(\"\");';\n case 9:\n return 'background-image: url(\"\");';\n case 10:\n return 'background-image: url(\"\");';\n default:\n return 'background-image: url(\"\");';\n }\n};\n/**\n * Bike rental icons are different from other vehicle rental types since they\n * typically have stations in addition to free-floating bikes. The stations are\n * drawn as svgs marking how full the stations are while the floating bikes have\n * their own unique icon.\n */\nexport const BaseBikeRentalIcon = styled.div<{ percent: number }>`\n background-position: center;\n background-repeat: no-repeat;\n background-size: contain;\n height: 24px;\n margin: auto;\n width: 24px;\n ${props => getPctIcon(props.percent)}\n`;\n\nexport const StationMarker = styled(MapMarkerAlt)`\n color: ${props => props.color};\n height: 12;\n width: 12;\n`;\n"],"mappings":";;;;;;;AAAA,IAAAA,iBAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,aAAA,GAAAD,OAAA;AAEA,MAAME,UAAU,GAAIC,OAAe,IAAK;EACtC,QAAQC,IAAI,CAACC,KAAK,CAACF,OAAO,GAAG,EAAE,CAAC;IAC9B,KAAK,CAAC;MACJ,OAAO,kjDAAkjD;IAC3jD,KAAK,CAAC;MACJ,OAAO,0sDAA0sD;IACntD,KAAK,CAAC;MACJ,OAAO,i0DAAi0D;IAC10D,KAAK,CAAC;MACJ,OAAO,07DAA07D;IACn8D,KAAK,CAAC;MACJ,OAAO,kjEAAkjE;IAC3jE,KAAK,CAAC;MACJ,OAAO,sqEAAsqE;IAC/qE,KAAK,CAAC;MACJ,OAAO,kyEAAkyE;IAC3yE,KAAK,CAAC;MACJ,OAAO,85EAA85E;IACv6E,KAAK,CAAC;MACJ,OAAO,khFAAkhF;IAC3hF,KAAK,CAAC;MACJ,OAAO,8oFAA8oF;IACvpF,KAAK,EAAE;MACL,OAAO,siCAAsiC;IAC/iC;MACE,OAAO,kjDAAkjD;EAC7jD;AACF,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACO,MAAMG,kBAAkB,GAAAC,OAAA,CAAAD,kBAAA,GAAGE,yBAAM,CAACC,GAAG,CAAAC,UAAA;EAAAC,WAAA;EAAAC,WAAA;AAAA,+HAOxCC,KAAK,IAAIX,UAAU,CAACW,KAAK,CAACV,OAAO,CAAC,CACrC;AAEM,MAAMW,aAAa,GAAAP,OAAA,CAAAO,aAAA,GAAG,IAAAN,yBAAM,EAACO,0BAAY,CAAC,CAAAL,UAAA;EAAAC,WAAA;EAAAC,WAAA;AAAA,uCACtCC,KAAK,IAAIA,KAAK,CAACG,KAAK,CAG9B","ignoreList":[]}
1
+ {"version":3,"file":"styled.js","sourceRoot":"","sources":["../src/styled.ts"],"names":[],"mappings":";;;;;;;;;;AAAA,wEAAuC;AACvC,oEAAmE;AAEnE,IAAM,UAAU,GAAG,UAAC,OAAe;IACjC,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,EAAE;QAChC,KAAK,CAAC;YACJ,OAAO,kjDAAkjD,CAAC;QAC5jD,KAAK,CAAC;YACJ,OAAO,0sDAA0sD,CAAC;QACptD,KAAK,CAAC;YACJ,OAAO,i0DAAi0D,CAAC;QAC30D,KAAK,CAAC;YACJ,OAAO,07DAA07D,CAAC;QACp8D,KAAK,CAAC;YACJ,OAAO,kjEAAkjE,CAAC;QAC5jE,KAAK,CAAC;YACJ,OAAO,sqEAAsqE,CAAC;QAChrE,KAAK,CAAC;YACJ,OAAO,kyEAAkyE,CAAC;QAC5yE,KAAK,CAAC;YACJ,OAAO,85EAA85E,CAAC;QACx6E,KAAK,CAAC;YACJ,OAAO,khFAAkhF,CAAC;QAC5hF,KAAK,CAAC;YACJ,OAAO,8oFAA8oF,CAAC;QACxpF,KAAK,EAAE;YACL,OAAO,siCAAsiC,CAAC;QAChjC;YACE,OAAO,kjDAAkjD,CAAC;KAC7jD;AACH,CAAC,CAAC;AACF;;;;;GAKG;AACU,QAAA,kBAAkB,GAAG,2BAAM,CAAC,GAAG,+NAAqB,sJAO7D,EAAkC,IACrC,KADG,UAAA,KAAK,IAAI,OAAA,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,EAAzB,CAAyB,EACpC;AAEW,QAAA,aAAa,GAAG,IAAA,2BAAM,EAAC,2BAAY,CAAC,oHAAA,aACtC,EAAoB,kCAG9B,KAHU,UAAA,KAAK,IAAI,OAAA,KAAK,CAAC,KAAK,EAAX,CAAW,EAG7B"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@opentripplanner/vehicle-rental-overlay",
3
- "version": "5.0.0",
3
+ "version": "6.0.0",
4
4
  "description": "A map overlay to show vehicle rentals from a specific company",
5
5
  "main": "lib/index.js",
6
6
  "module": "esm/index.js",
@@ -19,12 +19,12 @@
19
19
  "flat": "^5.0.2",
20
20
  "lodash.memoize": "^4.1.2",
21
21
  "@opentripplanner/base-map": "6.0.0",
22
- "@opentripplanner/core-utils": "13.0.1",
23
- "@opentripplanner/map-popup": "7.0.0",
24
- "@opentripplanner/from-to-location-picker": "4.0.1"
22
+ "@opentripplanner/from-to-location-picker": "4.0.1",
23
+ "@opentripplanner/map-popup": "8.0.0",
24
+ "@opentripplanner/core-utils": "14.2.0"
25
25
  },
26
26
  "devDependencies": {
27
- "@opentripplanner/types": "7.0.0"
27
+ "@opentripplanner/types": "8.2.0"
28
28
  },
29
29
  "peerDependencies": {
30
30
  "react": "^18.2.0",
@@ -1,7 +1,12 @@
1
+ /* eslint-disable react/display-name */
1
2
  import React, { ReactNode } from "react";
2
- import { action } from "@storybook/addon-actions";
3
+ import { action } from "storybook/actions";
3
4
 
4
- import { Company, Station } from "@opentripplanner/types";
5
+ import { Company } from "@opentripplanner/types";
6
+ import {
7
+ RentalVehicle,
8
+ VehicleRentalStation
9
+ } from "@opentripplanner/types/otp2";
5
10
  import bikeRentalStations from "../__mocks__/bike-rental-stations.json";
6
11
  import carRentalStations from "../__mocks__/car-rental-stations.json";
7
12
  import eScooterStations from "../__mocks__/e-scooter-rental-stations.json";
@@ -37,27 +42,30 @@ const INITIAL_ZOOM = 13;
37
42
 
38
43
  type StoryProps = {
39
44
  companies: string[];
40
- getStationName?: (configCompanies: Company[], station: Station) => string;
45
+ getStationName?: (
46
+ configCompanies: Company[],
47
+ station: VehicleRentalStation
48
+ ) => string;
41
49
  refreshVehicles: () => void;
42
- stations: Station[];
50
+ entities: (VehicleRentalStation | RentalVehicle)[];
43
51
  visible?: boolean;
44
52
  };
45
53
 
46
54
  const ZoomControlledMapWithVehicleRentalOverlay = ({
47
55
  companies,
56
+ entities,
48
57
  getStationName,
49
58
  refreshVehicles,
50
- stations,
51
59
  visible
52
60
  }: StoryProps) => (
53
61
  <VehicleRentalOverlay
54
62
  companies={companies}
55
63
  configCompanies={configCompanies}
64
+ entities={entities}
56
65
  getStationName={getStationName}
57
66
  id="test"
58
67
  refreshVehicles={refreshVehicles}
59
68
  setLocation={setLocation}
60
- stations={stations}
61
69
  visible={visible}
62
70
  />
63
71
  );
@@ -76,31 +84,30 @@ export const RentalBicycles = () => (
76
84
  <ZoomControlledMapWithVehicleRentalOverlay
77
85
  companies={["BIKETOWN"]}
78
86
  refreshVehicles={action("refresh bicycles")}
79
- stations={bikeRentalStations}
87
+ entities={bikeRentalStations}
80
88
  />
81
89
  );
82
90
 
83
- export const RentalBicyclesVisibilityControlledByKnob = ({
84
- visible
85
- }: {
86
- visible: boolean;
87
- }): ReactNode => {
88
- return (
89
- <ZoomControlledMapWithVehicleRentalOverlay
90
- companies={["BIKETOWN"]}
91
- refreshVehicles={action("refresh bicycles")}
92
- stations={bikeRentalStations}
93
- visible={visible}
94
- />
95
- );
91
+ export const RentalBicyclesVisibilityControlledByKnob = {
92
+ render: ({ visible }: { visible: boolean }): ReactNode => {
93
+ return (
94
+ <ZoomControlledMapWithVehicleRentalOverlay
95
+ companies={["BIKETOWN"]}
96
+ refreshVehicles={action("refresh bicycles")}
97
+ entities={bikeRentalStations}
98
+ visible={visible}
99
+ />
100
+ );
101
+ },
102
+
103
+ args: { visible: true }
96
104
  };
97
- RentalBicyclesVisibilityControlledByKnob.args = { visible: true };
98
105
 
99
106
  export const RentalCars = () => (
100
107
  <ZoomControlledMapWithVehicleRentalOverlay
101
108
  companies={["CAR2GO"]}
102
109
  refreshVehicles={action("refresh cars")}
103
- stations={carRentalStations}
110
+ entities={carRentalStations}
104
111
  />
105
112
  );
106
113
 
@@ -108,7 +115,7 @@ export const RentalEScooters = () => (
108
115
  <ZoomControlledMapWithVehicleRentalOverlay
109
116
  companies={["SHARED"]}
110
117
  refreshVehicles={action("refresh E-scooters")}
111
- stations={eScooterStations}
118
+ entities={eScooterStations}
112
119
  />
113
120
  );
114
121
 
@@ -117,6 +124,6 @@ export const RentalEScootersWithCustomNaming = () => (
117
124
  companies={["SHARED"]}
118
125
  getStationName={customStationName}
119
126
  refreshVehicles={action("refresh E-scooters")}
120
- stations={eScooterStations}
127
+ entities={eScooterStations}
121
128
  />
122
129
  );
package/src/index.tsx CHANGED
@@ -2,16 +2,15 @@ import { MarkerWithPopup, Popup } from "@opentripplanner/base-map";
2
2
  import {
3
3
  Company,
4
4
  ConfiguredCompany,
5
- MapLocationActionArg,
6
- Station
5
+ MapLocationActionArg
7
6
  } from "@opentripplanner/types";
8
- import React, { useEffect, useState } from "react";
9
7
  import {
10
- Layer,
11
- Source,
12
- useMap,
13
- ViewStateChangeEvent
14
- } from "react-map-gl/maplibre";
8
+ RentalVehicle,
9
+ VehicleRentalStation
10
+ } from "@opentripplanner/types/otp2";
11
+ import { Geometry } from "geojson";
12
+ import React, { useEffect, useState } from "react";
13
+ import { Layer, Source, useMap } from "react-map-gl/maplibre";
15
14
 
16
15
  import StationPopup from "@opentripplanner/map-popup";
17
16
  import { BaseBikeRentalIcon, StationMarker } from "./styled";
@@ -19,11 +18,20 @@ import { BaseBikeRentalIcon, StationMarker } from "./styled";
19
18
  // TODO: Make configurable?
20
19
  const DETAILED_MARKER_CUTOFF = 16;
21
20
 
22
- const getColorForStation = (v: Station) => {
23
- if (v.isFloatingCar) return "#009cde";
24
- if (v.isFloatingVehicle) return "#f5a729";
25
- // TODO: nicer color to match transitive
26
- if (v.bikesAvailable !== undefined || v.isFloatingBike) return "#f00";
21
+ function entityIsStation(
22
+ entity: VehicleRentalStation | RentalVehicle
23
+ ): entity is VehicleRentalStation {
24
+ return "availableVehicles" in entity;
25
+ }
26
+
27
+ const getColorForEntity = (entity: VehicleRentalStation | RentalVehicle) => {
28
+ if (entityIsStation(entity)) {
29
+ if (entity.availableVehicles && entity.availableVehicles.total > 0)
30
+ return "#f00";
31
+ } else {
32
+ if (entity.vehicleType.formFactor.startsWith("SCOOTER")) return "#f5a729";
33
+ if (entity.vehicleType.formFactor === "BICYCLE") return "#009cde";
34
+ }
27
35
  return "gray";
28
36
  };
29
37
 
@@ -37,6 +45,11 @@ type Props = {
37
45
  * The entire companies config array.
38
46
  */
39
47
  configCompanies: ConfiguredCompany[];
48
+ /**
49
+ * The entities to be represented in the overlay. They can be a combination of VehicleRentalStation type
50
+ * (for stationary stations) and RentalVehicle type (for floating vehicles)
51
+ */
52
+ entities?: (VehicleRentalStation | RentalVehicle)[];
40
53
  /**
41
54
  * An id, used to make this layer uniquely identifiable
42
55
  */
@@ -46,7 +59,10 @@ type Props = {
46
59
  * rental station. This function takes two arguments of the configCompanies
47
60
  * prop and a vehicle rental station. The function must return a string.
48
61
  */
49
- getStationName?: (configCompanies: Company[], station: Station) => string;
62
+ getStationName?: (
63
+ configCompanies: Company[],
64
+ station: VehicleRentalStation
65
+ ) => string;
50
66
  /**
51
67
  * If specified, a function that will be triggered every 30 seconds whenever this layer is
52
68
  * visible.
@@ -70,10 +86,6 @@ type Props = {
70
86
  * ```
71
87
  */
72
88
  setLocation?: (arg: MapLocationActionArg) => void;
73
- /**
74
- * A list of the vehicle rental stations specific to this overlay instance.
75
- */
76
- stations: Station[];
77
89
  /**
78
90
  * Whether the overlay is currently visible.
79
91
  */
@@ -91,18 +103,20 @@ type Props = {
91
103
  const VehicleRentalOverlay = ({
92
104
  companies,
93
105
  configCompanies,
106
+ entities,
94
107
  getStationName,
95
108
  id,
96
109
  refreshVehicles,
97
110
  setLocation,
98
- stations,
99
111
  visible
100
112
  }: Props): JSX.Element => {
101
113
  const { current: map } = useMap();
102
114
  const [zoom, setZoom] = useState(map?.getZoom());
103
115
 
104
116
  const layerId = `rental-vehicles-${id}`;
105
- const [clickedVehicle, setClickedVehicle] = useState(null);
117
+ const [clickedVehicle, setClickedVehicle] = useState<
118
+ RentalVehicle | VehicleRentalStation | undefined
119
+ >();
106
120
 
107
121
  useEffect(() => {
108
122
  // TODO: Make 30s configurable?
@@ -115,52 +129,71 @@ const VehicleRentalOverlay = ({
115
129
  }, [refreshVehicles]);
116
130
 
117
131
  useEffect(() => {
118
- const VEHICLE_LAYERS = [layerId];
119
- VEHICLE_LAYERS.forEach(stopLayer => {
120
- map?.on("mouseenter", stopLayer, () => {
121
- map.getCanvas().style.cursor = "pointer";
122
- });
123
- map?.on("mouseleave", stopLayer, () => {
124
- map.getCanvas().style.cursor = "";
125
- });
126
- map?.on("click", stopLayer, event => {
127
- setClickedVehicle(event.features?.[0].properties);
128
- });
129
- });
130
- map.on("zoom", (e: ViewStateChangeEvent) => {
132
+ const mouseEnterFunc = () => {
133
+ map.getCanvas().style.cursor = "pointer";
134
+ };
135
+ const mouseLeaveFunc = () => {
136
+ map.getCanvas().style.cursor = "";
137
+ };
138
+ const clickFunc = event => {
139
+ const p = event.features?.[0].properties;
140
+ setClickedVehicle({
141
+ ...p,
142
+ // the properties field of the GeoJSON Feature object serializes these
143
+ // two objects into JSON strings, so we need to parse them back into objects
144
+ availableSpaces: JSON.parse(p?.availableSpaces),
145
+ availableVehicles: JSON.parse(p?.availableVehicles)
146
+ } as RentalVehicle | VehicleRentalStation);
147
+ };
148
+ const zoomFunc = e => {
131
149
  // Avoid too many re-renders by only updating state if we are a whole number value different
132
150
  const { zoom: newZoom } = e.viewState;
133
151
  if (Math.floor(zoom / 2) !== Math.floor(newZoom / 2)) {
134
152
  setZoom(newZoom);
135
153
  }
136
- });
154
+ };
155
+
156
+ map?.on("mouseenter", layerId, mouseEnterFunc);
157
+ map?.on("mouseleave", layerId, mouseLeaveFunc);
158
+ map?.on("click", layerId, clickFunc);
159
+ map?.on("zoom", zoomFunc);
160
+
161
+ return () => {
162
+ map?.off("mouseenter", layerId, mouseEnterFunc);
163
+ map?.off("mouseleave", layerId, mouseLeaveFunc);
164
+ map?.off("click", layerId, clickFunc);
165
+ map?.off("zoom", zoomFunc);
166
+ };
137
167
  }, [map]);
138
168
 
139
169
  // Don't render if no map or no stops are defined.
140
- if (visible === false || !stations || stations.length === 0) {
170
+ if (visible === false || !entities || entities.length === 0) {
141
171
  // Null can't be returned here -- react-map-gl dislikes null values as children
142
172
  return <></>;
143
173
  }
144
174
 
145
- const vehiclesGeoJSON: GeoJSON.FeatureCollection = {
175
+ const vehiclesGeoJSON: GeoJSON.FeatureCollection<
176
+ Geometry,
177
+ VehicleRentalStation | RentalVehicle
178
+ > = {
146
179
  type: "FeatureCollection",
147
- features: stations
180
+ features: entities
181
+ .filter(entity => !!entity)
148
182
  .filter(
149
- vehicle =>
183
+ entity =>
150
184
  // Include specified companies only if companies is specified and network info is available
151
185
  !companies ||
152
- !vehicle.networks ||
153
- companies.includes(vehicle.networks[0])
186
+ !entity.rentalNetwork.networkId ||
187
+ companies.includes(entity.rentalNetwork.networkId)
154
188
  )
155
- .map(vehicle => ({
189
+ .map(entity => ({
156
190
  type: "Feature",
157
- geometry: { type: "Point", coordinates: [vehicle.x, vehicle.y] },
191
+ geometry: { type: "Point", coordinates: [entity.lon, entity.lat] },
158
192
  properties: {
159
- ...vehicle,
160
- networks: JSON.stringify(vehicle.networks),
161
- "stroke-width":
162
- vehicle.isFloatingBike || vehicle.isFloatingVehicle ? 1 : 2,
163
- color: getColorForStation(vehicle)
193
+ ...entity,
194
+ networks: entity.rentalNetwork.networkId,
195
+ "stroke-width": entityIsStation(entity) ? 1 : 2,
196
+ color: getColorForEntity(entity)
164
197
  }
165
198
  }))
166
199
  };
@@ -183,47 +216,50 @@ const VehicleRentalOverlay = ({
183
216
  </Source>
184
217
  )}
185
218
  {zoom >= DETAILED_MARKER_CUTOFF &&
186
- stations.map(station => (
187
- <MarkerWithPopup
188
- key={station.id}
189
- popupContents={
190
- <StationPopup
191
- configCompanies={configCompanies}
192
- setLocation={location => {
193
- setClickedVehicle(null);
194
- setLocation(location);
195
- }}
196
- getEntityName={
197
- // @ts-expect-error no stop support. Avoid a breaking change
198
- getStationName && ((s, cc) => getStationName(cc, s))
199
- }
200
- entity={station}
201
- />
202
- }
203
- position={[station.y, station.x]}
204
- >
205
- {station.bikesAvailable !== undefined &&
206
- !station.isFloatingBike &&
207
- !station.isFloatingVehicle &&
208
- station.spacesAvailable !== undefined ? (
209
- <BaseBikeRentalIcon
210
- percent={
211
- station?.bikesAvailable /
212
- (station?.bikesAvailable + station?.spacesAvailable)
213
- }
214
- />
215
- ) : (
216
- <StationMarker width={12} color={getColorForStation(station)} />
217
- )}
218
- </MarkerWithPopup>
219
- ))}
219
+ entities
220
+ .filter(entity => !!entity)
221
+ .map(entity => (
222
+ <MarkerWithPopup
223
+ key={entity.id}
224
+ popupContents={
225
+ <StationPopup
226
+ configCompanies={configCompanies}
227
+ setLocation={location => {
228
+ setClickedVehicle(undefined);
229
+ setLocation(location);
230
+ }}
231
+ getEntityName={
232
+ // @ts-expect-error no stop support. Avoid a breaking change
233
+ getStationName && ((s, cc) => getStationName(cc, s))
234
+ }
235
+ entity={entity}
236
+ />
237
+ }
238
+ position={[entity.lat, entity.lon]}
239
+ >
240
+ {entityIsStation(entity) &&
241
+ entity.availableSpaces !== undefined &&
242
+ entity.availableVehicles !== undefined &&
243
+ entity.availableVehicles.total > 0 ? (
244
+ <BaseBikeRentalIcon
245
+ percent={
246
+ entity?.availableVehicles.total /
247
+ (entity?.availableVehicles.total +
248
+ entity?.availableSpaces.total)
249
+ }
250
+ />
251
+ ) : (
252
+ <StationMarker width={12} color={getColorForEntity(entity)} />
253
+ )}
254
+ </MarkerWithPopup>
255
+ ))}
220
256
  {clickedVehicle && (
221
257
  <Popup
222
- latitude={clickedVehicle.y}
223
- longitude={clickedVehicle.x}
258
+ latitude={clickedVehicle.lat}
259
+ longitude={clickedVehicle.lon}
224
260
  maxWidth="100%"
225
261
  onClose={() => {
226
- setClickedVehicle(null);
262
+ setClickedVehicle(undefined);
227
263
  }}
228
264
  >
229
265
  <StationPopup
@@ -233,13 +269,10 @@ const VehicleRentalOverlay = ({
233
269
  getStationName && ((s, cc) => getStationName(cc, s))
234
270
  }
235
271
  setLocation={location => {
236
- setClickedVehicle(null);
272
+ setClickedVehicle(undefined);
237
273
  setLocation(location);
238
274
  }}
239
- entity={{
240
- ...clickedVehicle,
241
- networks: JSON.parse(clickedVehicle.networks)
242
- }}
275
+ entity={clickedVehicle}
243
276
  />
244
277
  </Popup>
245
278
  )}