@opentripplanner/core-utils 12.0.1 → 12.1.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/esm/__mocks__/fake-route-data.story.json +5425 -0
- package/esm/__mocks__/fake-transit-operators.story.json +97 -0
- package/esm/__tests__/__mocks__/routes.json +234 -105
- package/esm/query-gen.js +1 -3
- package/esm/query-gen.js.map +1 -1
- package/esm/query.js +8 -4
- package/esm/query.js.map +1 -1
- package/esm/route.js +38 -26
- package/esm/route.js.map +1 -1
- package/esm/routes.json +234 -105
- package/esm/time.js +10 -0
- package/esm/time.js.map +1 -1
- package/lib/__mocks__/fake-route-data.story.json +5425 -0
- package/lib/__mocks__/fake-transit-operators.story.json +97 -0
- package/lib/__tests__/__mocks__/routes.json +234 -105
- package/lib/query-gen.js +1 -1
- package/lib/query-gen.js.map +1 -1
- package/lib/query.js +4 -3
- package/lib/query.js.map +1 -1
- package/lib/route.d.ts +35 -1
- package/lib/route.d.ts.map +1 -1
- package/lib/route.js +41 -18
- package/lib/route.js.map +1 -1
- package/lib/time.d.ts +7 -0
- package/lib/time.d.ts.map +1 -1
- package/lib/time.js +12 -1
- package/lib/time.js.map +1 -1
- package/package.json +1 -1
- package/src/__mocks__/fake-route-data.story.json +5425 -0
- package/src/__mocks__/fake-transit-operators.story.json +97 -0
- package/src/__snapshots__/core-utils.story.tsx.snap +8382 -0
- package/src/__tests__/__mocks__/routes.json +234 -105
- package/src/__tests__/__snapshots__/route.js.snap +285 -95
- package/src/__tests__/route.js +27 -55
- package/src/core-utils.story.tsx +200 -1
- package/src/query-gen.ts +1 -1
- package/src/query.js +12 -3
- package/src/route.ts +41 -21
- package/src/time.ts +9 -0
- package/tsconfig.tsbuildinfo +1 -1
package/esm/query.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/query.js"],"names":["format","isMatch","parse","getGeocoder","qs","getTransitModes","hasCar","isAccessMode","coordsToString","stringToCoords","queryParams","getCurrentTime","getCurrentDate","OTP_API_DATE_FORMAT","OTP_API_TIME_FORMAT","defaultParams","TIME_FORMATS","getQueryParamProperty","paramInfo","property","query","ensureSingleAccessMode","queryModes","accessCount","filter","m","length","firstAccess","find","push","getUrlParams","window","location","href","split","getOtpUrlParams","Object","keys","key","startsWith","getTripOptionsFromQuery","keepPlace","options","time","departArrive","date","from","to","getDefaultQueryParamValue","param","getDefaultQuery","config","defaultQuery","routingType","qp","forEach","name","routingTypes","defaultQueryParams","isParamApplicable","applicable","includes","reduceOtpFlexModes","modes","enabled","reduce","prev","cur","newModes","expandOtpFlexMode","mode","map","join","isNotDefaultQuery","activeModes","sort","mergeFlex","defaultModes","concat","modesEqual","every","value","index","i","getFirstGeocodeResult","text","geocoderConfig","geocoder","search","then","result","firstResult","features","getLocationFromGeocodedFeature","parseLocationString","parts","coordinates","lat","lon","queryParamToLocation","planParamsToQuery","params","fromPlace","toPlace","arriveBy","matchedTimeFormat","timeFormat","Date","intermediatePlaces","maybeNumber","Number","isNaN","planParamsToQueryAsync","getRoutingParams","currentQuery","ignoreRealtimeUpdates","isItinerary","indexOf","rewriteFunction","itineraryRewrite","profileRewrite","assign","dateValid","timeValid","minTransitDistance","searchTimeout","onlyTransitTrips","startTimeValid","startTime","endTimeValid","endTime","reducedMode"],"mappings":"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","sourcesContent":["import { format, isMatch, parse } from \"date-fns\";\nimport getGeocoder from \"@opentripplanner/geocoder/lib\";\nimport qs from \"qs\";\n\nimport { getTransitModes, hasCar, isAccessMode } from \"./itinerary\";\nimport { coordsToString, stringToCoords } from \"./map\";\nimport queryParams from \"./query-params\";\nimport {\n getCurrentTime,\n getCurrentDate,\n OTP_API_DATE_FORMAT,\n OTP_API_TIME_FORMAT\n} from \"./time\";\n\n/* The list of default parameters considered in the settings panel */\n\nexport const defaultParams = [\n \"wheelchair\",\n \"maxWalkDistance\",\n \"walkReluctance\",\n \"maxWalkTime\",\n \"maxBikeDistance\",\n \"maxBikeTime\",\n \"bikeSpeed\",\n \"optimize\",\n \"optimizeBike\",\n \"maxEScooterDistance\",\n \"watts\"\n];\n\n/**\n * List of time formats to parse when reading query params.\n */\nconst TIME_FORMATS = [\n \"HH:mm:ss\",\n \"HH:mm\",\n \"H:mm\",\n \"h:mm:ss a\",\n \"h:mm:ssa\",\n \"h:mm a\",\n \"h:mma\",\n \"h:mm\",\n \"HHmm\",\n \"hmm\",\n \"ha\"\n];\n\n/* A function to retrieve a property value from an entry in the query-params\n * table, checking for either a static value or a function */\n\nexport function getQueryParamProperty(paramInfo, property, query) {\n return typeof paramInfo[property] === \"function\"\n ? paramInfo[property](query)\n : paramInfo[property];\n}\n\nexport function ensureSingleAccessMode(queryModes) {\n // Count the number of access modes\n const accessCount = queryModes.filter(m => isAccessMode(m)).length;\n\n // If multiple access modes are specified, keep only the first one\n if (accessCount > 1) {\n const firstAccess = queryModes.find(m => isAccessMode(m));\n queryModes = queryModes.filter(m => !isAccessMode(m) || m === firstAccess);\n\n // If no access modes are specified, add 'WALK' as the default\n } else if (accessCount === 0) {\n queryModes.push(\"WALK\");\n }\n\n return queryModes;\n}\n\nexport function getUrlParams() {\n return qs.parse(window.location.href.split(\"?\")[1]);\n}\n\nexport function getOtpUrlParams() {\n return Object.keys(getUrlParams()).filter(key => !key.startsWith(\"ui_\"));\n}\n\nexport function getTripOptionsFromQuery(query, keepPlace = false) {\n const options = { ...query };\n // Delete time/date options and from/to\n delete options.time;\n delete options.departArrive;\n delete options.date;\n if (!keepPlace) {\n delete options.from;\n delete options.to;\n }\n return options;\n}\n\n/**\n * Gets the query param's default value that is either a constant or by\n * executing the default value function.\n */\nfunction getDefaultQueryParamValue(param) {\n return typeof param.default === \"function\" ? param.default() : param.default;\n}\n\n/**\n * Get the default query to OTP based on the given config.\n */\nexport function getDefaultQuery(config = null) {\n const defaultQuery = { routingType: \"ITINERARY\" };\n queryParams\n .filter(qp => \"default\" in qp)\n .forEach(qp => {\n defaultQuery[qp.name] = getDefaultQueryParamValue(qp);\n });\n if (config) {\n if (config.routingTypes && config.routingTypes.length > 0) {\n defaultQuery.routingType = config.routingTypes[0].key;\n }\n if (config.defaultQueryParams) {\n Object.keys(config.defaultQueryParams).forEach(key => {\n defaultQuery[key] = config.defaultQueryParams[key];\n });\n }\n }\n return defaultQuery;\n}\n\n/**\n * Determine if the specified query param applies to the given query (based on\n * routing type and the param's own applicable function).\n * @param paramInfo an entry from query-params.js\n * @param query the query against which to check if the param applies\n * @param config OTP config\n * @return {Boolean}\n */\nfunction isParamApplicable(paramInfo, query, config) {\n const { applicable, routingTypes } = paramInfo;\n if (!routingTypes.includes(query.routingType)) return false;\n if (typeof applicable === \"function\" && !applicable(query, config)) {\n return false;\n }\n return true;\n}\n\n/**\n * Helper method which replaces OTP flex modes with single FLEX mode that's\n * more useful and easier to work with.\n */\nexport function reduceOtpFlexModes(modes, enabled = true) {\n if (!enabled) return modes;\n\n return modes.reduce((prev, cur) => {\n const newModes = prev;\n // Add the current mode if it is not a flex mode\n if (!cur.includes(\"FLEX\")) {\n newModes.push(cur);\n // If it is a flex mode, do not add it but rather add the custom flex mode\n // if not already present\n } else if (!newModes.includes(\"FLEX\")) {\n newModes.push(\"FLEX\");\n }\n return newModes;\n }, []);\n}\n\n/**\n * Helper method to process a mode string, replacing all instances of FLEX\n * with the full set of FLEX modes used by otp-2\n * @param {*} mode a mode String, not an array\n * @returns a mode String, not an array (with flex modes expanded)\n */\nexport function expandOtpFlexMode(mode) {\n const modes = reduceOtpFlexModes(mode.split(\",\"));\n return modes\n .map(m => {\n // If both the expanded and shrunk modes are included, remove the expanded one\n if (m === \"FLEX_EGRESS\" || m === \"FLEX_ACCESS\" || m === \"FLEX_DIRECT\") {\n if (mode.includes(\"FLEX\")) return \"\";\n }\n if (m === \"FLEX\") {\n return \"FLEX_EGRESS,FLEX_ACCESS,FLEX_DIRECT\";\n }\n return m;\n })\n .join(\",\");\n}\n\n/**\n * Determines whether the specified query differs from the default query, i.e.,\n * whether the user has modified any trip options (including mode) from their\n * default values.\n */\nexport function isNotDefaultQuery(query, config) {\n const activeModes = reduceOtpFlexModes(\n query.mode.split(\",\").sort(),\n config.modes?.mergeFlex\n );\n if (\n activeModes.length !== 2 ||\n activeModes[0] !== \"TRANSIT\" ||\n activeModes[1] !== \"WALK\"\n ) {\n // Default mode is TRANSIT,WALK. If general TRANSIT is not used, check\n // against available transit modes in config.\n const defaultModes = getTransitModes(config)\n .concat([\"WALK\"])\n .sort();\n const modesEqual =\n activeModes.length === defaultModes.length &&\n activeModes.every((value, index) => {\n return value === defaultModes[index];\n });\n if (!modesEqual) return true;\n }\n // If modes are equal, check the remaining params.\n const defaultQuery = getDefaultQuery(config);\n for (let i = 0; i < defaultParams.length; i++) {\n const param = defaultParams[i];\n const paramInfo = queryParams.find(qp => qp.name === param);\n // If the parameter applies to the query and does not match the default\n // value, the query is not default.\n if (\n isParamApplicable(paramInfo, query, config) &&\n query[param] !== defaultQuery[param]\n ) {\n return true;\n }\n }\n return false;\n}\n\n/**\n * Geocode utility for returning the first result for the provided place name text.\n * @param {string} text - text to search\n * @param {Object} geocoderConfig\n * @return {Location}\n */\nasync function getFirstGeocodeResult(text, geocoderConfig) {\n const geocoder = getGeocoder(geocoderConfig);\n // Attempt to geocode search text and return first result if found.\n // TODO: Import geocoder from @opentripplanner\n return geocoder.search({ text }).then(result => {\n const firstResult = result.features && result.features[0];\n if (firstResult) {\n return geocoder.getLocationFromGeocodedFeature(firstResult);\n }\n return null;\n });\n}\n\n/**\n * OTP allows passing a location in the form '123 Main St::lat,lon', so we check\n * for the double colon and parse the coordinates accordingly.\n * @param {string} value - query param for place described above\n * @return {Location} - location or null if the value is falsey or the parsed\n * coordinates do not result in both a lat and lon\n */\nexport function parseLocationString(value) {\n if (!value) return null;\n const parts = value.split(\"::\");\n const coordinates = parts[1]\n ? stringToCoords(parts[1])\n : stringToCoords(parts[0]);\n const name = parts[1] ? parts[0] : coordsToString(coordinates);\n return coordinates.length === 2\n ? {\n name: name || null,\n lat: coordinates[0] || null,\n lon: coordinates[1] || null\n }\n : null;\n}\n\n/**\n * Convert a string query param for a from or to place into a location. If\n * coordinates not provided and geocoder config is present, use the first\n * geocoded result.\n * @param {string} value\n * @param {Object} [geocoderConfig=null]\n * @return {Location}\n */\nasync function queryParamToLocation(value, geocoderConfig) {\n let location = parseLocationString(value);\n if (!location && value && geocoderConfig) {\n // If a valid location was not found, but the place name text exists,\n // attempt to geocode the name.\n location = await getFirstGeocodeResult(value, geocoderConfig);\n }\n return location;\n}\n\n/**\n * Create a otp query based on a the url params.\n *\n * @param {Object} params An object representing the parsed querystring of url\n * params.\n */\nexport function planParamsToQuery(params) {\n const query = {};\n Object.keys(params).forEach(key => {\n switch (key) {\n case \"fromPlace\":\n query.from = parseLocationString(params.fromPlace);\n break;\n case \"toPlace\":\n query.to = parseLocationString(params.toPlace);\n break;\n case \"arriveBy\":\n query.departArrive =\n params.arriveBy === \"true\"\n ? \"ARRIVE\"\n : params.arriveBy === \"false\"\n ? \"DEPART\"\n : \"NOW\";\n break;\n case \"date\":\n query.date = params.date || getCurrentDate();\n break;\n case \"time\":\n {\n // Match one of the supported time formats\n const matchedTimeFormat = TIME_FORMATS.find(timeFormat =>\n isMatch(params.time, timeFormat)\n );\n query.time = matchedTimeFormat\n ? format(\n parse(params.time, matchedTimeFormat, new Date()),\n OTP_API_TIME_FORMAT\n )\n : getCurrentTime();\n }\n break;\n case \"intermediatePlaces\":\n // If query has intermediate places, ensure that they are parsed\n // as locations.\n query.intermediatePlaces = params.intermediatePlaces\n ? params.intermediatePlaces.map(parseLocationString)\n : [];\n break;\n default: {\n const maybeNumber = Number(params[key]);\n // If the param value is an empty string literal and is not a number,\n // use string value. Else, use parsed number value.\n // See https://github.com/opentripplanner/otp-ui/issues/50\n query[key] =\n params[key] === \"\" || Number.isNaN(maybeNumber)\n ? params[key]\n : maybeNumber;\n break;\n }\n }\n });\n return query;\n}\n\n/**\n * Async method to create a otp query based on a the url params. This provides\n * the same functionality as planParamsToQuery, except that it will also attempt\n * to geocode the input from and to strings if no lat/lng values were provided.\n *\n * @param {Object} params An object representing the parsed querystring of url\n * params.\n * @param config the config in the otp-rr store.\n */\nexport async function planParamsToQueryAsync(params, config = {}) {\n // Construct query from plan params.\n const query = planParamsToQuery(params);\n // Attempt to geocode from and to params if the string parsing does not return\n // valid locations.\n if (!query.from) {\n query.from = await queryParamToLocation(params.fromPlace, config.geocoder);\n }\n if (!query.to) {\n query.to = await queryParamToLocation(params.toPlace, config.geocoder);\n }\n return query;\n}\n\n/**\n * Create an object that can be used as a querystring in making an OTP\n * PlannerResource request.\n *\n * See http://otp-docs.ibi-transit.com/api/resource_PlannerResource.html\n *\n * @param {Object} config The OTP application config. See types#configType\n * @param {Object} currentQuery The current query parameters as saved in the\n * application state. This method does some extra logic on top of this data\n * in order to create a request suitable for OTP. See __tests__/query.js#L14 for more.\n * @param {boolean} ignoreRealtimeUpdates If true, will create a request that\n * does not use realtime data.\n */\nexport function getRoutingParams(config, currentQuery, ignoreRealtimeUpdates) {\n const routingType = currentQuery.routingType;\n const isItinerary = routingType === \"ITINERARY\";\n let params = {};\n\n // Start with the universe of OTP parameters defined in query-params.js:\n queryParams\n .filter(qp => {\n // A given parameter is included in the request if all of the following:\n // 1. Must apply to the active routing type (ITINERARY or PROFILE)\n // 2. Must be included in the current user-defined query\n // 3. Must pass the parameter's applicability test, if one is specified\n return (\n qp.routingTypes.indexOf(routingType) !== -1 &&\n qp.name in currentQuery &&\n (typeof qp.applicable !== \"function\" ||\n qp.applicable(currentQuery, config))\n );\n })\n .forEach(qp => {\n // Translate the applicable parameters according to their rewrite\n // functions (if provided)\n const rewriteFunction = isItinerary\n ? qp.itineraryRewrite\n : qp.profileRewrite;\n params = Object.assign(\n params,\n rewriteFunction\n ? rewriteFunction(currentQuery[qp.name])\n : { [qp.name]: currentQuery[qp.name] }\n );\n });\n\n // Additional processing specific to ITINERARY mode\n if (isItinerary) {\n // override ignoreRealtimeUpdates if provided\n if (typeof ignoreRealtimeUpdates === \"boolean\") {\n params.ignoreRealtimeUpdates = ignoreRealtimeUpdates;\n }\n\n // check date/time validity; ignore both if either is invalid\n const dateValid = isMatch(params.date, OTP_API_DATE_FORMAT);\n const timeValid = isMatch(params.time, OTP_API_TIME_FORMAT);\n\n if (!dateValid || !timeValid) {\n delete params.time;\n delete params.date;\n }\n\n // temp: set additional parameters for CAR_HAIL or CAR_RENT trips\n if (\n params.mode &&\n (params.mode.includes(\"CAR_HAIL\") || params.mode.includes(\"CAR_RENT\"))\n ) {\n params.minTransitDistance = \"50%\";\n // increase search timeout because these queries can take a while\n params.searchTimeout = 10000;\n }\n\n // set onlyTransitTrips for car rental searches\n if (params.mode && params.mode.includes(\"CAR_RENT\")) {\n params.onlyTransitTrips = true;\n }\n\n // Additional processing specific to PROFILE mode\n } else {\n // check start and end time validity; ignore both if either is invalid\n const startTimeValid = isMatch(params.startTime, OTP_API_TIME_FORMAT);\n const endTimeValid = isMatch(params.endTime, OTP_API_TIME_FORMAT);\n\n if (!startTimeValid || !endTimeValid) {\n delete params.startTimeValid;\n delete params.endTimeValid;\n }\n }\n\n // TODO: check that valid from/to locations are provided\n\n // hack to add walking to driving/TNC trips\n if (hasCar(params.mode)) {\n params.mode += \",WALK\";\n }\n\n // Replace FLEX placeholder with OTP flex modes\n // Explicit false check allows avoiding a breaking change -- undefined is true\n if (params.mode && config.modes?.mergeFlex !== false) {\n // Ensure query is in reduced format to avoid replacing twice\n const reducedMode = reduceOtpFlexModes(params.mode.split(\",\")).join(\",\");\n params.mode = expandOtpFlexMode(reducedMode);\n }\n\n return params;\n}\n"],"file":"query.js"}
|
|
1
|
+
{"version":3,"sources":["../src/query.js"],"names":["format","isMatch","parse","getGeocoder","qs","getTransitModes","hasCar","isAccessMode","coordsToString","stringToCoords","queryParams","getCurrentTime","getCurrentDate","OTP_API_DATE_FORMAT","OTP_API_TIME_FORMAT","defaultParams","TIME_FORMATS","getQueryParamProperty","paramInfo","property","query","ensureSingleAccessMode","queryModes","accessCount","filter","m","length","firstAccess","find","push","getUrlParams","window","location","href","split","getOtpUrlParams","Object","keys","key","startsWith","getTripOptionsFromQuery","keepPlace","options","time","departArrive","date","from","to","getDefaultQueryParamValue","param","getDefaultQuery","config","defaultQuery","routingType","qp","forEach","name","routingTypes","defaultQueryParams","isParamApplicable","applicable","includes","reduceOtpFlexModes","modes","enabled","reduce","prev","cur","newModes","expandOtpFlexMode","mode","map","join","isNotDefaultQuery","activeModes","sort","mergeFlex","defaultModes","concat","modesEqual","every","value","index","i","getFirstGeocodeResult","text","geocoderConfig","geocoderMethod","geocoder","then","result","firstResult","features","getLocationFromGeocodedFeature","parseLocationString","parts","coordinates","lat","lon","queryParamToLocation","autoMethod","planParamsToQuery","params","fromPlace","toPlace","arriveBy","matchedTimeFormat","timeFormat","Date","intermediatePlaces","maybeNumber","Number","isNaN","planParamsToQueryAsync","getRoutingParams","currentQuery","ignoreRealtimeUpdates","isItinerary","indexOf","rewriteFunction","itineraryRewrite","profileRewrite","assign","dateValid","timeValid","minTransitDistance","searchTimeout","onlyTransitTrips","startTimeValid","startTime","endTimeValid","endTime","reducedMode"],"mappings":"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","sourcesContent":["import { format, isMatch, parse } from \"date-fns\";\nimport getGeocoder from \"@opentripplanner/geocoder/lib\";\nimport qs from \"qs\";\n\nimport { getTransitModes, hasCar, isAccessMode } from \"./itinerary\";\nimport { coordsToString, stringToCoords } from \"./map\";\nimport queryParams from \"./query-params\";\nimport {\n getCurrentTime,\n getCurrentDate,\n OTP_API_DATE_FORMAT,\n OTP_API_TIME_FORMAT\n} from \"./time\";\n\n/* The list of default parameters considered in the settings panel */\n\nexport const defaultParams = [\n \"wheelchair\",\n \"maxWalkDistance\",\n \"walkReluctance\",\n \"maxWalkTime\",\n \"maxBikeDistance\",\n \"maxBikeTime\",\n \"bikeSpeed\",\n \"optimize\",\n \"optimizeBike\",\n \"maxEScooterDistance\",\n \"watts\"\n];\n\n/**\n * List of time formats to parse when reading query params.\n */\nconst TIME_FORMATS = [\n \"HH:mm:ss\",\n \"HH:mm\",\n \"H:mm\",\n \"h:mm:ss a\",\n \"h:mm:ssa\",\n \"h:mm a\",\n \"h:mma\",\n \"h:mm\",\n \"HHmm\",\n \"hmm\",\n \"ha\"\n];\n\n/* A function to retrieve a property value from an entry in the query-params\n * table, checking for either a static value or a function */\n\nexport function getQueryParamProperty(paramInfo, property, query) {\n return typeof paramInfo[property] === \"function\"\n ? paramInfo[property](query)\n : paramInfo[property];\n}\n\nexport function ensureSingleAccessMode(queryModes) {\n // Count the number of access modes\n const accessCount = queryModes.filter(m => isAccessMode(m)).length;\n\n // If multiple access modes are specified, keep only the first one\n if (accessCount > 1) {\n const firstAccess = queryModes.find(m => isAccessMode(m));\n queryModes = queryModes.filter(m => !isAccessMode(m) || m === firstAccess);\n\n // If no access modes are specified, add 'WALK' as the default\n } else if (accessCount === 0) {\n queryModes.push(\"WALK\");\n }\n\n return queryModes;\n}\n\nexport function getUrlParams() {\n return qs.parse(window.location.href.split(\"?\")[1]);\n}\n\nexport function getOtpUrlParams() {\n return Object.keys(getUrlParams()).filter(key => !key.startsWith(\"ui_\"));\n}\n\nexport function getTripOptionsFromQuery(query, keepPlace = false) {\n const options = { ...query };\n // Delete time/date options and from/to\n delete options.time;\n delete options.departArrive;\n delete options.date;\n if (!keepPlace) {\n delete options.from;\n delete options.to;\n }\n return options;\n}\n\n/**\n * Gets the query param's default value that is either a constant or by\n * executing the default value function.\n */\nfunction getDefaultQueryParamValue(param) {\n return typeof param.default === \"function\" ? param.default() : param.default;\n}\n\n/**\n * Get the default query to OTP based on the given config.\n */\nexport function getDefaultQuery(config = null) {\n const defaultQuery = { routingType: \"ITINERARY\" };\n queryParams\n .filter(qp => \"default\" in qp)\n .forEach(qp => {\n defaultQuery[qp.name] = getDefaultQueryParamValue(qp);\n });\n if (config) {\n if (config.routingTypes && config.routingTypes.length > 0) {\n defaultQuery.routingType = config.routingTypes[0].key;\n }\n if (config.defaultQueryParams) {\n Object.keys(config.defaultQueryParams).forEach(key => {\n defaultQuery[key] = config.defaultQueryParams[key];\n });\n }\n }\n return defaultQuery;\n}\n\n/**\n * Determine if the specified query param applies to the given query (based on\n * routing type and the param's own applicable function).\n * @param paramInfo an entry from query-params.js\n * @param query the query against which to check if the param applies\n * @param config OTP config\n * @return {Boolean}\n */\nfunction isParamApplicable(paramInfo, query, config) {\n const { applicable, routingTypes } = paramInfo;\n if (!routingTypes.includes(query.routingType)) return false;\n if (typeof applicable === \"function\" && !applicable(query, config)) {\n return false;\n }\n return true;\n}\n\n/**\n * Helper method which replaces OTP flex modes with single FLEX mode that's\n * more useful and easier to work with.\n */\nexport function reduceOtpFlexModes(modes, enabled = true) {\n if (!enabled) return modes;\n\n return modes.reduce((prev, cur) => {\n const newModes = prev;\n // Add the current mode if it is not a flex mode\n if (!cur.includes(\"FLEX\")) {\n newModes.push(cur);\n // If it is a flex mode, do not add it but rather add the custom flex mode\n // if not already present\n } else if (!newModes.includes(\"FLEX\")) {\n newModes.push(\"FLEX\");\n }\n return newModes;\n }, []);\n}\n\n/**\n * Helper method to process a mode string, replacing all instances of FLEX\n * with the full set of FLEX modes used by otp-2\n * @param {*} mode a mode String, not an array\n * @returns a mode String, not an array (with flex modes expanded)\n */\nexport function expandOtpFlexMode(mode) {\n const modes = reduceOtpFlexModes(mode.split(\",\"));\n return modes\n .map(m => {\n // If both the expanded and shrunk modes are included, remove the expanded one\n if (m === \"FLEX_EGRESS\" || m === \"FLEX_ACCESS\" || m === \"FLEX_DIRECT\") {\n if (mode.includes(\"FLEX\")) return \"\";\n }\n if (m === \"FLEX\") {\n return \"FLEX_EGRESS,FLEX_ACCESS,FLEX_DIRECT\";\n }\n return m;\n })\n .join(\",\");\n}\n\n/**\n * Determines whether the specified query differs from the default query, i.e.,\n * whether the user has modified any trip options (including mode) from their\n * default values.\n */\nexport function isNotDefaultQuery(query, config) {\n const activeModes = reduceOtpFlexModes(\n query.mode.split(\",\").sort(),\n config.modes?.mergeFlex\n );\n if (\n activeModes.length !== 2 ||\n activeModes[0] !== \"TRANSIT\" ||\n activeModes[1] !== \"WALK\"\n ) {\n // Default mode is TRANSIT,WALK. If general TRANSIT is not used, check\n // against available transit modes in config.\n const defaultModes = getTransitModes(config)\n .concat([\"WALK\"])\n .sort();\n const modesEqual =\n activeModes.length === defaultModes.length &&\n activeModes.every((value, index) => {\n return value === defaultModes[index];\n });\n if (!modesEqual) return true;\n }\n // If modes are equal, check the remaining params.\n const defaultQuery = getDefaultQuery(config);\n for (let i = 0; i < defaultParams.length; i++) {\n const param = defaultParams[i];\n const paramInfo = queryParams.find(qp => qp.name === param);\n // If the parameter applies to the query and does not match the default\n // value, the query is not default.\n if (\n isParamApplicable(paramInfo, query, config) &&\n query[param] !== defaultQuery[param]\n ) {\n return true;\n }\n }\n return false;\n}\n\n/**\n * Geocode utility for returning the first result for the provided place name text.\n * @param {string} text - text to search\n * @param {Object} geocoderConfig\n * @param {string} geocoderMethod - an alternate geocoder api method to use. Defaults to `search`\n * @return {Location}\n */\nasync function getFirstGeocodeResult(\n text,\n geocoderConfig,\n geocoderMethod = \"search\"\n) {\n const geocoder = getGeocoder(geocoderConfig);\n // Attempt to geocode search text and return first result if found.\n // TODO: Import geocoder from @opentripplanner\n return geocoder[geocoderMethod]({ text }).then(result => {\n const firstResult = result.features && result.features[0];\n if (firstResult) {\n return geocoder.getLocationFromGeocodedFeature(firstResult);\n }\n return null;\n });\n}\n\n/**\n * OTP allows passing a location in the form '123 Main St::lat,lon', so we check\n * for the double colon and parse the coordinates accordingly.\n * @param {string} value - query param for place described above\n * @return {Location} - location or null if the value is falsey or the parsed\n * coordinates do not result in both a lat and lon\n */\nexport function parseLocationString(value) {\n if (!value) return null;\n const parts = value.split(\"::\");\n const coordinates = parts[1]\n ? stringToCoords(parts[1])\n : stringToCoords(parts[0]);\n const name = parts[1] ? parts[0] : coordsToString(coordinates);\n return coordinates.length === 2\n ? {\n name: name || null,\n lat: coordinates[0] || null,\n lon: coordinates[1] || null\n }\n : null;\n}\n\n/**\n * Convert a string query param for a from or to place into a location. If\n * coordinates not provided and geocoder config is present, use the first\n * geocoded result.\n * @param {string} value\n * @param {Object} [geocoderConfig=null]\n * @return {Location}\n */\nasync function queryParamToLocation(value, geocoderConfig) {\n let location = parseLocationString(value);\n if (!location && value && geocoderConfig) {\n // If a valid location was not found, but the place name text exists,\n // attempt to geocode the name.\n location = await getFirstGeocodeResult(\n value,\n geocoderConfig,\n geocoderConfig.autoMethod\n );\n }\n return location;\n}\n\n/**\n * Create a otp query based on a the url params.\n *\n * @param {Object} params An object representing the parsed querystring of url\n * params.\n */\nexport function planParamsToQuery(params) {\n const query = {};\n Object.keys(params).forEach(key => {\n switch (key) {\n case \"fromPlace\":\n query.from = parseLocationString(params.fromPlace);\n break;\n case \"toPlace\":\n query.to = parseLocationString(params.toPlace);\n break;\n case \"arriveBy\":\n query.departArrive =\n params.arriveBy === \"true\"\n ? \"ARRIVE\"\n : params.arriveBy === \"false\"\n ? \"DEPART\"\n : \"NOW\";\n break;\n case \"date\":\n query.date = params.date || getCurrentDate();\n break;\n case \"time\":\n {\n // Match one of the supported time formats\n const matchedTimeFormat = TIME_FORMATS.find(timeFormat =>\n isMatch(params.time, timeFormat)\n );\n query.time = matchedTimeFormat\n ? format(\n parse(params.time, matchedTimeFormat, new Date()),\n OTP_API_TIME_FORMAT\n )\n : getCurrentTime();\n }\n break;\n case \"intermediatePlaces\":\n // If query has intermediate places, ensure that they are parsed\n // as locations.\n query.intermediatePlaces = params.intermediatePlaces\n ? params.intermediatePlaces.map(parseLocationString)\n : [];\n break;\n default: {\n const maybeNumber = Number(params[key]);\n // If the param value is an empty string literal and is not a number,\n // use string value. Else, use parsed number value.\n // See https://github.com/opentripplanner/otp-ui/issues/50\n query[key] =\n params[key] === \"\" || Number.isNaN(maybeNumber)\n ? params[key]\n : maybeNumber;\n break;\n }\n }\n });\n return query;\n}\n\n/**\n * Async method to create a otp query based on a the url params. This provides\n * the same functionality as planParamsToQuery, except that it will also attempt\n * to geocode the input from and to strings if no lat/lng values were provided.\n *\n * @param {Object} params An object representing the parsed querystring of url\n * params.\n * @param config the config in the otp-rr store.\n */\nexport async function planParamsToQueryAsync(params, config = {}) {\n // Construct query from plan params.\n const query = planParamsToQuery(params);\n // Attempt to geocode from and to params if the string parsing does not return\n // valid locations.\n if (!query.from) {\n query.from = await queryParamToLocation(params.fromPlace, config.geocoder);\n }\n if (!query.to) {\n query.to = await queryParamToLocation(params.toPlace, config.geocoder);\n }\n return query;\n}\n\n/**\n * Create an object that can be used as a querystring in making an OTP\n * PlannerResource request.\n *\n * See http://otp-docs.ibi-transit.com/api/resource_PlannerResource.html\n *\n * @param {Object} config The OTP application config. See types#configType\n * @param {Object} currentQuery The current query parameters as saved in the\n * application state. This method does some extra logic on top of this data\n * in order to create a request suitable for OTP. See __tests__/query.js#L14 for more.\n * @param {boolean} ignoreRealtimeUpdates If true, will create a request that\n * does not use realtime data.\n */\nexport function getRoutingParams(config, currentQuery, ignoreRealtimeUpdates) {\n const routingType = currentQuery.routingType;\n const isItinerary = routingType === \"ITINERARY\";\n let params = {};\n\n // Start with the universe of OTP parameters defined in query-params.js:\n queryParams\n .filter(qp => {\n // A given parameter is included in the request if all of the following:\n // 1. Must apply to the active routing type (ITINERARY or PROFILE)\n // 2. Must be included in the current user-defined query\n // 3. Must pass the parameter's applicability test, if one is specified\n return (\n qp.routingTypes.indexOf(routingType) !== -1 &&\n qp.name in currentQuery &&\n (typeof qp.applicable !== \"function\" ||\n qp.applicable(currentQuery, config))\n );\n })\n .forEach(qp => {\n // Translate the applicable parameters according to their rewrite\n // functions (if provided)\n const rewriteFunction = isItinerary\n ? qp.itineraryRewrite\n : qp.profileRewrite;\n params = Object.assign(\n params,\n rewriteFunction\n ? rewriteFunction(currentQuery[qp.name])\n : { [qp.name]: currentQuery[qp.name] }\n );\n });\n\n // Additional processing specific to ITINERARY mode\n if (isItinerary) {\n // override ignoreRealtimeUpdates if provided\n if (typeof ignoreRealtimeUpdates === \"boolean\") {\n params.ignoreRealtimeUpdates = ignoreRealtimeUpdates;\n }\n\n // check date/time validity; ignore both if either is invalid\n const dateValid = isMatch(params.date, OTP_API_DATE_FORMAT);\n const timeValid = isMatch(params.time, OTP_API_TIME_FORMAT);\n\n if (!dateValid || !timeValid) {\n delete params.time;\n delete params.date;\n }\n\n // temp: set additional parameters for CAR_HAIL or CAR_RENT trips\n if (\n params.mode &&\n (params.mode.includes(\"CAR_HAIL\") || params.mode.includes(\"CAR_RENT\"))\n ) {\n params.minTransitDistance = \"50%\";\n // increase search timeout because these queries can take a while\n params.searchTimeout = 10000;\n }\n\n // set onlyTransitTrips for car rental searches\n if (params.mode && params.mode.includes(\"CAR_RENT\")) {\n params.onlyTransitTrips = true;\n }\n\n // Additional processing specific to PROFILE mode\n } else {\n // check start and end time validity; ignore both if either is invalid\n const startTimeValid = isMatch(params.startTime, OTP_API_TIME_FORMAT);\n const endTimeValid = isMatch(params.endTime, OTP_API_TIME_FORMAT);\n\n if (!startTimeValid || !endTimeValid) {\n delete params.startTimeValid;\n delete params.endTimeValid;\n }\n }\n\n // TODO: check that valid from/to locations are provided\n\n // hack to add walking to driving/TNC trips\n if (hasCar(params.mode)) {\n params.mode += \",WALK\";\n }\n\n // Replace FLEX placeholder with OTP flex modes\n // Explicit false check allows avoiding a breaking change -- undefined is true\n if (params.mode && config.modes?.mergeFlex !== false) {\n // Ensure query is in reduced format to avoid replacing twice\n const reducedMode = reduceOtpFlexModes(params.mode.split(\",\")).join(\",\");\n params.mode = expandOtpFlexMode(reducedMode);\n }\n\n return params;\n}\n"],"file":"query.js"}
|
package/esm/route.js
CHANGED
|
@@ -48,10 +48,10 @@ export function getTransitOperatorFromOtpRoute(route, transitOperators) {
|
|
|
48
48
|
var agencyId;
|
|
49
49
|
|
|
50
50
|
if (route.agency) {
|
|
51
|
-
// This is returned in
|
|
51
|
+
// This is returned in OTP2
|
|
52
52
|
agencyId = route.agency.id;
|
|
53
53
|
} else if (route.agencyId) {
|
|
54
|
-
// This is returned in
|
|
54
|
+
// This is returned in OTP1
|
|
55
55
|
agencyId = route.agencyId;
|
|
56
56
|
} else {
|
|
57
57
|
return null;
|
|
@@ -88,8 +88,8 @@ function getTransitOperatorComparatorValue(route, transitOperators) {
|
|
|
88
88
|
// if the transitOperators is undefined or has zero length, use the route's
|
|
89
89
|
// agency name as the comparator value
|
|
90
90
|
if (!transitOperators || transitOperators.length === 0) {
|
|
91
|
-
//
|
|
92
|
-
if (route.agency) return route.agency.name; //
|
|
91
|
+
// OTP2 Route
|
|
92
|
+
if (route.agency) return route.agency.name; // OTP1 Route
|
|
93
93
|
|
|
94
94
|
if (route.agencyName) return route.agencyName; // shouldn't happen as agency names will be defined
|
|
95
95
|
|
|
@@ -110,7 +110,7 @@ function getTransitOperatorComparatorValue(route, transitOperators) {
|
|
|
110
110
|
*/
|
|
111
111
|
|
|
112
112
|
|
|
113
|
-
function makeTransitOperatorComparator(transitOperators) {
|
|
113
|
+
export function makeTransitOperatorComparator(transitOperators) {
|
|
114
114
|
return function (a, b) {
|
|
115
115
|
var aVal = getTransitOperatorComparatorValue(a, transitOperators);
|
|
116
116
|
var bVal = getTransitOperatorComparatorValue(b, transitOperators);
|
|
@@ -133,7 +133,6 @@ function makeTransitOperatorComparator(transitOperators) {
|
|
|
133
133
|
* getter function is not defined, the original sort values are returned.
|
|
134
134
|
*/
|
|
135
135
|
|
|
136
|
-
|
|
137
136
|
function getSortValues(getterFn, a, b) {
|
|
138
137
|
var aVal;
|
|
139
138
|
var bVal;
|
|
@@ -223,7 +222,7 @@ function getRouteTypeComparatorValue(route) {
|
|
|
223
222
|
*/
|
|
224
223
|
|
|
225
224
|
|
|
226
|
-
function routeTypeComparator(a, b) {
|
|
225
|
+
export function routeTypeComparator(a, b) {
|
|
227
226
|
return getRouteTypeComparatorValue(a) - getRouteTypeComparatorValue(b);
|
|
228
227
|
}
|
|
229
228
|
/**
|
|
@@ -231,7 +230,6 @@ function routeTypeComparator(a, b) {
|
|
|
231
230
|
* ascii character.
|
|
232
231
|
*/
|
|
233
232
|
|
|
234
|
-
|
|
235
233
|
function startsWithAlphabeticCharacter(val) {
|
|
236
234
|
if (typeof val === "string" && val.length > 0) {
|
|
237
235
|
var firstCharCode = val.charCodeAt(0);
|
|
@@ -247,7 +245,7 @@ function startsWithAlphabeticCharacter(val) {
|
|
|
247
245
|
*/
|
|
248
246
|
|
|
249
247
|
|
|
250
|
-
function alphabeticShortNameComparator(a, b) {
|
|
248
|
+
export function alphabeticShortNameComparator(a, b) {
|
|
251
249
|
var aStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(a.shortName);
|
|
252
250
|
var bStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(b.shortName);
|
|
253
251
|
|
|
@@ -264,6 +262,14 @@ function alphabeticShortNameComparator(a, b) {
|
|
|
264
262
|
|
|
265
263
|
return 0;
|
|
266
264
|
}
|
|
265
|
+
|
|
266
|
+
var isNullOrNaN = function isNullOrNaN(val) {
|
|
267
|
+
// isNaN(null) returns false so we have to check for null explicitly.
|
|
268
|
+
// Note: Using the global version of isNaN (the Number version behaves differently.
|
|
269
|
+
// eslint-disable-next-line no-restricted-globals
|
|
270
|
+
if (typeof val === null || isNaN(val)) return true;
|
|
271
|
+
return typeof val !== "number";
|
|
272
|
+
};
|
|
267
273
|
/**
|
|
268
274
|
* Checks whether an appropriate comparison of numeric values can be made for
|
|
269
275
|
* sorting purposes. If both values are not valid numbers according to the
|
|
@@ -286,21 +292,21 @@ function alphabeticShortNameComparator(a, b) {
|
|
|
286
292
|
|
|
287
293
|
|
|
288
294
|
export function makeNumericValueComparator(objGetterFn) {
|
|
289
|
-
/* Note: Using the global version of isNaN (the Number version behaves differently. */
|
|
290
|
-
|
|
291
|
-
/* eslint-disable no-restricted-globals */
|
|
292
295
|
return function (a, b) {
|
|
293
296
|
var _getSortValues = getSortValues(objGetterFn, a, b),
|
|
294
297
|
aVal = _getSortValues.aVal,
|
|
295
|
-
bVal = _getSortValues.bVal;
|
|
298
|
+
bVal = _getSortValues.bVal; // if both values aren't valid numbers, use the next sort criteria
|
|
299
|
+
|
|
296
300
|
|
|
297
|
-
if (
|
|
301
|
+
if (isNullOrNaN(aVal) && isNullOrNaN(bVal)) {
|
|
302
|
+
return 0;
|
|
303
|
+
} // b is a valid number, b gets priority
|
|
298
304
|
|
|
299
|
-
if (isNaN(aVal) && isNaN(bVal)) return 0; // b is a valid number, b gets priority
|
|
300
305
|
|
|
301
|
-
if (
|
|
306
|
+
if (isNullOrNaN(aVal)) return 1; // a is a valid number, a gets priority
|
|
302
307
|
|
|
303
|
-
if (
|
|
308
|
+
if (isNullOrNaN(bVal)) return -1; // a and b are valid numbers, return the sort value
|
|
309
|
+
// @ts-expect-error We know from the checks above that both aVal and bVal are valid numbers.
|
|
304
310
|
|
|
305
311
|
return aVal - bVal;
|
|
306
312
|
};
|
|
@@ -335,16 +341,24 @@ export function makeStringValueComparator(objGetterFn) {
|
|
|
335
341
|
};
|
|
336
342
|
}
|
|
337
343
|
/**
|
|
338
|
-
*
|
|
339
|
-
*
|
|
340
|
-
*
|
|
344
|
+
* OTP1 sets the routeSortOrder to -999 by default. If we're encountering that value in OTP1,
|
|
345
|
+
* assume that it actually means that the route sortOrder is not set in the GTFS. If we encounter
|
|
346
|
+
* it in OTP2, it's a valid value, so we should return it.
|
|
341
347
|
*
|
|
342
348
|
* See https://github.com/opentripplanner/OpenTripPlanner/issues/2938
|
|
343
349
|
* Also see https://github.com/opentripplanner/otp-react-redux/issues/122
|
|
350
|
+
* This was updated in OTP2 TO be empty by default. https://docs.opentripplanner.org/en/v2.3.0/OTP2-MigrationGuide/#:~:text=the%20Alerts-,Changes%20to%20the%20Index%20API,-Error%20handling%20is
|
|
344
351
|
*/
|
|
345
352
|
|
|
346
|
-
function getRouteSortOrderValue(
|
|
347
|
-
|
|
353
|
+
export function getRouteSortOrderValue(route) {
|
|
354
|
+
var isOTP1 = !!route.agencyId;
|
|
355
|
+
var sortOrder = route.sortOrder;
|
|
356
|
+
|
|
357
|
+
if (isOTP1 && sortOrder === -999 || sortOrder === undefined) {
|
|
358
|
+
return null;
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
return sortOrder;
|
|
348
362
|
}
|
|
349
363
|
/**
|
|
350
364
|
* Create a multi-criteria sort comparator function composed of other sort
|
|
@@ -354,8 +368,7 @@ function getRouteSortOrderValue(val) {
|
|
|
354
368
|
* are assumed to be equivalent.
|
|
355
369
|
*/
|
|
356
370
|
|
|
357
|
-
|
|
358
|
-
function makeMultiCriteriaSort() {
|
|
371
|
+
export function makeMultiCriteriaSort() {
|
|
359
372
|
for (var _len = arguments.length, criteria = new Array(_len), _key = 0; _key < _len; _key++) {
|
|
360
373
|
criteria[_key] = arguments[_key];
|
|
361
374
|
}
|
|
@@ -405,10 +418,9 @@ function makeMultiCriteriaSort() {
|
|
|
405
418
|
* 7. longName as string.
|
|
406
419
|
*/
|
|
407
420
|
|
|
408
|
-
|
|
409
421
|
export function makeRouteComparator(transitOperators) {
|
|
410
422
|
return makeMultiCriteriaSort(makeTransitOperatorComparator(transitOperators), makeNumericValueComparator(function (obj) {
|
|
411
|
-
return getRouteSortOrderValue(obj
|
|
423
|
+
return getRouteSortOrderValue(obj);
|
|
412
424
|
}), routeTypeComparator, alphabeticShortNameComparator, makeNumericValueComparator(function (obj) {
|
|
413
425
|
return parseInt(obj.shortName, 10);
|
|
414
426
|
}), makeStringValueComparator(function (obj) {
|
package/esm/route.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/route.ts"],"names":["chroma","getTransitOperatorFromFeedIdAndAgencyId","feedId","agencyId","transitOperators","find","transitOperator","getTransitOperatorFromLeg","leg","routeId","split","getTransitOperatorFromOtpRoute","route","id","agency","END_OF_LIST_COMPARATOR_VALUE","getTransitOperatorComparatorValue","length","name","agencyName","order","makeTransitOperatorComparator","a","b","aVal","bVal","getSortValues","getterFn","modeComparatorValue","SUBWAY","TRAM","TROLLEYBUS","RAIL","GONDOLA","FERRY","CABLE_CAR","FUNICULAR","BUS","routeTypeComparatorValue","getRouteTypeComparatorValue","Error","mode","type","console","warn","routeTypeComparator","startsWithAlphabeticCharacter","val","firstCharCode","charCodeAt","alphabeticShortNameComparator","aStartsWithAlphabeticCharacter","shortName","bStartsWithAlphabeticCharacter","makeNumericValueComparator","objGetterFn","isNaN","makeStringValueComparator","getRouteSortOrderValue","undefined","makeMultiCriteriaSort","criteria","i","curCriteriaComparatorValue","makeRouteComparator","obj","sortOrder","parseInt","longName","getMostReadableTextColor","backgroundColor","proposedTextColor","startsWith","fgLuminance","luminance","bgLuminance","Math","abs"],"mappings":"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","sourcesContent":["import { Leg, Route, TransitOperator } from \"@opentripplanner/types\";\nimport chroma from \"chroma-js\";\n/**\n * Returns the transit operator (if an exact match is found) from the transit\n * operators config value. It is critical to use both the feedId and agencyId in\n * this method because it is possible in OTP for there to be a duplicate\n * agencyId in separate feeds.\n *\n * @param {string} feedId The feedId that this transit agency belongs to\n * @param {string} agencyId The agencyId of the transit agency\n * @param {array} transitOperators The transitOperators list from the config\n * @return {object} The transitOperator if a match was found or null if no match\n * was found\n */\nexport function getTransitOperatorFromFeedIdAndAgencyId(\n feedId: string,\n agencyId: string | number,\n transitOperators: TransitOperator[]\n): TransitOperator {\n return (\n transitOperators.find(\n transitOperator =>\n transitOperator.feedId === feedId &&\n transitOperator.agencyId === agencyId\n ) || null\n );\n}\n\n/**\n * Looks up an operator from the provided leg.\n *\n * @param {object} leg The Itinerary Leg from which to find the transit\n * operator\n * @param {object} transitOperators transitOperators from config.\n * @return {object} the operator if one was found or null if no match was found\n */\nexport function getTransitOperatorFromLeg(\n leg: Leg,\n transitOperators: TransitOperator[]\n): TransitOperator {\n if (!leg.routeId || !leg.agencyId) return null;\n const feedId = leg.routeId.split(\":\")[0];\n return getTransitOperatorFromFeedIdAndAgencyId(\n feedId,\n leg.agencyId,\n transitOperators\n );\n}\n\n/**\n * Looks up an operator from the provided configuration given an OTP route.\n * NOTE: this assumes the use of the OTP Route model or a modified OTP\n * RouteShort model (such as the one found in the IBI fork of OTP) that also\n * returns the agencyId.\n *\n * @param {object} route Either an OTP Route or RouteShort model\n * @param {array} transitOperators transitOperators from config\n * @return {object} the operator if one was found or null if no match was found\n */\nexport function getTransitOperatorFromOtpRoute(\n route: Route,\n transitOperators: TransitOperator[]\n): TransitOperator {\n if (!route.id) return null;\n const feedId = route.id.split(\":\")[0];\n let agencyId: string | number;\n if (route.agency) {\n // This is returned in the OTP Route model\n agencyId = route.agency.id;\n } else if (route.agencyId) {\n // This is returned in the OTP RouteShort model (such as in the IBI fork)\n agencyId = route.agencyId;\n } else {\n return null;\n }\n return getTransitOperatorFromFeedIdAndAgencyId(\n feedId,\n agencyId,\n transitOperators\n );\n}\n\n// The functions below are for enhanced route sorting functions for the route\n// viewer on OTP-react-redux.\n// They address route ordering issues discussed in\n// https://github.com/opentripplanner/otp-react-redux/pull/123 and\n// https://github.com/opentripplanner/otp-react-redux/pull/124.\n\n/**\n * A large comparator value that can safely be used in mathematical sort\n * comparisons to place things at the end of lists\n */\nconst END_OF_LIST_COMPARATOR_VALUE = 999999999999;\n\n/**\n * Returns a transit operator comparator value given a route and an optional\n * transitOperators config value. This function will do its best to handle all\n * kinds of input data as certain deployments of an implementing webapp may have\n * incomplete data and certain versions of OTP might not have a modified\n * implementation of the RouteShort model.\n *\n * @param {object} route Either an OTP Route or RouteShort model\n * @param {array} transitOperators transitOperators from config\n * @return {mixed} this could return a string value (the route's agency name) if\n * the transitOperators value is not defined. Otherwise an integer will be\n * returned.\n */\nfunction getTransitOperatorComparatorValue(\n route: Route,\n transitOperators: TransitOperator[]\n): number | string {\n // if the transitOperators is undefined or has zero length, use the route's\n // agency name as the comparator value\n if (!transitOperators || transitOperators.length === 0) {\n // OTP Route\n if (route.agency) return route.agency.name;\n // OTP RouteShort (base OTP repo or IBI fork)\n if (route.agencyName) return route.agencyName;\n // shouldn't happen as agency names will be defined\n return \"zzz\";\n }\n\n // find operator associated with route\n const transitOperator = getTransitOperatorFromOtpRoute(\n route,\n transitOperators\n );\n\n // if transit operator not found, return infinity\n if (!transitOperator) return END_OF_LIST_COMPARATOR_VALUE;\n\n // return the transit operator's sort value or END_OF_LIST_COMPARATOR_VALUE if\n // the sort value is not a number\n return typeof transitOperator.order === \"number\"\n ? transitOperator.order\n : END_OF_LIST_COMPARATOR_VALUE;\n}\n\n/**\n * Calculates the sort comparator value given two routes based off of the\n * route's agency and provided transitOperators config data.\n */\nfunction makeTransitOperatorComparator(transitOperators: TransitOperator[]) {\n return (a: Route, b: Route) => {\n const aVal = getTransitOperatorComparatorValue(a, transitOperators);\n const bVal = getTransitOperatorComparatorValue(b, transitOperators);\n if (typeof aVal === \"string\") {\n // happens when transitOperators is undefined. Both aVal are guaranteed to\n // be strings. Make a string comparison.\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n }\n // @ts-expect-error transitOperators are defined and therefore a numeric value is guaranteed\n // to be returned\n return aVal - bVal;\n };\n}\n\n/**\n * Gets the desired sort values according to an optional getter function. If the\n * getter function is not defined, the original sort values are returned.\n */\nfunction getSortValues(\n getterFn: (item: unknown) => unknown,\n a: unknown,\n b: unknown\n) {\n let aVal: unknown;\n let bVal: unknown;\n if (typeof getterFn === \"function\") {\n aVal = getterFn(a);\n bVal = getterFn(b);\n } else {\n aVal = a;\n bVal = b;\n }\n return { aVal, bVal };\n}\n\n// Lookup for the sort values associated with various OTP modes.\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// https://github.com/documentationjs/documentation/issues/372\nconst modeComparatorValue = {\n SUBWAY: 1,\n TRAM: 2,\n TROLLEYBUS: 9,\n RAIL: 3,\n GONDOLA: 4,\n FERRY: 5,\n CABLE_CAR: 6,\n FUNICULAR: 7,\n BUS: 8\n};\n\n// Lookup that maps route types to the OTP mode sort values.\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// https://github.com/documentationjs/documentation/issues/372\nconst routeTypeComparatorValue = {\n 0: modeComparatorValue.TRAM, // - Tram, Streetcar, Light rail.\n 1: modeComparatorValue.SUBWAY, // - Subway, Metro.\n 2: modeComparatorValue.RAIL, // - Rail. Used for intercity or long-distance travel.\n 3: modeComparatorValue.BUS, // - Bus.\n 4: modeComparatorValue.FERRY, // - Ferry.\n 5: modeComparatorValue.CABLE_CAR, // - Cable tram.\n 6: modeComparatorValue.GONDOLA, // - Gondola, etc.\n 7: modeComparatorValue.FUNICULAR, // - Funicular.\n // TODO: 11 and 12 are not a part of OTP as of 2019-02-14, but for now just\n // associate them with bus/rail.\n 11: modeComparatorValue.BUS, // - Trolleybus.\n 12: modeComparatorValue.RAIL, // - Monorail.\n 13: modeComparatorValue.TROLLEYBUS\n};\n\n// Gets a comparator value for a given route's type (OTP mode).\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// ttps://github.com/documentationjs/documentation/issues/372\nfunction getRouteTypeComparatorValue(route: Route): number {\n // For some strange reason, the short route response in OTP returns the\n // string-based modes, but the long route response returns the\n // integer route type. This attempts to account for both of those cases.\n if (!route) throw new Error(`Route is undefined. ${route}`);\n if (typeof modeComparatorValue[route.mode] !== \"undefined\") {\n return modeComparatorValue[route.mode];\n }\n if (typeof routeTypeComparatorValue[route.type] !== \"undefined\") {\n return routeTypeComparatorValue[route.type];\n }\n // Default the comparator value to a large number (placing the route at the\n // end of the list).\n // eslint-disable-next-line no-console\n console.warn(\"no mode/route type found for route\", route);\n return END_OF_LIST_COMPARATOR_VALUE;\n}\n\n/**\n * Calculates the sort comparator value given two routes based off of route type\n * (OTP mode).\n */\nfunction routeTypeComparator(a: Route, b: Route): number {\n return getRouteTypeComparatorValue(a) - getRouteTypeComparatorValue(b);\n}\n\n/**\n * Determines whether a value is a string that starts with an alphabetic\n * ascii character.\n */\nfunction startsWithAlphabeticCharacter(val: unknown): boolean {\n if (typeof val === \"string\" && val.length > 0) {\n const firstCharCode = val.charCodeAt(0);\n return (\n (firstCharCode >= 65 && firstCharCode <= 90) ||\n (firstCharCode >= 97 && firstCharCode <= 122)\n );\n }\n return false;\n}\n\n/**\n * Sorts routes based off of whether the shortName begins with an alphabetic\n * character. Routes with shortn that do start with an alphabetic character will\n * be prioritized over those that don't.\n */\nfunction alphabeticShortNameComparator(a: Route, b: Route): number {\n const aStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(\n a.shortName\n );\n const bStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(\n b.shortName\n );\n\n if (aStartsWithAlphabeticCharacter && bStartsWithAlphabeticCharacter) {\n // both start with an alphabetic character, return equivalence\n return 0;\n }\n // a does start with an alphabetic character, but b does not. Prioritize a\n if (aStartsWithAlphabeticCharacter) return -1;\n // b does start with an alphabetic character, but a does not. Prioritize b\n if (bStartsWithAlphabeticCharacter) return 1;\n // neither route has a shortName that starts with an alphabetic character.\n // Return equivalence\n return 0;\n}\n\n/**\n * Checks whether an appropriate comparison of numeric values can be made for\n * sorting purposes. If both values are not valid numbers according to the\n * isNaN check, then this function returns undefined which indicates that a\n * secondary sorting criteria should be used instead. If one value is valid and\n * the other is not, then the valid value will be given sorting priority. If\n * both values are valid numbers, the difference is obtained as the sort value.\n *\n * An optional argument can be provided which will be used to obtain the\n * comparison value from the comparison function arguments.\n *\n * IMPORTANT: the comparison values must be numeric values or at least be\n * attempted to be converted to numeric values! If one of the arguments is\n * something crazy like an empty string, unexpected behavior will occur because\n * JavaScript.\n *\n * @param {function} [objGetterFn] An optional function to obtain the\n * comparison value from the comparator function arguments\n */\nexport function makeNumericValueComparator(\n objGetterFn?: (item: Route) => number\n) {\n /* Note: Using the global version of isNaN (the Number version behaves differently. */\n /* eslint-disable no-restricted-globals */\n return (a: number, b: number): number => {\n const { aVal, bVal } = getSortValues(objGetterFn, a, b);\n if (typeof aVal !== \"number\" || typeof bVal !== \"number\") return 0;\n\n // if both values aren't valid numbers, use the next sort criteria\n if (isNaN(aVal) && isNaN(bVal)) return 0;\n // b is a valid number, b gets priority\n if (isNaN(aVal)) return 1;\n // a is a valid number, a gets priority\n if (isNaN(bVal)) return -1;\n // a and b are valid numbers, return the sort value\n return aVal - bVal;\n };\n}\n\n/**\n * Create a comparator function that compares string values. The comparison\n * values feed to the sort comparator function are assumed to be objects that\n * will have either undefined, null or string values at the given key. If one\n * object has undefined, null or an empty string, but the other does have a\n * string with length > 0, then that string will get priority.\n *\n * @param {function} [objGetterFn] An optional function to obtain the\n * comparison value from the comparator function arguments\n */\nexport function makeStringValueComparator(\n objGetterFn?: (item: Route) => string\n) {\n return (a: string, b: string): number => {\n const { aVal, bVal } = getSortValues(objGetterFn, a, b);\n // both a and b are uncomparable strings, return equivalent value\n if (!aVal && !bVal) return 0;\n // a is not a comparable string, b gets priority\n if (!aVal) return 1;\n // b is not a comparable string, a gets priority\n if (!bVal) return -1;\n // a and b are comparable strings, return the sort value\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n };\n}\n\n/**\n * OpenTripPlanner sets the routeSortOrder to -999 by default. So, if that value\n * is encountered, assume that it actually means that the routeSortOrder is not\n * set in the GTFS.\n *\n * See https://github.com/opentripplanner/OpenTripPlanner/issues/2938\n * Also see https://github.com/opentripplanner/otp-react-redux/issues/122\n */\nfunction getRouteSortOrderValue(val: number): number {\n return val === -999 ? undefined : val;\n}\n\n/**\n * Create a multi-criteria sort comparator function composed of other sort\n * comparator functions. Each comparator function will be ran in the order given\n * until a non-zero comparison value is obtained which is then immediately\n * returned. If all comparison functions return equivalence, then the values\n * are assumed to be equivalent.\n */\nfunction makeMultiCriteriaSort(\n ...criteria: ((a: unknown, b: unknown) => number)[]\n) {\n return (a: number, b: number): number => {\n for (let i = 0; i < criteria.length; i++) {\n const curCriteriaComparatorValue = criteria[i](a, b);\n // if the comparison objects are not equivalent, return the value obtained\n // in this current criteria comparison\n if (curCriteriaComparatorValue !== 0) {\n return curCriteriaComparatorValue;\n }\n }\n return 0;\n };\n}\n\n/**\n * Creates a sort comparator function to compares routes for the purposes of\n * sorting and displaying in a user interface. This takes in a single optional\n * argument which should be a list of transitOperators as defined in the config\n * file. Due to GTFS feeds having varying levels of data quality, a multi-\n * criteria sort is needed to account for various differences. The criteria\n * included here are each applied to the routes in the order listed. If a given\n * sort criterion yields equivalence (e.g., two routes have the short name\n * \"20\"), the comparator falls back onto the next sort criterion (e.g., long\n * name). The sort operates on the following values (in order):\n *\n * 1. Transit Operator. The transit operator will be attempted to be obtained\n * for each route. If no argument is provided when creating this comparator\n * function, then routes will be sorted by their agency's name. If an\n * argument is provided and a match is found based off of the route's feed_id\n * and agency_id and a transitOperator's feed_id and agency_id, then the\n * field transitOperator.order will be used as the comparator value as long\n * as it is numeric. If it is not numeric, a value is returned indicating\n * that this transit operator should be placed at the end of the list.\n * 2. sortOrder. Routes that do not have a valid sortOrder will be placed\n * beneath those that do.\n * 3. route type (OTP mode). See routeTypeComparator code for prioritization of\n * route types.\n * 4. shortNames that begin with alphabetic characters. shortNames that do not\n * start with alphabetic characters will be place beneath those that do.\n * 5. shortName as integer. shortNames that cannot be parsed as integers will\n * be placed beneath those that are valid.\n * 6. shortName as string. Routes without shortNames will be placed beneath\n * those with shortNames.\n * 7. longName as string.\n */\nexport function makeRouteComparator(\n transitOperators: TransitOperator[]\n): (a: number, b: number) => number {\n return makeMultiCriteriaSort(\n makeTransitOperatorComparator(transitOperators),\n makeNumericValueComparator(obj => getRouteSortOrderValue(obj.sortOrder)),\n routeTypeComparator,\n alphabeticShortNameComparator,\n makeNumericValueComparator(obj => parseInt(obj.shortName, 10)),\n makeStringValueComparator(obj => obj.shortName),\n makeStringValueComparator(obj => obj.longName)\n );\n}\n\n/**\n * Tests if a pair of colors is readable. If it is, that readable color is returned.\n * If it is not, a more appropriate alternative is returned.\n *\n * Uses algorithm based on combined luminance. Values have been derived from\n * looking at real agency color pairings. These pairings are difficult to\n * generate for, as some colors see both white and black used by different agencies.\n *\n * This method therefore can accept multiple colors (including black and white) for the same background color.\n *\n * @param backgroundColor A hex string, usually the \"routeColor\"\n * @param proposedTextColor A hex string, usually the \"routeTextColor\"\n */\nexport function getMostReadableTextColor(\n backgroundColor: string,\n proposedTextColor?: string\n): string {\n // Sometimes input will defy the method signature. Therefore we need extra fallbacks\n if (!backgroundColor) backgroundColor = \"#333333\";\n if (!proposedTextColor) proposedTextColor = \"#ffffff\";\n\n if (!backgroundColor.startsWith(\"#\")) {\n backgroundColor = `#${backgroundColor}`;\n }\n if (!proposedTextColor.startsWith(\"#\")) {\n proposedTextColor = `#${proposedTextColor}`;\n }\n\n // Check if proposed color is readable\n // Luminance thresholds have been selected based on actual transit agency colors\n const fgLuminance = chroma(proposedTextColor).luminance();\n const bgLuminance = chroma(backgroundColor).luminance();\n if (\n bgLuminance + fgLuminance < 1.41 &&\n bgLuminance + fgLuminance > 0.25 &&\n Math.abs(bgLuminance - fgLuminance) > 0.2\n ) {\n return proposedTextColor;\n }\n\n // Return black or white based on luminance of background color\n // When generating colors, white is preferred.\n return chroma(backgroundColor).luminance() < 0.4 ? \"#ffffff\" : \"#000000\";\n}\n"],"file":"route.js"}
|
|
1
|
+
{"version":3,"sources":["../src/route.ts"],"names":["chroma","getTransitOperatorFromFeedIdAndAgencyId","feedId","agencyId","transitOperators","find","transitOperator","getTransitOperatorFromLeg","leg","routeId","split","getTransitOperatorFromOtpRoute","route","id","agency","END_OF_LIST_COMPARATOR_VALUE","getTransitOperatorComparatorValue","length","name","agencyName","order","makeTransitOperatorComparator","a","b","aVal","bVal","getSortValues","getterFn","modeComparatorValue","SUBWAY","TRAM","TROLLEYBUS","RAIL","GONDOLA","FERRY","CABLE_CAR","FUNICULAR","BUS","routeTypeComparatorValue","getRouteTypeComparatorValue","Error","mode","type","console","warn","routeTypeComparator","startsWithAlphabeticCharacter","val","firstCharCode","charCodeAt","alphabeticShortNameComparator","aStartsWithAlphabeticCharacter","shortName","bStartsWithAlphabeticCharacter","isNullOrNaN","isNaN","makeNumericValueComparator","objGetterFn","makeStringValueComparator","getRouteSortOrderValue","isOTP1","sortOrder","undefined","makeMultiCriteriaSort","criteria","i","curCriteriaComparatorValue","makeRouteComparator","obj","parseInt","longName","getMostReadableTextColor","backgroundColor","proposedTextColor","startsWith","fgLuminance","luminance","bgLuminance","Math","abs"],"mappings":"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","sourcesContent":["import { Leg, Route, TransitOperator } from \"@opentripplanner/types\";\nimport chroma from \"chroma-js\";\n/**\n * Returns the transit operator (if an exact match is found) from the transit\n * operators config value. It is critical to use both the feedId and agencyId in\n * this method because it is possible in OTP for there to be a duplicate\n * agencyId in separate feeds.\n *\n * @param {string} feedId The feedId that this transit agency belongs to\n * @param {string} agencyId The agencyId of the transit agency\n * @param {array} transitOperators The transitOperators list from the config\n * @return {object} The transitOperator if a match was found or null if no match\n * was found\n */\nexport function getTransitOperatorFromFeedIdAndAgencyId(\n feedId: string,\n agencyId: string | number,\n transitOperators: TransitOperator[]\n): TransitOperator {\n return (\n transitOperators.find(\n transitOperator =>\n transitOperator.feedId === feedId &&\n transitOperator.agencyId === agencyId\n ) || null\n );\n}\n\n/**\n * Looks up an operator from the provided leg.\n *\n * @param {object} leg The Itinerary Leg from which to find the transit\n * operator\n * @param {object} transitOperators transitOperators from config.\n * @return {object} the operator if one was found or null if no match was found\n */\nexport function getTransitOperatorFromLeg(\n leg: Leg,\n transitOperators: TransitOperator[]\n): TransitOperator {\n if (!leg.routeId || !leg.agencyId) return null;\n const feedId = leg.routeId.split(\":\")[0];\n return getTransitOperatorFromFeedIdAndAgencyId(\n feedId,\n leg.agencyId,\n transitOperators\n );\n}\n\n/**\n * Looks up an operator from the provided configuration given an OTP route.\n * NOTE: this assumes the use of the OTP Route model or a modified OTP\n * RouteShort model (such as the one found in the IBI fork of OTP) that also\n * returns the agencyId.\n *\n * @param {object} route Either an OTP Route or RouteShort model\n * @param {array} transitOperators transitOperators from config\n * @return {object} the operator if one was found or null if no match was found\n */\nexport function getTransitOperatorFromOtpRoute(\n route: Route,\n transitOperators: TransitOperator[]\n): TransitOperator {\n if (!route.id) return null;\n const feedId = route.id.split(\":\")[0];\n let agencyId: string | number;\n if (route.agency) {\n // This is returned in OTP2\n agencyId = route.agency.id;\n } else if (route.agencyId) {\n // This is returned in OTP1\n agencyId = route.agencyId;\n } else {\n return null;\n }\n return getTransitOperatorFromFeedIdAndAgencyId(\n feedId,\n agencyId,\n transitOperators\n );\n}\n\n// The functions below are for enhanced route sorting functions for the route\n// viewer on OTP-react-redux.\n// They address route ordering issues discussed in\n// https://github.com/opentripplanner/otp-react-redux/pull/123 and\n// https://github.com/opentripplanner/otp-react-redux/pull/124.\n\n/**\n * A large comparator value that can safely be used in mathematical sort\n * comparisons to place things at the end of lists\n */\nconst END_OF_LIST_COMPARATOR_VALUE = 999999999999;\n\n/**\n * Returns a transit operator comparator value given a route and an optional\n * transitOperators config value. This function will do its best to handle all\n * kinds of input data as certain deployments of an implementing webapp may have\n * incomplete data and certain versions of OTP might not have a modified\n * implementation of the RouteShort model.\n *\n * @param {object} route Either an OTP Route or RouteShort model\n * @param {array} transitOperators transitOperators from config\n * @return {mixed} this could return a string value (the route's agency name) if\n * the transitOperators value is not defined. Otherwise an integer will be\n * returned.\n */\nfunction getTransitOperatorComparatorValue(\n route: Route,\n transitOperators: TransitOperator[]\n): number | string {\n // if the transitOperators is undefined or has zero length, use the route's\n // agency name as the comparator value\n if (!transitOperators || transitOperators.length === 0) {\n // OTP2 Route\n if (route.agency) return route.agency.name;\n // OTP1 Route\n if (route.agencyName) return route.agencyName;\n // shouldn't happen as agency names will be defined\n return \"zzz\";\n }\n\n // find operator associated with route\n const transitOperator = getTransitOperatorFromOtpRoute(\n route,\n transitOperators\n );\n\n // if transit operator not found, return infinity\n if (!transitOperator) return END_OF_LIST_COMPARATOR_VALUE;\n\n // return the transit operator's sort value or END_OF_LIST_COMPARATOR_VALUE if\n // the sort value is not a number\n return typeof transitOperator.order === \"number\"\n ? transitOperator.order\n : END_OF_LIST_COMPARATOR_VALUE;\n}\n\n/**\n * Calculates the sort comparator value given two routes based off of the\n * route's agency and provided transitOperators config data.\n */\nexport function makeTransitOperatorComparator(\n transitOperators: TransitOperator[]\n) {\n return (a: Route, b: Route) => {\n const aVal = getTransitOperatorComparatorValue(a, transitOperators);\n const bVal = getTransitOperatorComparatorValue(b, transitOperators);\n if (typeof aVal === \"string\") {\n // happens when transitOperators is undefined. Both aVal are guaranteed to\n // be strings. Make a string comparison.\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n }\n // @ts-expect-error transitOperators are defined and therefore a numeric value is guaranteed\n // to be returned\n return aVal - bVal;\n };\n}\n\n/**\n * Gets the desired sort values according to an optional getter function. If the\n * getter function is not defined, the original sort values are returned.\n */\nfunction getSortValues(\n getterFn: (item: unknown) => unknown,\n a: unknown,\n b: unknown\n) {\n let aVal: unknown;\n let bVal: unknown;\n if (typeof getterFn === \"function\") {\n aVal = getterFn(a);\n bVal = getterFn(b);\n } else {\n aVal = a;\n bVal = b;\n }\n return { aVal, bVal };\n}\n\n// Lookup for the sort values associated with various OTP modes.\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// https://github.com/documentationjs/documentation/issues/372\nconst modeComparatorValue = {\n SUBWAY: 1,\n TRAM: 2,\n TROLLEYBUS: 9,\n RAIL: 3,\n GONDOLA: 4,\n FERRY: 5,\n CABLE_CAR: 6,\n FUNICULAR: 7,\n BUS: 8\n};\n\n// Lookup that maps route types to the OTP mode sort values.\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// https://github.com/documentationjs/documentation/issues/372\nconst routeTypeComparatorValue = {\n 0: modeComparatorValue.TRAM, // - Tram, Streetcar, Light rail.\n 1: modeComparatorValue.SUBWAY, // - Subway, Metro.\n 2: modeComparatorValue.RAIL, // - Rail. Used for intercity or long-distance travel.\n 3: modeComparatorValue.BUS, // - Bus.\n 4: modeComparatorValue.FERRY, // - Ferry.\n 5: modeComparatorValue.CABLE_CAR, // - Cable tram.\n 6: modeComparatorValue.GONDOLA, // - Gondola, etc.\n 7: modeComparatorValue.FUNICULAR, // - Funicular.\n // TODO: 11 and 12 are not a part of OTP as of 2019-02-14, but for now just\n // associate them with bus/rail.\n 11: modeComparatorValue.BUS, // - Trolleybus.\n 12: modeComparatorValue.RAIL, // - Monorail.\n 13: modeComparatorValue.TROLLEYBUS\n};\n\n// Gets a comparator value for a given route's type (OTP mode).\n// Note: JSDoc format not used to avoid bug in documentationjs.\n// ttps://github.com/documentationjs/documentation/issues/372\nfunction getRouteTypeComparatorValue(route: Route): number {\n // For some strange reason, the short route response in OTP returns the\n // string-based modes, but the long route response returns the\n // integer route type. This attempts to account for both of those cases.\n if (!route) throw new Error(`Route is undefined. ${route}`);\n if (typeof modeComparatorValue[route.mode] !== \"undefined\") {\n return modeComparatorValue[route.mode];\n }\n if (typeof routeTypeComparatorValue[route.type] !== \"undefined\") {\n return routeTypeComparatorValue[route.type];\n }\n // Default the comparator value to a large number (placing the route at the\n // end of the list).\n // eslint-disable-next-line no-console\n console.warn(\"no mode/route type found for route\", route);\n return END_OF_LIST_COMPARATOR_VALUE;\n}\n\n/**\n * Calculates the sort comparator value given two routes based off of route type\n * (OTP mode).\n */\nexport function routeTypeComparator(a: Route, b: Route): number {\n return getRouteTypeComparatorValue(a) - getRouteTypeComparatorValue(b);\n}\n\n/**\n * Determines whether a value is a string that starts with an alphabetic\n * ascii character.\n */\nfunction startsWithAlphabeticCharacter(val: unknown): boolean {\n if (typeof val === \"string\" && val.length > 0) {\n const firstCharCode = val.charCodeAt(0);\n return (\n (firstCharCode >= 65 && firstCharCode <= 90) ||\n (firstCharCode >= 97 && firstCharCode <= 122)\n );\n }\n return false;\n}\n\n/**\n * Sorts routes based off of whether the shortName begins with an alphabetic\n * character. Routes with shortn that do start with an alphabetic character will\n * be prioritized over those that don't.\n */\nexport function alphabeticShortNameComparator(a: Route, b: Route): number {\n const aStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(\n a.shortName\n );\n const bStartsWithAlphabeticCharacter = startsWithAlphabeticCharacter(\n b.shortName\n );\n\n if (aStartsWithAlphabeticCharacter && bStartsWithAlphabeticCharacter) {\n // both start with an alphabetic character, return equivalence\n return 0;\n }\n // a does start with an alphabetic character, but b does not. Prioritize a\n if (aStartsWithAlphabeticCharacter) return -1;\n // b does start with an alphabetic character, but a does not. Prioritize b\n if (bStartsWithAlphabeticCharacter) return 1;\n // neither route has a shortName that starts with an alphabetic character.\n // Return equivalence\n return 0;\n}\n\nconst isNullOrNaN = (val: any): boolean => {\n // isNaN(null) returns false so we have to check for null explicitly.\n // Note: Using the global version of isNaN (the Number version behaves differently.\n // eslint-disable-next-line no-restricted-globals\n if (typeof val === null || isNaN(val)) return true;\n\n return typeof val !== \"number\";\n};\n/**\n * Checks whether an appropriate comparison of numeric values can be made for\n * sorting purposes. If both values are not valid numbers according to the\n * isNaN check, then this function returns undefined which indicates that a\n * secondary sorting criteria should be used instead. If one value is valid and\n * the other is not, then the valid value will be given sorting priority. If\n * both values are valid numbers, the difference is obtained as the sort value.\n *\n * An optional argument can be provided which will be used to obtain the\n * comparison value from the comparison function arguments.\n *\n * IMPORTANT: the comparison values must be numeric values or at least be\n * attempted to be converted to numeric values! If one of the arguments is\n * something crazy like an empty string, unexpected behavior will occur because\n * JavaScript.\n *\n * @param {function} [objGetterFn] An optional function to obtain the\n * comparison value from the comparator function arguments\n */\nexport function makeNumericValueComparator(\n objGetterFn?: (item: Route) => number\n) {\n return (a: number, b: number): number | null => {\n const { aVal, bVal } = getSortValues(objGetterFn, a, b);\n\n // if both values aren't valid numbers, use the next sort criteria\n if (isNullOrNaN(aVal) && isNullOrNaN(bVal)) {\n return 0;\n }\n // b is a valid number, b gets priority\n if (isNullOrNaN(aVal)) return 1;\n\n // a is a valid number, a gets priority\n if (isNullOrNaN(bVal)) return -1;\n\n // a and b are valid numbers, return the sort value\n // @ts-expect-error We know from the checks above that both aVal and bVal are valid numbers.\n return aVal - bVal;\n };\n}\n\n/**\n * Create a comparator function that compares string values. The comparison\n * values feed to the sort comparator function are assumed to be objects that\n * will have either undefined, null or string values at the given key. If one\n * object has undefined, null or an empty string, but the other does have a\n * string with length > 0, then that string will get priority.\n *\n * @param {function} [objGetterFn] An optional function to obtain the\n * comparison value from the comparator function arguments\n */\nexport function makeStringValueComparator(\n objGetterFn?: (item: Route) => string\n) {\n return (a: string, b: string): number => {\n const { aVal, bVal } = getSortValues(objGetterFn, a, b);\n // both a and b are uncomparable strings, return equivalent value\n if (!aVal && !bVal) return 0;\n // a is not a comparable string, b gets priority\n if (!aVal) return 1;\n // b is not a comparable string, a gets priority\n if (!bVal) return -1;\n // a and b are comparable strings, return the sort value\n if (aVal < bVal) return -1;\n if (aVal > bVal) return 1;\n return 0;\n };\n}\n\n/**\n * OTP1 sets the routeSortOrder to -999 by default. If we're encountering that value in OTP1,\n * assume that it actually means that the route sortOrder is not set in the GTFS. If we encounter\n * it in OTP2, it's a valid value, so we should return it.\n *\n * See https://github.com/opentripplanner/OpenTripPlanner/issues/2938\n * Also see https://github.com/opentripplanner/otp-react-redux/issues/122\n * This was updated in OTP2 TO be empty by default. https://docs.opentripplanner.org/en/v2.3.0/OTP2-MigrationGuide/#:~:text=the%20Alerts-,Changes%20to%20the%20Index%20API,-Error%20handling%20is\n */\nexport function getRouteSortOrderValue(route: Route): number {\n const isOTP1 = !!route.agencyId;\n const { sortOrder } = route;\n\n if ((isOTP1 && sortOrder === -999) || sortOrder === undefined) {\n return null;\n }\n\n return sortOrder;\n}\n\n/**\n * Create a multi-criteria sort comparator function composed of other sort\n * comparator functions. Each comparator function will be ran in the order given\n * until a non-zero comparison value is obtained which is then immediately\n * returned. If all comparison functions return equivalence, then the values\n * are assumed to be equivalent.\n */\nexport function makeMultiCriteriaSort(\n ...criteria: ((a: unknown, b: unknown) => number)[]\n) {\n return (a: number, b: number): number => {\n for (let i = 0; i < criteria.length; i++) {\n const curCriteriaComparatorValue = criteria[i](a, b);\n // if the comparison objects are not equivalent, return the value obtained\n // in this current criteria comparison\n if (curCriteriaComparatorValue !== 0) {\n return curCriteriaComparatorValue;\n }\n }\n return 0;\n };\n}\n\n/**\n * Creates a sort comparator function to compares routes for the purposes of\n * sorting and displaying in a user interface. This takes in a single optional\n * argument which should be a list of transitOperators as defined in the config\n * file. Due to GTFS feeds having varying levels of data quality, a multi-\n * criteria sort is needed to account for various differences. The criteria\n * included here are each applied to the routes in the order listed. If a given\n * sort criterion yields equivalence (e.g., two routes have the short name\n * \"20\"), the comparator falls back onto the next sort criterion (e.g., long\n * name). The sort operates on the following values (in order):\n *\n * 1. Transit Operator. The transit operator will be attempted to be obtained\n * for each route. If no argument is provided when creating this comparator\n * function, then routes will be sorted by their agency's name. If an\n * argument is provided and a match is found based off of the route's feed_id\n * and agency_id and a transitOperator's feed_id and agency_id, then the\n * field transitOperator.order will be used as the comparator value as long\n * as it is numeric. If it is not numeric, a value is returned indicating\n * that this transit operator should be placed at the end of the list.\n * 2. sortOrder. Routes that do not have a valid sortOrder will be placed\n * beneath those that do.\n * 3. route type (OTP mode). See routeTypeComparator code for prioritization of\n * route types.\n * 4. shortNames that begin with alphabetic characters. shortNames that do not\n * start with alphabetic characters will be place beneath those that do.\n * 5. shortName as integer. shortNames that cannot be parsed as integers will\n * be placed beneath those that are valid.\n * 6. shortName as string. Routes without shortNames will be placed beneath\n * those with shortNames.\n * 7. longName as string.\n */\nexport function makeRouteComparator(\n transitOperators: TransitOperator[]\n): (a: number, b: number) => number {\n return makeMultiCriteriaSort(\n makeTransitOperatorComparator(transitOperators),\n makeNumericValueComparator(obj => getRouteSortOrderValue(obj)),\n routeTypeComparator,\n alphabeticShortNameComparator,\n makeNumericValueComparator(obj => parseInt(obj.shortName, 10)),\n makeStringValueComparator(obj => obj.shortName),\n makeStringValueComparator(obj => obj.longName)\n );\n}\n\n/**\n * Tests if a pair of colors is readable. If it is, that readable color is returned.\n * If it is not, a more appropriate alternative is returned.\n *\n * Uses algorithm based on combined luminance. Values have been derived from\n * looking at real agency color pairings. These pairings are difficult to\n * generate for, as some colors see both white and black used by different agencies.\n *\n * This method therefore can accept multiple colors (including black and white) for the same background color.\n *\n * @param backgroundColor A hex string, usually the \"routeColor\"\n * @param proposedTextColor A hex string, usually the \"routeTextColor\"\n */\nexport function getMostReadableTextColor(\n backgroundColor: string,\n proposedTextColor?: string\n): string {\n // Sometimes input will defy the method signature. Therefore we need extra fallbacks\n if (!backgroundColor) backgroundColor = \"#333333\";\n if (!proposedTextColor) proposedTextColor = \"#ffffff\";\n\n if (!backgroundColor.startsWith(\"#\")) {\n backgroundColor = `#${backgroundColor}`;\n }\n if (!proposedTextColor.startsWith(\"#\")) {\n proposedTextColor = `#${proposedTextColor}`;\n }\n\n // Check if proposed color is readable\n // Luminance thresholds have been selected based on actual transit agency colors\n const fgLuminance = chroma(proposedTextColor).luminance();\n const bgLuminance = chroma(backgroundColor).luminance();\n if (\n bgLuminance + fgLuminance < 1.41 &&\n bgLuminance + fgLuminance > 0.25 &&\n Math.abs(bgLuminance - fgLuminance) > 0.2\n ) {\n return proposedTextColor;\n }\n\n // Return black or white based on luminance of background color\n // When generating colors, white is preferred.\n return chroma(backgroundColor).luminance() < 0.4 ? \"#ffffff\" : \"#000000\";\n}\n"],"file":"route.js"}
|