@opentripplanner/core-utils 12.0.0-alpha.1 → 12.0.0-alpha.10
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/itinerary.js +55 -18
- package/esm/itinerary.js.map +1 -1
- package/esm/query-gen.js +11 -4
- package/esm/query-gen.js.map +1 -1
- package/esm/query.js +1 -1
- package/esm/query.js.map +1 -1
- package/esm/route.js +4 -2
- package/esm/route.js.map +1 -1
- package/esm/ui.js +0 -8
- package/esm/ui.js.map +1 -1
- package/lib/itinerary.d.ts +15 -2
- package/lib/itinerary.d.ts.map +1 -1
- package/lib/itinerary.js +71 -20
- package/lib/itinerary.js.map +1 -1
- package/lib/query-gen.d.ts +4 -1
- package/lib/query-gen.d.ts.map +1 -1
- package/lib/query-gen.js +11 -4
- package/lib/query-gen.js.map +1 -1
- package/lib/query.js +1 -1
- package/lib/query.js.map +1 -1
- package/lib/route.d.ts.map +1 -1
- package/lib/route.js +4 -2
- package/lib/route.js.map +1 -1
- package/lib/ui.d.ts +0 -4
- package/lib/ui.d.ts.map +1 -1
- package/lib/ui.js +0 -13
- package/lib/ui.js.map +1 -1
- package/package.json +3 -4
- package/src/__tests__/{itinerary.js → itinerary.ts} +91 -1
- package/src/__tests__/query-params.ts +10 -0
- package/src/itinerary.ts +71 -18
- package/src/otpSchema.json +313 -4
- package/src/planQuery.graphql +53 -15
- package/src/query-gen.ts +12 -3
- package/src/query.js +1 -1
- package/src/route.ts +3 -1
- package/src/ui.ts +0 -10
- package/tsconfig.tsbuildinfo +1 -1
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","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 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};\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","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"}
|
package/esm/ui.js
CHANGED
|
@@ -1,15 +1,7 @@
|
|
|
1
|
-
import bowser from "bowser";
|
|
2
1
|
export function isMobile() {
|
|
3
2
|
// TODO: consider using 3rd-party library?
|
|
4
3
|
return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
|
|
5
4
|
}
|
|
6
|
-
/**
|
|
7
|
-
* Returns true if the user is using a [redacted] browser
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
export function isIE() {
|
|
11
|
-
return bowser.parse(navigator.userAgent).browser === "Internet Explorer";
|
|
12
|
-
}
|
|
13
5
|
/**
|
|
14
6
|
* Enables scrolling for a specified selector, while disabling scrolling for all
|
|
15
7
|
* other targets. This is adapted from https://stackoverflow.com/a/41601290/915811
|
package/esm/ui.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/ui.ts"],"names":["
|
|
1
|
+
{"version":3,"sources":["../src/ui.ts"],"names":["isMobile","test","navigator","userAgent","enableScrollForSelector","selector","overlay","document","querySelector","clientY","isOverlayTotallyScrolled","scrollHeight","scrollTop","clientHeight","disableRubberBand","event","clientYDelta","targetTouches","preventDefault","addEventListener","length"],"mappings":"AAAA,OAAO,SAASA,QAAT,GAA6B;AAClC;AACA,SAAO,iEAAiEC,IAAjE,CACLC,SAAS,CAACC,SADL,CAAP;AAGD;AACD;AACA;AACA;AACA;AACA;AACA;;AACA,OAAO,SAASC,uBAAT,CAAiCC,QAAjC,EAAyD;AAC9D,MAAMC,OAAO,GAAGC,QAAQ,CAACC,aAAT,CAAuBH,QAAvB,CAAhB;AACA,MAAII,OAAO,GAAG,IAAd,CAF8D,CAE1C;;AAEpB,WAASC,wBAAT,GAA6C;AAC3C;AACA,WAAOJ,OAAO,CAACK,YAAR,GAAuBL,OAAO,CAACM,SAA/B,IAA4CN,OAAO,CAACO,YAA3D;AACD;;AAED,WAASC,iBAAT,CAA2BC,KAA3B,EAA8C;AAC5C,QAAMC,YAAY,GAAGD,KAAK,CAACE,aAAN,CAAoB,CAApB,EAAuBR,OAAvB,GAAiCA,OAAtD;;AAEA,QAAIH,OAAO,CAACM,SAAR,KAAsB,CAAtB,IAA2BI,YAAY,GAAG,CAA9C,EAAiD;AAC/C;AACAD,MAAAA,KAAK,CAACG,cAAN;AACD;;AAED,QAAIR,wBAAwB,MAAMM,YAAY,GAAG,CAAjD,EAAoD;AAClD;AACAD,MAAAA,KAAK,CAACG,cAAN;AACD;AACF;;AAEDZ,EAAAA,OAAO,CAACa,gBAAR,CACE,YADF,EAEE,UAASJ,KAAT,EAA4B;AAC1B,QAAIA,KAAK,CAACE,aAAN,CAAoBG,MAApB,KAA+B,CAAnC,EAAsC;AACpC;AACAX,MAAAA,OAAO,GAAGM,KAAK,CAACE,aAAN,CAAoB,CAApB,EAAuBR,OAAjC;AACD;AACF,GAPH,EAQE,KARF;AAWAH,EAAAA,OAAO,CAACa,gBAAR,CACE,WADF,EAEE,UAASJ,KAAT,EAA4B;AAC1B,QAAIA,KAAK,CAACE,aAAN,CAAoBG,MAApB,KAA+B,CAAnC,EAAsC;AACpC;AACAN,MAAAA,iBAAiB,CAACC,KAAD,CAAjB;AACD;AACF,GAPH,EAQE,KARF;AAUD","sourcesContent":["export function isMobile(): boolean {\n // TODO: consider using 3rd-party library?\n return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(\n navigator.userAgent\n );\n}\n/**\n * Enables scrolling for a specified selector, while disabling scrolling for all\n * other targets. This is adapted from https://stackoverflow.com/a/41601290/915811\n * and intended to fix issues with iOS elastic scrolling, e.g.,\n * https://github.com/conveyal/trimet-mod-otp/issues/92.\n */\nexport function enableScrollForSelector(selector: string): void {\n const overlay = document.querySelector(selector);\n let clientY = null; // remember Y position on touch start\n\n function isOverlayTotallyScrolled(): boolean {\n // https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollHeight#Problems_and_solutions\n return overlay.scrollHeight - overlay.scrollTop <= overlay.clientHeight;\n }\n\n function disableRubberBand(event: TouchEvent) {\n const clientYDelta = event.targetTouches[0].clientY - clientY;\n\n if (overlay.scrollTop === 0 && clientYDelta > 0) {\n // element is at the top of its scroll\n event.preventDefault();\n }\n\n if (isOverlayTotallyScrolled() && clientYDelta < 0) {\n // element is at the top of its scroll\n event.preventDefault();\n }\n }\n\n overlay.addEventListener(\n \"touchstart\",\n function(event: TouchEvent) {\n if (event.targetTouches.length === 1) {\n // detect single touch\n clientY = event.targetTouches[0].clientY;\n }\n },\n false\n );\n\n overlay.addEventListener(\n \"touchmove\",\n function(event: TouchEvent) {\n if (event.targetTouches.length === 1) {\n // detect single touch\n disableRubberBand(event);\n }\n },\n false\n );\n}\n"],"file":"ui.js"}
|
package/lib/itinerary.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Company, Config, ElevationProfile, FlexBookingInfo, ItineraryOnlyLegsRequired, LatLngArray, Leg, MassUnitOption, Money, Place, Step, Stop, TncFare } from "@opentripplanner/types";
|
|
1
|
+
import { Company, Config, ElevationProfile, ElevationProfileComponent, FlexBookingInfo, ItineraryOnlyLegsRequired, LatLngArray, Leg, MassUnitOption, Money, Place, Step, Stop, TncFare } from "@opentripplanner/types";
|
|
2
2
|
export declare const transitModes: string[];
|
|
3
3
|
/**
|
|
4
4
|
* @param {config} config OTP-RR configuration object
|
|
@@ -75,6 +75,10 @@ export declare function getItineraryBounds(itinerary: ItineraryOnlyLegsRequired)
|
|
|
75
75
|
export declare function getLegBounds(leg: Leg): number[][];
|
|
76
76
|
export declare function legLocationAtDistance(leg: Leg, distance: number): number[];
|
|
77
77
|
export declare function legElevationAtDistance(points: number[][], distance: number): number;
|
|
78
|
+
export declare function mapOldElevationComponentToNew(oldElev: {
|
|
79
|
+
first: number;
|
|
80
|
+
second: number;
|
|
81
|
+
}): ElevationProfileComponent;
|
|
78
82
|
export declare function getElevationProfile(steps: Step[], unitConversion?: number): ElevationProfile;
|
|
79
83
|
/**
|
|
80
84
|
* Uses canvas.measureText to compute and return the width of the given text of given font in pixels.
|
|
@@ -97,7 +101,7 @@ export declare function getCompanyForNetwork(networkString: string, companies?:
|
|
|
97
101
|
* @param {Array<object>} [companies=[]] An optional list of the companies config.
|
|
98
102
|
* @return {string} A label for use in presentation on a website.
|
|
99
103
|
*/
|
|
100
|
-
export declare function getCompaniesLabelFromNetworks(networks: string | string
|
|
104
|
+
export declare function getCompaniesLabelFromNetworks(networks: string[] | string, companies?: Company[]): string;
|
|
101
105
|
export declare function getTNCLocation(leg: Leg, type: string): string;
|
|
102
106
|
export declare function calculatePhysicalActivity(itinerary: ItineraryOnlyLegsRequired): {
|
|
103
107
|
bikeDuration: number;
|
|
@@ -145,4 +149,13 @@ export declare function getLegCost(leg: Leg, mediumId: string | null, riderCateg
|
|
|
145
149
|
*/
|
|
146
150
|
export declare function getItineraryCost(legs: Leg[], mediumId: string | null, riderCategoryId: string | null): Money | undefined;
|
|
147
151
|
export declare const convertGraphQLResponseToLegacy: (leg: any) => any;
|
|
152
|
+
/** Extracts the route number for a leg returned from OTP1 or OTP2. */
|
|
153
|
+
export declare const getLegRouteShortName: (leg: Pick<Leg, "route" | "routeShortName">) => string | null;
|
|
154
|
+
/** Extract the route long name for a leg returned from OTP1 or OTP2. */
|
|
155
|
+
export declare const getLegRouteLongName: (leg: Pick<Leg, "route" | "routeLongName">) => string | null;
|
|
156
|
+
/**
|
|
157
|
+
* Returns the route short name, or the route long name if no short name is provided.
|
|
158
|
+
* This is happens with Seattle area streetcars and ferries.
|
|
159
|
+
*/
|
|
160
|
+
export declare const getLegRouteName: (leg: Pick<Leg, "route" | "routeLongName" | "routeShortName">) => string;
|
|
148
161
|
//# sourceMappingURL=itinerary.d.ts.map
|
package/lib/itinerary.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"itinerary.d.ts","sourceRoot":"","sources":["../src/itinerary.ts"],"names":[],"mappings":"AACA,OAAO,EACL,OAAO,EACP,MAAM,EACN,gBAAgB,EAChB,eAAe,EACf,yBAAyB,EACzB,WAAW,EACX,GAAG,EACH,cAAc,EACd,KAAK,EACL,KAAK,EACL,IAAI,EACJ,IAAI,EACJ,OAAO,EACR,MAAM,wBAAwB,CAAC;AAIhC,eAAO,MAAM,YAAY,
|
|
1
|
+
{"version":3,"file":"itinerary.d.ts","sourceRoot":"","sources":["../src/itinerary.ts"],"names":[],"mappings":"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"}
|
package/lib/itinerary.js
CHANGED
|
@@ -32,6 +32,7 @@ exports.getItineraryBounds = getItineraryBounds;
|
|
|
32
32
|
exports.getLegBounds = getLegBounds;
|
|
33
33
|
exports.legLocationAtDistance = legLocationAtDistance;
|
|
34
34
|
exports.legElevationAtDistance = legElevationAtDistance;
|
|
35
|
+
exports.mapOldElevationComponentToNew = mapOldElevationComponentToNew;
|
|
35
36
|
exports.getElevationProfile = getElevationProfile;
|
|
36
37
|
exports.getTextWidth = getTextWidth;
|
|
37
38
|
exports.getCompanyForNetwork = getCompanyForNetwork;
|
|
@@ -43,14 +44,14 @@ exports.calculateEmissions = calculateEmissions;
|
|
|
43
44
|
exports.getDisplayedStopId = getDisplayedStopId;
|
|
44
45
|
exports.getLegCost = getLegCost;
|
|
45
46
|
exports.getItineraryCost = getItineraryCost;
|
|
46
|
-
exports.convertGraphQLResponseToLegacy = exports.transitModes = void 0;
|
|
47
|
+
exports.getLegRouteName = exports.getLegRouteLongName = exports.getLegRouteShortName = exports.convertGraphQLResponseToLegacy = exports.transitModes = void 0;
|
|
47
48
|
|
|
48
49
|
var _polyline = _interopRequireDefault(require("@mapbox/polyline"));
|
|
49
50
|
|
|
50
51
|
var _along = _interopRequireDefault(require("@turf/along"));
|
|
51
52
|
|
|
52
53
|
// All OTP transit modes
|
|
53
|
-
const transitModes = ["TRAM", "BUS", "SUBWAY", "FERRY", "RAIL", "GONDOLA"];
|
|
54
|
+
const transitModes = ["TRAM", "TROLLEYBUS", "BUS", "SUBWAY", "FERRY", "RAIL", "GONDOLA"];
|
|
54
55
|
/**
|
|
55
56
|
* @param {config} config OTP-RR configuration object
|
|
56
57
|
* @return {Array} List of all transit modes defined in config; otherwise default mode list
|
|
@@ -203,6 +204,7 @@ function getMapColor(mode) {
|
|
|
203
204
|
if (mode === "SUBWAY") return "#e60000";
|
|
204
205
|
if (mode === "RAIL") return "#b00";
|
|
205
206
|
if (mode === "BUS") return "#080";
|
|
207
|
+
if (mode === "TROLLEYBUS") return "#080";
|
|
206
208
|
if (mode === "TRAM") return "#800";
|
|
207
209
|
if (mode === "FERRY") return "#008";
|
|
208
210
|
if (mode === "CAR") return "#444";
|
|
@@ -322,7 +324,7 @@ function legElevationAtDistance(points, distance) {
|
|
|
322
324
|
|
|
323
325
|
if (distance >= traversed && distance <= traversed + elevDistanceSpan) {
|
|
324
326
|
// Distance falls within this point and the previous one;
|
|
325
|
-
// compute & return
|
|
327
|
+
// compute & return interpolated elevation value
|
|
326
328
|
if (start[1] === null) {
|
|
327
329
|
console.warn("Elevation value does not exist for distance.", distance, traversed);
|
|
328
330
|
return null;
|
|
@@ -338,6 +340,13 @@ function legElevationAtDistance(points, distance) {
|
|
|
338
340
|
|
|
339
341
|
console.warn("Elevation value does not exist for distance.", distance, traversed);
|
|
340
342
|
return null;
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
function mapOldElevationComponentToNew(oldElev) {
|
|
346
|
+
return {
|
|
347
|
+
distance: oldElev.first,
|
|
348
|
+
elevation: oldElev.second
|
|
349
|
+
};
|
|
341
350
|
} // Iterate through the steps, building the array of elevation points and
|
|
342
351
|
// keeping track of the minimum and maximum elevations reached
|
|
343
352
|
|
|
@@ -351,29 +360,34 @@ function getElevationProfile(steps, unitConversion = 1) {
|
|
|
351
360
|
let previous = null;
|
|
352
361
|
const points = [];
|
|
353
362
|
steps.forEach(step => {
|
|
354
|
-
|
|
363
|
+
var _step$elevation;
|
|
364
|
+
|
|
365
|
+
// Support for old REST response data (in step.elevation)
|
|
366
|
+
const stepElevationProfile = step.elevationProfile || Array.isArray(step.elevation) && ((_step$elevation = step.elevation) === null || _step$elevation === void 0 ? void 0 : _step$elevation.map(mapOldElevationComponentToNew));
|
|
367
|
+
|
|
368
|
+
if (!stepElevationProfile || stepElevationProfile.length === 0) {
|
|
355
369
|
traversed += step.distance;
|
|
356
370
|
return;
|
|
357
371
|
}
|
|
358
372
|
|
|
359
|
-
for (let i = 0; i <
|
|
360
|
-
const elev =
|
|
373
|
+
for (let i = 0; i < stepElevationProfile.length; i++) {
|
|
374
|
+
const elev = stepElevationProfile[i];
|
|
361
375
|
|
|
362
376
|
if (previous) {
|
|
363
|
-
const diff = (elev.
|
|
377
|
+
const diff = (elev.elevation - previous.elevation) * unitConversion;
|
|
364
378
|
if (diff > 0) gain += diff;else loss += diff;
|
|
365
379
|
}
|
|
366
380
|
|
|
367
|
-
if (i === 0 && elev.
|
|
381
|
+
if (i === 0 && elev.distance !== 0) {// console.warn(`No elevation data available for step ${stepIndex}-${i} at beginning of segment`, elev)
|
|
368
382
|
}
|
|
369
383
|
|
|
370
|
-
const convertedElevation = elev.
|
|
384
|
+
const convertedElevation = elev.elevation * unitConversion;
|
|
371
385
|
if (convertedElevation < minElev) minElev = convertedElevation;
|
|
372
386
|
if (convertedElevation > maxElev) maxElev = convertedElevation;
|
|
373
|
-
points.push([traversed + elev.
|
|
387
|
+
points.push([traversed + elev.distance, elev.elevation]); // Insert "filler" point if the last point in elevation profile does not
|
|
374
388
|
// reach the full distance of the step.
|
|
375
389
|
|
|
376
|
-
if (i ===
|
|
390
|
+
if (i === stepElevationProfile.length - 1 && elev.distance !== step.distance) {// points.push([traversed + step.distance, elev.second])
|
|
377
391
|
}
|
|
378
392
|
|
|
379
393
|
previous = elev;
|
|
@@ -401,8 +415,8 @@ function getElevationProfile(steps, unitConversion = 1) {
|
|
|
401
415
|
|
|
402
416
|
|
|
403
417
|
function getTextWidth(text, font = "22px Arial") {
|
|
404
|
-
// Create custom type for function including
|
|
405
|
-
//
|
|
418
|
+
// Create custom type for function including reused canvas object
|
|
419
|
+
// reuse canvas object for better performance
|
|
406
420
|
const canvas = getTextWidth.canvas || (getTextWidth.canvas = document.createElement("canvas"));
|
|
407
421
|
const context = canvas.getContext("2d");
|
|
408
422
|
context.font = font;
|
|
@@ -434,9 +448,7 @@ function getCompanyForNetwork(networkString, companies = []) {
|
|
|
434
448
|
|
|
435
449
|
|
|
436
450
|
function getCompaniesLabelFromNetworks(networks, companies = []) {
|
|
437
|
-
|
|
438
|
-
if (typeof networks === "string") networksArray = [networks];
|
|
439
|
-
return networksArray.map(network => getCompanyForNetwork(network, companies)).filter(co => !!co).map(co => co.label).join("/");
|
|
451
|
+
return (Array.isArray(networks) ? networks : [networks]).map(network => getCompanyForNetwork(network, companies)).filter(co => !!co).map(co => co.label).join("/");
|
|
440
452
|
}
|
|
441
453
|
|
|
442
454
|
function getTNCLocation(leg, type) {
|
|
@@ -502,6 +514,7 @@ const CARBON_INTENSITY_DEFAULTS = {
|
|
|
502
514
|
bicycle: 0.017,
|
|
503
515
|
car: 0.162,
|
|
504
516
|
tram: 0.066,
|
|
517
|
+
trolleybus: 0.066,
|
|
505
518
|
subway: 0.066,
|
|
506
519
|
rail: 0.066,
|
|
507
520
|
bus: 0.09,
|
|
@@ -670,12 +683,13 @@ const pickupDropoffTypeToOtp1 = otp2Type => {
|
|
|
670
683
|
};
|
|
671
684
|
|
|
672
685
|
const convertGraphQLResponseToLegacy = leg => {
|
|
673
|
-
var _leg$agency, _leg$agency2, _leg$agency3, _leg$from$stop, _leg$from$stop2, _leg$route, _leg$route2, _leg$route3, _leg$route4, _leg$route5, _leg$route6, _leg$to$stop, _leg$to$stop2, _leg$trip, _leg$trip2;
|
|
686
|
+
var _leg$agency, _leg$agency2, _leg$agency3, _leg$agency4, _leg$from$stop, _leg$from$stop2, _leg$route, _leg$route2, _leg$route3, _leg$route4, _leg$route5, _leg$route6, _leg$to$stop, _leg$to$stop2, _leg$trip, _leg$trip2;
|
|
674
687
|
|
|
675
688
|
return { ...leg,
|
|
676
689
|
agencyBrandingUrl: (_leg$agency = leg.agency) === null || _leg$agency === void 0 ? void 0 : _leg$agency.url,
|
|
677
|
-
|
|
678
|
-
|
|
690
|
+
agencyId: (_leg$agency2 = leg.agency) === null || _leg$agency2 === void 0 ? void 0 : _leg$agency2.id,
|
|
691
|
+
agencyName: (_leg$agency3 = leg.agency) === null || _leg$agency3 === void 0 ? void 0 : _leg$agency3.name,
|
|
692
|
+
agencyUrl: (_leg$agency4 = leg.agency) === null || _leg$agency4 === void 0 ? void 0 : _leg$agency4.url,
|
|
679
693
|
alightRule: pickupDropoffTypeToOtp1(leg.dropoffType),
|
|
680
694
|
boardRule: pickupDropoffTypeToOtp1(leg.pickupType),
|
|
681
695
|
dropOffBookingInfo: {
|
|
@@ -687,7 +701,7 @@ const convertGraphQLResponseToLegacy = leg => {
|
|
|
687
701
|
},
|
|
688
702
|
route: (_leg$route = leg.route) === null || _leg$route === void 0 ? void 0 : _leg$route.shortName,
|
|
689
703
|
routeColor: (_leg$route2 = leg.route) === null || _leg$route2 === void 0 ? void 0 : _leg$route2.color,
|
|
690
|
-
routeId: (_leg$route3 = leg.route) === null || _leg$route3 === void 0 ? void 0 : _leg$route3.
|
|
704
|
+
routeId: (_leg$route3 = leg.route) === null || _leg$route3 === void 0 ? void 0 : _leg$route3.gtfsId,
|
|
691
705
|
routeLongName: (_leg$route4 = leg.route) === null || _leg$route4 === void 0 ? void 0 : _leg$route4.longName,
|
|
692
706
|
routeShortName: (_leg$route5 = leg.route) === null || _leg$route5 === void 0 ? void 0 : _leg$route5.shortName,
|
|
693
707
|
routeTextColor: (_leg$route6 = leg.route) === null || _leg$route6 === void 0 ? void 0 : _leg$route6.textColor,
|
|
@@ -699,6 +713,43 @@ const convertGraphQLResponseToLegacy = leg => {
|
|
|
699
713
|
tripId: (_leg$trip2 = leg.trip) === null || _leg$trip2 === void 0 ? void 0 : _leg$trip2.gtfsId
|
|
700
714
|
};
|
|
701
715
|
};
|
|
716
|
+
/** Extracts the route number for a leg returned from OTP1 or OTP2. */
|
|
717
|
+
|
|
702
718
|
|
|
703
719
|
exports.convertGraphQLResponseToLegacy = convertGraphQLResponseToLegacy;
|
|
720
|
+
|
|
721
|
+
const getLegRouteShortName = leg => {
|
|
722
|
+
const {
|
|
723
|
+
route,
|
|
724
|
+
routeShortName
|
|
725
|
+
} = leg; // typeof route === "object" denotes newer OTP2 responses. routeShortName and route as string is OTP1.
|
|
726
|
+
|
|
727
|
+
return typeof route === "object" ? route === null || route === void 0 ? void 0 : route.shortName : routeShortName || route;
|
|
728
|
+
};
|
|
729
|
+
/** Extract the route long name for a leg returned from OTP1 or OTP2. */
|
|
730
|
+
|
|
731
|
+
|
|
732
|
+
exports.getLegRouteShortName = getLegRouteShortName;
|
|
733
|
+
|
|
734
|
+
const getLegRouteLongName = leg => {
|
|
735
|
+
const {
|
|
736
|
+
route,
|
|
737
|
+
routeLongName
|
|
738
|
+
} = leg; // typeof route === "object" denotes newer OTP2 responses. routeLongName is OTP1.
|
|
739
|
+
|
|
740
|
+
return typeof route === "object" ? route === null || route === void 0 ? void 0 : route.longName : routeLongName;
|
|
741
|
+
};
|
|
742
|
+
/**
|
|
743
|
+
* Returns the route short name, or the route long name if no short name is provided.
|
|
744
|
+
* This is happens with Seattle area streetcars and ferries.
|
|
745
|
+
*/
|
|
746
|
+
|
|
747
|
+
|
|
748
|
+
exports.getLegRouteLongName = getLegRouteLongName;
|
|
749
|
+
|
|
750
|
+
const getLegRouteName = leg => {
|
|
751
|
+
return getLegRouteShortName(leg) || getLegRouteLongName(leg);
|
|
752
|
+
};
|
|
753
|
+
|
|
754
|
+
exports.getLegRouteName = getLegRouteName;
|
|
704
755
|
//# sourceMappingURL=itinerary.js.map
|