@tiba-spark/client-shared-lib 25.4.1-204 → 25.4.1-212
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/esm2022/libraries/services/ticket.service.mjs +52 -37
- package/fesm2022/tiba-spark-client-shared-lib.mjs +51 -37
- package/fesm2022/tiba-spark-client-shared-lib.mjs.map +1 -1
- package/libraries/services/ticket.service.d.ts +3 -3
- package/libraries/services/ticket.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -4,6 +4,7 @@ import { Store } from '@ngxs/store';
|
|
|
4
4
|
import moment from 'moment';
|
|
5
5
|
import { CommandNotificationService } from '../components/notification/command-notification/command-notification.service';
|
|
6
6
|
import { NotificationClass } from '../components/notification/notification-config';
|
|
7
|
+
import { DEFAULT_SMART_PARK_DATE_TIME } from '../constants/global-constants';
|
|
7
8
|
import { BarcodeTypeLength } from '../enums/barcode-type.enum';
|
|
8
9
|
import { FacilityState } from '../modules/facility/facility.state';
|
|
9
10
|
import { ParkerState } from '../modules/parker/parker.state';
|
|
@@ -39,47 +40,52 @@ export class TicketService {
|
|
|
39
40
|
return this.sendCommand(commandType, options, analyticsPayload)
|
|
40
41
|
.pipe(this.commandNotificationService.display(config));
|
|
41
42
|
}
|
|
42
|
-
extractTicketInfo(content) {
|
|
43
|
+
extractTicketInfo(content, timezone) {
|
|
43
44
|
let ticketNum;
|
|
44
|
-
let
|
|
45
|
+
let ticketDateTimeInSeconds;
|
|
45
46
|
try {
|
|
46
47
|
// To determine what type, check content length
|
|
47
48
|
switch (content.length) {
|
|
48
49
|
case BarcodeTypeLength.QR1:
|
|
49
50
|
case BarcodeTypeLength.QR2:
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
51
|
+
{
|
|
52
|
+
ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14
|
|
53
|
+
// Location of ticket date; from 14 to 20
|
|
54
|
+
// Location of ticket time; from 20 to 24
|
|
55
|
+
const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });
|
|
56
|
+
ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
57
|
+
}
|
|
54
58
|
break;
|
|
55
59
|
case BarcodeTypeLength.QR4:
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
+
{
|
|
61
|
+
ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.
|
|
62
|
+
// Location of ticket date; from 16 to 22
|
|
63
|
+
// Location of ticket time; from 22 to 28
|
|
64
|
+
const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });
|
|
65
|
+
ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
66
|
+
}
|
|
60
67
|
break;
|
|
61
68
|
case BarcodeTypeLength.Barcode1:
|
|
62
69
|
ticketNum = content.slice(7, 14); // Location of ticket number; from 7 to 14
|
|
63
70
|
// Location of ticket time info; from 1 to 7
|
|
64
|
-
|
|
71
|
+
ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 1, end: 7 }, timezone);
|
|
65
72
|
break;
|
|
66
73
|
case BarcodeTypeLength.Barcode2:
|
|
67
74
|
ticketNum = content.slice(13, 22); // Location of ticket number; from 14 to 22
|
|
68
75
|
// Location of ticket time info; from 7 to 13
|
|
69
|
-
|
|
76
|
+
ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 7, end: 13 }, timezone);
|
|
70
77
|
break;
|
|
71
78
|
default:
|
|
72
79
|
{
|
|
73
80
|
const qValue = this.extractQueryParam(content, 'q');
|
|
74
|
-
if (qValue !== null) {
|
|
75
|
-
return this.extractTicketInfo(qValue);
|
|
81
|
+
if (qValue !== null && qValue !== content) {
|
|
82
|
+
return this.extractTicketInfo(qValue, timezone);
|
|
76
83
|
}
|
|
77
84
|
}
|
|
78
85
|
break;
|
|
79
86
|
}
|
|
80
87
|
}
|
|
81
88
|
catch (_) { }
|
|
82
|
-
const ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
83
89
|
return { ticketNum, ticketDateTimeInSeconds };
|
|
84
90
|
}
|
|
85
91
|
extractQueryParam(url, param) {
|
|
@@ -95,7 +101,7 @@ export class TicketService {
|
|
|
95
101
|
return null;
|
|
96
102
|
}
|
|
97
103
|
}
|
|
98
|
-
|
|
104
|
+
getTicketDateTimeByQRContent(content, date, time) {
|
|
99
105
|
const _date = content.slice(date.start, date.end);
|
|
100
106
|
const _time = content.slice(time.start, time.end);
|
|
101
107
|
return moment({
|
|
@@ -107,27 +113,36 @@ export class TicketService {
|
|
|
107
113
|
second: _time.length === 6 ? +_time.slice(4, 6) : 0
|
|
108
114
|
});
|
|
109
115
|
}
|
|
110
|
-
|
|
111
|
-
//
|
|
112
|
-
//
|
|
113
|
-
//
|
|
114
|
-
//
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
//
|
|
120
|
-
//
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
116
|
+
getTicketDateTimeInSecondsByBarcodeContent(content, location, timezone) {
|
|
117
|
+
// Barcode time encoding (based on SQL Date2Number / Number2Date):
|
|
118
|
+
// The original datetime is converted in SQL to total seconds since ~2000-01-01,
|
|
119
|
+
// then effectively to total minutes. The barcode stores only the LAST 6 digits
|
|
120
|
+
// of those minutes (totalMinutes % 1,000,000).
|
|
121
|
+
//
|
|
122
|
+
// Because of this truncation, the value wraps every 1,000,000 minutes (~694 days),
|
|
123
|
+
// so the same value can represent multiple dates.
|
|
124
|
+
//
|
|
125
|
+
// To reconstruct the correct datetime, we rebuild possible full values and
|
|
126
|
+
// select the one closest to "now".
|
|
127
|
+
const MINUTE_CYCLE = 1_000_000;
|
|
128
|
+
const CYCLE_SEARCH_RADIUS = 1;
|
|
129
|
+
const truncatedMinutesFromBarcode = Number(content.slice(location.start, location.end));
|
|
130
|
+
const now = moment.tz(timezone).seconds(0).milliseconds(0);
|
|
131
|
+
const barcodeBaseDate = moment.tz(DEFAULT_SMART_PARK_DATE_TIME, 'YYYY-MM-DD HH:mm:ss', timezone);
|
|
132
|
+
const currentTotalMinutesFromBase = now.diff(barcodeBaseDate, 'minutes');
|
|
133
|
+
const estimatedCycleIndex = Math.floor(currentTotalMinutesFromBase / MINUTE_CYCLE);
|
|
134
|
+
let resolvedTicketDateTime = barcodeBaseDate.clone();
|
|
135
|
+
let smallestDifferenceFromNow = Number.MAX_SAFE_INTEGER;
|
|
136
|
+
for (let cycleIndex = estimatedCycleIndex - CYCLE_SEARCH_RADIUS; cycleIndex <= estimatedCycleIndex + CYCLE_SEARCH_RADIUS; cycleIndex++) {
|
|
137
|
+
const fullMinutesFromBase = cycleIndex * MINUTE_CYCLE + truncatedMinutesFromBarcode;
|
|
138
|
+
const candidateDateTime = barcodeBaseDate.clone().add(fullMinutesFromBase, 'minutes');
|
|
139
|
+
const differenceFromNow = Math.abs(candidateDateTime.diff(now));
|
|
140
|
+
if (differenceFromNow < smallestDifferenceFromNow) {
|
|
141
|
+
smallestDifferenceFromNow = differenceFromNow;
|
|
142
|
+
resolvedTicketDateTime = candidateDateTime;
|
|
143
|
+
}
|
|
124
144
|
}
|
|
125
|
-
|
|
126
|
-
var curTimeTicketFormat = curTimeInMinutes % 1000000;
|
|
127
|
-
if (ticketTime > curTimeTicketFormat)
|
|
128
|
-
curTimeTicketFormat += 1000000;
|
|
129
|
-
const ticketDateTimeInMinutes = (defaultSmartParkDateTime / 60000) + curTimeInMinutes - (curTimeTicketFormat - ticketTime);
|
|
130
|
-
return moment(new Date(ticketDateTimeInMinutes * 60000));
|
|
145
|
+
return resolvedTicketDateTime.diff(barcodeBaseDate, 'seconds');
|
|
131
146
|
}
|
|
132
147
|
navigateToTicket(ticketNum, guid, parkId, activatedRoute, ticketDateTimeInSeconds = null) {
|
|
133
148
|
const tibaKeys = this.store.selectSnapshot(FacilityState.tibaKeys);
|
|
@@ -150,4 +165,4 @@ export class TicketService {
|
|
|
150
165
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TicketService, decorators: [{
|
|
151
166
|
type: Injectable
|
|
152
167
|
}], ctorParameters: () => [{ type: i0.Injector }] });
|
|
153
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket.service.js","sourceRoot":"","sources":["../../../../../projects/client-shared-lib/src/libraries/services/ticket.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Injector } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { Store } from '@ngxs/store';\nimport moment from 'moment';\nimport { Observable } from 'rxjs';\nimport { CommandNotificationConfig } from '../components/notification/command-notification/command-notification-config';\nimport { CommandNotificationService } from '../components/notification/command-notification/command-notification.service';\nimport { NotificationClass } from '../components/notification/notification-config';\nimport { BarcodeTypeLength } from '../enums/barcode-type.enum';\nimport { TracePayload } from '../interfaces/analytics.interface';\nimport { CmdDataDto } from '../interfaces/cmd-data.interface';\nimport { ConfigData } from '../interfaces/config-data.interface';\nimport { TicketIdentifier } from '../interfaces/ticket.interface';\nimport { FacilityState } from '../modules/facility/facility.state';\nimport { ParkerState } from '../modules/parker/parker.state';\nimport { RemoteCommandTypes } from '../service-proxy/cloud-service-proxies';\nimport { RemoteCommandBase } from '../service-proxy/rpc-service-proxies';\nimport { parseDateTimeToSparkParkTime } from '../utils/date.util';\nimport { RemoteCommandService } from './remote-command.service';\n\n@Injectable()\nexport abstract class TicketService {\n\n  protected store: Store;\n  protected commandNotificationService: CommandNotificationService;\n  protected remoteCommandService: RemoteCommandService;\n  protected router: Router;\n\n  constructor(public injector: Injector) {\n    this.store = injector.get(Store);\n    this.commandNotificationService = injector.get(CommandNotificationService);\n    this.remoteCommandService = injector.get(RemoteCommandService);\n    this.router = injector.get(Router);\n  }\n\n  sendCommand(commandType: RemoteCommandTypes, options: any = null, analyticsPayload: TracePayload = {}): Observable<{ payloadData: RemoteCommandBase, response: any }> {\n    const guid = this.store.selectSnapshot(FacilityState.guid);\n    const ticketIdentifier = this.store.selectSnapshot(ParkerState.ticketIdentifier);\n    const data: ConfigData = { commandType, ticketIdentifier, options };\n    const cmdDataDto$ = this.getTicketRemoteCommandData(commandType, data, ticketIdentifier, options);\n    return this.remoteCommandService.send(cmdDataDto$, guid, null, analyticsPayload);\n  }\n\n  sendCommandWithSnackData(commandType: RemoteCommandTypes, options: any = null, analyticsPayload: TracePayload = {}): Observable<{ payloadData: RemoteCommandBase, response: any }> {\n    const ticketIdentifier = this.store.selectSnapshot(ParkerState.ticketIdentifier);\n    const entityTypeTranslate = options?.entityTypeTranslate || ticketIdentifier?.entityTypeTranslate;\n    const entityDisplayName = options?.entityDisplayName || ticketIdentifier?.entityDisplayName;\n\n    const config: CommandNotificationConfig = {\n      messages: [{\n        value: `${entityTypeTranslate} ${entityDisplayName}`,\n        class: entityDisplayName ? NotificationClass.subTitle : NotificationClass.none,\n      }],\n      command: this.remoteCommandService.getRemoteCommandText(commandType),\n    };\n\n    return this.sendCommand(commandType, options, analyticsPayload)\n      .pipe(this.commandNotificationService.display(config));\n  }\n\n  protected abstract getTicketRemoteCommandData(commandType: RemoteCommandTypes, data: ConfigData, ticketIdentifier?: TicketIdentifier, options?: any): Observable<CmdDataDto>;\n\n  extractTicketInfo(content: string): { ticketNum: string, ticketDateTimeInSeconds: number } {\n    let ticketNum: string;\n    let ticketDateTime: moment.Moment;\n    try {\n      // To determine what type, check content length\n      switch (content.length) {\n        case BarcodeTypeLength.QR1:\n        case BarcodeTypeLength.QR2:\n          ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14\n          // Location of ticket date; from 14 to 20\n          // Location of ticket time; from 20 to 24\n          ticketDateTime = this.getTicketDateTimeByContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });\n          break;\n        case BarcodeTypeLength.QR4:\n          ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.\n          // Location of ticket date; from 16 to 22\n          // Location of ticket time; from 22 to 28\n          ticketDateTime = this.getTicketDateTimeByContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });\n          break;\n        case BarcodeTypeLength.Barcode1:\n          ticketNum = content.slice(7, 14); // Location of ticket number; from 7 to 14\n          // Location of ticket time info; from 1 to 7\n          ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 1, end: 7 });\n          break;\n        case BarcodeTypeLength.Barcode2:\n          ticketNum = content.slice(13, 22); // Location of ticket number; from 14 to 22\n          // Location of ticket time info; from 7 to 13\n          ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 7, end: 13 });\n          break;\n        default:\n          {\n            const qValue = this.extractQueryParam(content, 'q');\n            if (qValue !== null) {\n              return this.extractTicketInfo(qValue);\n            }\n          }\n          break;\n      }\n    } catch (_) { }\n\n    const ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());\n    return { ticketNum, ticketDateTimeInSeconds };\n  }\n\n  private extractQueryParam(url: string, param: string): string | null {\n    if (!url || !param) {\n      return null;\n    }\n  \n    try {\n      const parsedUrl = new URL(url);\n      return parsedUrl.searchParams.get(param);\n    } catch {\n      // In case the string is not a valid URL\n      return null;\n    }\n  }\n\n  private getTicketDateTimeByContent(content: string, date: { start: number, end: number }, time: { start: number, end: number }): moment.Moment {\n    const _date = content.slice(date.start, date.end);\n    const _time = content.slice(time.start, time.end);\n    return moment({\n      year: 2000 + +_date.slice(4, 6),\n      month: +_date.slice(2, 4) - 1,\n      day: +_date.slice(0, 2),\n      hour: +_time.slice(0, 2),\n      minute: +_time.slice(2, 4),\n      second: _time.length === 6 ? +_time.slice(4, 6) : 0\n    });\n  }\n\n  private getTicketDateTimeByTicketTime(content: string, location: { start: number, end: number }): moment.Moment {\n    // multiple 60000:\n    // getTime method return date in miliseconds\n    // to convert to seconds multiple by 10000\n    // to convert seconds to minutes multiple by 60\n\n    const ticketTime = +content.slice(location.start, location.end);\n    var curTimeDateTime = new Date();\n    curTimeDateTime.setSeconds(0);\n\n    let defaultSmartParkDateTime = new Date(2000, 0, 1, 0, 0, 0).getTime();\n    // The default smart park date time is always: Sat Jan 01 2000 00:00:00 GMT+0200 (Israel Standard Time)\n    // If the current time date is in DST (GMT+0300) then to avoid incorrect results we need to remove 1 hour\n    if (moment(curTimeDateTime).isDST()) {\n      // 60 Minutes ( 1 hour ) in miliseconds equal 3600000\n      defaultSmartParkDateTime -= 3600000;\n    }\n\n    var curTimeInMinutes = (curTimeDateTime.getTime() - defaultSmartParkDateTime) / 60000;\n    var curTimeTicketFormat = curTimeInMinutes % 1000000;\n    if (ticketTime > curTimeTicketFormat) curTimeTicketFormat += 1000000;\n\n    const ticketDateTimeInMinutes = (defaultSmartParkDateTime / 60000) + curTimeInMinutes - (curTimeTicketFormat - ticketTime);\n    return moment(new Date(ticketDateTimeInMinutes * 60000));\n  }\n\n  navigateToTicket(ticketNum: string, guid: string, parkId: number, activatedRoute: ActivatedRoute, ticketDateTimeInSeconds: number = null) {\n    const tibaKeys = this.store.selectSnapshot(FacilityState.tibaKeys);\n    const commands: any[] = [\n      tibaKeys[guid],\n      parkId,\n      ticketNum,\n    ];\n    if (ticketDateTimeInSeconds) {\n      commands.push(ticketDateTimeInSeconds);\n    }\n    this.router.navigate(commands, { relativeTo: activatedRoute });\n  }\n\n  scanTicket(ticketNum: string, guid: string, parkId: number, activatedRoute: ActivatedRoute, ticketDateTimeInSeconds: number) {\n    this.navigateToTicket(ticketNum, guid, parkId, activatedRoute, ticketDateTimeInSeconds);\n  }\n}\n"]}
|
|
168
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket.service.js","sourceRoot":"","sources":["../../../../../projects/client-shared-lib/src/libraries/services/ticket.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Injector } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { Store } from '@ngxs/store';\nimport moment from 'moment';\nimport { Observable } from 'rxjs';\nimport { CommandNotificationConfig } from '../components/notification/command-notification/command-notification-config';\nimport { CommandNotificationService } from '../components/notification/command-notification/command-notification.service';\nimport { NotificationClass } from '../components/notification/notification-config';\nimport { DEFAULT_SMART_PARK_DATE_TIME } from '../constants/global-constants';\nimport { BarcodeTypeLength } from '../enums/barcode-type.enum';\nimport { TracePayload } from '../interfaces/analytics.interface';\nimport { CmdDataDto } from '../interfaces/cmd-data.interface';\nimport { ConfigData } from '../interfaces/config-data.interface';\nimport { TicketIdentifier } from '../interfaces/ticket.interface';\nimport { FacilityState } from '../modules/facility/facility.state';\nimport { ParkerState } from '../modules/parker/parker.state';\nimport { RemoteCommandTypes } from '../service-proxy/cloud-service-proxies';\nimport { RemoteCommandBase } from '../service-proxy/rpc-service-proxies';\nimport { parseDateTimeToSparkParkTime } from '../utils/date.util';\nimport { RemoteCommandService } from './remote-command.service';\n\n@Injectable()\nexport abstract class TicketService {\n\n  protected store: Store;\n  protected commandNotificationService: CommandNotificationService;\n  protected remoteCommandService: RemoteCommandService;\n  protected router: Router;\n\n  constructor(public injector: Injector) {\n    this.store = injector.get(Store);\n    this.commandNotificationService = injector.get(CommandNotificationService);\n    this.remoteCommandService = injector.get(RemoteCommandService);\n    this.router = injector.get(Router);\n  }\n\n  sendCommand(commandType: RemoteCommandTypes, options: any = null, analyticsPayload: TracePayload = {}): Observable<{ payloadData: RemoteCommandBase, response: any }> {\n    const guid = this.store.selectSnapshot(FacilityState.guid);\n    const ticketIdentifier = this.store.selectSnapshot(ParkerState.ticketIdentifier);\n    const data: ConfigData = { commandType, ticketIdentifier, options };\n    const cmdDataDto$ = this.getTicketRemoteCommandData(commandType, data, ticketIdentifier, options);\n    return this.remoteCommandService.send(cmdDataDto$, guid, null, analyticsPayload);\n  }\n\n  sendCommandWithSnackData(commandType: RemoteCommandTypes, options: any = null, analyticsPayload: TracePayload = {}): Observable<{ payloadData: RemoteCommandBase, response: any }> {\n    const ticketIdentifier = this.store.selectSnapshot(ParkerState.ticketIdentifier);\n    const entityTypeTranslate = options?.entityTypeTranslate || ticketIdentifier?.entityTypeTranslate;\n    const entityDisplayName = options?.entityDisplayName || ticketIdentifier?.entityDisplayName;\n\n    const config: CommandNotificationConfig = {\n      messages: [{\n        value: `${entityTypeTranslate} ${entityDisplayName}`,\n        class: entityDisplayName ? NotificationClass.subTitle : NotificationClass.none,\n      }],\n      command: this.remoteCommandService.getRemoteCommandText(commandType),\n    };\n\n    return this.sendCommand(commandType, options, analyticsPayload)\n      .pipe(this.commandNotificationService.display(config));\n  }\n\n  protected abstract getTicketRemoteCommandData(commandType: RemoteCommandTypes, data: ConfigData, ticketIdentifier?: TicketIdentifier, options?: any): Observable<CmdDataDto>;\n\n  extractTicketInfo(content: string, timezone: string): { ticketNum: string, ticketDateTimeInSeconds: number } {\n    let ticketNum: string;\n    let ticketDateTimeInSeconds: number;\n    try {\n      // To determine what type, check content length\n      switch (content.length) {\n        case BarcodeTypeLength.QR1:\n        case BarcodeTypeLength.QR2:\n          {\n            ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14\n            // Location of ticket date; from 14 to 20\n            // Location of ticket time; from 20 to 24\n            const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });\n            ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());\n          }\n          break;\n        case BarcodeTypeLength.QR4:\n          {\n            ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.\n            // Location of ticket date; from 16 to 22\n            // Location of ticket time; from 22 to 28\n            const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });\n            ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());\n          }\n          break;\n        case BarcodeTypeLength.Barcode1:\n          ticketNum = content.slice(7, 14); // Location of ticket number; from 7 to 14\n          // Location of ticket time info; from 1 to 7\n          ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 1, end: 7 }, timezone);\n          break;\n        case BarcodeTypeLength.Barcode2:\n          ticketNum = content.slice(13, 22); // Location of ticket number; from 14 to 22\n          // Location of ticket time info; from 7 to 13\n          ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 7, end: 13 }, timezone);\n          break;\n        default:\n          {\n            const qValue = this.extractQueryParam(content, 'q');\n            if (qValue !== null && qValue !== content) {\n              return this.extractTicketInfo(qValue, timezone);\n            }\n          }\n          break;\n      }\n    } catch (_) { }\n\n    return { ticketNum, ticketDateTimeInSeconds };\n  }\n\n  private extractQueryParam(url: string, param: string): string | null {\n    if (!url || !param) {\n      return null;\n    }\n  \n    try {\n      const parsedUrl = new URL(url);\n      return parsedUrl.searchParams.get(param);\n    } catch {\n      // In case the string is not a valid URL\n      return null;\n    }\n  }\n\n  private getTicketDateTimeByQRContent(content: string, date: { start: number, end: number }, time: { start: number, end: number }): moment.Moment {\n    const _date = content.slice(date.start, date.end);\n    const _time = content.slice(time.start, time.end);\n    return moment({\n      year: 2000 + +_date.slice(4, 6),\n      month: +_date.slice(2, 4) - 1,\n      day: +_date.slice(0, 2),\n      hour: +_time.slice(0, 2),\n      minute: +_time.slice(2, 4),\n      second: _time.length === 6 ? +_time.slice(4, 6) : 0\n    });\n  }\n\n  private getTicketDateTimeInSecondsByBarcodeContent(content: string, location: { start: number, end: number; }, timezone: string): number {\n    // Barcode time encoding (based on SQL Date2Number / Number2Date):\n    // The original datetime is converted in SQL to total seconds since ~2000-01-01,\n    // then effectively to total minutes. The barcode stores only the LAST 6 digits\n    // of those minutes (totalMinutes % 1,000,000).\n    //\n    // Because of this truncation, the value wraps every 1,000,000 minutes (~694 days),\n    // so the same value can represent multiple dates.\n    //\n    // To reconstruct the correct datetime, we rebuild possible full values and\n    // select the one closest to \"now\".\n\n    const MINUTE_CYCLE  = 1_000_000;\n    const CYCLE_SEARCH_RADIUS = 1;\n\n    const truncatedMinutesFromBarcode = Number(content.slice(location.start, location.end));\n    \n    const now = moment.tz(timezone).seconds(0).milliseconds(0);\n    const barcodeBaseDate = moment.tz(DEFAULT_SMART_PARK_DATE_TIME, 'YYYY-MM-DD HH:mm:ss', timezone);\n    \n    const currentTotalMinutesFromBase = now.diff(barcodeBaseDate, 'minutes');\n    const estimatedCycleIndex = Math.floor(currentTotalMinutesFromBase / MINUTE_CYCLE );\n    \n    let resolvedTicketDateTime = barcodeBaseDate.clone();\n    let smallestDifferenceFromNow = Number.MAX_SAFE_INTEGER;\n    \n    for (let cycleIndex = estimatedCycleIndex - CYCLE_SEARCH_RADIUS; cycleIndex <= estimatedCycleIndex + CYCLE_SEARCH_RADIUS; cycleIndex++) {\n      const fullMinutesFromBase = cycleIndex * MINUTE_CYCLE  + truncatedMinutesFromBarcode ;\n      const candidateDateTime = barcodeBaseDate.clone().add(fullMinutesFromBase, 'minutes');\n      const differenceFromNow  = Math.abs(candidateDateTime.diff(now));\n    \n      if (differenceFromNow  < smallestDifferenceFromNow) {\n        smallestDifferenceFromNow = differenceFromNow ;\n        resolvedTicketDateTime = candidateDateTime;\n      }\n    }\n    \n    return resolvedTicketDateTime.diff(barcodeBaseDate, 'seconds');\n  }\n\n  navigateToTicket(ticketNum: string, guid: string, parkId: number, activatedRoute: ActivatedRoute, ticketDateTimeInSeconds: number = null) {\n    const tibaKeys = this.store.selectSnapshot(FacilityState.tibaKeys);\n    const commands: any[] = [\n      tibaKeys[guid],\n      parkId,\n      ticketNum,\n    ];\n    if (ticketDateTimeInSeconds) {\n      commands.push(ticketDateTimeInSeconds);\n    }\n    this.router.navigate(commands, { relativeTo: activatedRoute });\n  }\n\n  scanTicket(ticketNum: string, guid: string, parkId: number, activatedRoute: ActivatedRoute, ticketDateTimeInSeconds: number) {\n    this.navigateToTicket(ticketNum, guid, parkId, activatedRoute, ticketDateTimeInSeconds);\n  }\n}\n"]}
|
|
@@ -113920,47 +113920,52 @@ class TicketService {
|
|
|
113920
113920
|
return this.sendCommand(commandType, options, analyticsPayload)
|
|
113921
113921
|
.pipe(this.commandNotificationService.display(config));
|
|
113922
113922
|
}
|
|
113923
|
-
extractTicketInfo(content) {
|
|
113923
|
+
extractTicketInfo(content, timezone) {
|
|
113924
113924
|
let ticketNum;
|
|
113925
|
-
let
|
|
113925
|
+
let ticketDateTimeInSeconds;
|
|
113926
113926
|
try {
|
|
113927
113927
|
// To determine what type, check content length
|
|
113928
113928
|
switch (content.length) {
|
|
113929
113929
|
case BarcodeTypeLength.QR1:
|
|
113930
113930
|
case BarcodeTypeLength.QR2:
|
|
113931
|
-
|
|
113932
|
-
|
|
113933
|
-
|
|
113934
|
-
|
|
113931
|
+
{
|
|
113932
|
+
ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14
|
|
113933
|
+
// Location of ticket date; from 14 to 20
|
|
113934
|
+
// Location of ticket time; from 20 to 24
|
|
113935
|
+
const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });
|
|
113936
|
+
ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
113937
|
+
}
|
|
113935
113938
|
break;
|
|
113936
113939
|
case BarcodeTypeLength.QR4:
|
|
113937
|
-
|
|
113938
|
-
|
|
113939
|
-
|
|
113940
|
-
|
|
113940
|
+
{
|
|
113941
|
+
ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.
|
|
113942
|
+
// Location of ticket date; from 16 to 22
|
|
113943
|
+
// Location of ticket time; from 22 to 28
|
|
113944
|
+
const ticketDateTime = this.getTicketDateTimeByQRContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });
|
|
113945
|
+
ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
113946
|
+
}
|
|
113941
113947
|
break;
|
|
113942
113948
|
case BarcodeTypeLength.Barcode1:
|
|
113943
113949
|
ticketNum = content.slice(7, 14); // Location of ticket number; from 7 to 14
|
|
113944
113950
|
// Location of ticket time info; from 1 to 7
|
|
113945
|
-
|
|
113951
|
+
ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 1, end: 7 }, timezone);
|
|
113946
113952
|
break;
|
|
113947
113953
|
case BarcodeTypeLength.Barcode2:
|
|
113948
113954
|
ticketNum = content.slice(13, 22); // Location of ticket number; from 14 to 22
|
|
113949
113955
|
// Location of ticket time info; from 7 to 13
|
|
113950
|
-
|
|
113956
|
+
ticketDateTimeInSeconds = this.getTicketDateTimeInSecondsByBarcodeContent(content, { start: 7, end: 13 }, timezone);
|
|
113951
113957
|
break;
|
|
113952
113958
|
default:
|
|
113953
113959
|
{
|
|
113954
113960
|
const qValue = this.extractQueryParam(content, 'q');
|
|
113955
|
-
if (qValue !== null) {
|
|
113956
|
-
return this.extractTicketInfo(qValue);
|
|
113961
|
+
if (qValue !== null && qValue !== content) {
|
|
113962
|
+
return this.extractTicketInfo(qValue, timezone);
|
|
113957
113963
|
}
|
|
113958
113964
|
}
|
|
113959
113965
|
break;
|
|
113960
113966
|
}
|
|
113961
113967
|
}
|
|
113962
113968
|
catch (_) { }
|
|
113963
|
-
const ticketDateTimeInSeconds = parseDateTimeToSparkParkTime(ticketDateTime?.toDate());
|
|
113964
113969
|
return { ticketNum, ticketDateTimeInSeconds };
|
|
113965
113970
|
}
|
|
113966
113971
|
extractQueryParam(url, param) {
|
|
@@ -113976,7 +113981,7 @@ class TicketService {
|
|
|
113976
113981
|
return null;
|
|
113977
113982
|
}
|
|
113978
113983
|
}
|
|
113979
|
-
|
|
113984
|
+
getTicketDateTimeByQRContent(content, date, time) {
|
|
113980
113985
|
const _date = content.slice(date.start, date.end);
|
|
113981
113986
|
const _time = content.slice(time.start, time.end);
|
|
113982
113987
|
return moment({
|
|
@@ -113988,27 +113993,36 @@ class TicketService {
|
|
|
113988
113993
|
second: _time.length === 6 ? +_time.slice(4, 6) : 0
|
|
113989
113994
|
});
|
|
113990
113995
|
}
|
|
113991
|
-
|
|
113992
|
-
//
|
|
113993
|
-
//
|
|
113994
|
-
//
|
|
113995
|
-
//
|
|
113996
|
-
|
|
113997
|
-
|
|
113998
|
-
|
|
113999
|
-
|
|
114000
|
-
//
|
|
114001
|
-
//
|
|
114002
|
-
|
|
114003
|
-
|
|
114004
|
-
|
|
114005
|
-
|
|
114006
|
-
|
|
114007
|
-
|
|
114008
|
-
|
|
114009
|
-
|
|
114010
|
-
|
|
114011
|
-
|
|
113996
|
+
getTicketDateTimeInSecondsByBarcodeContent(content, location, timezone) {
|
|
113997
|
+
// Barcode time encoding (based on SQL Date2Number / Number2Date):
|
|
113998
|
+
// The original datetime is converted in SQL to total seconds since ~2000-01-01,
|
|
113999
|
+
// then effectively to total minutes. The barcode stores only the LAST 6 digits
|
|
114000
|
+
// of those minutes (totalMinutes % 1,000,000).
|
|
114001
|
+
//
|
|
114002
|
+
// Because of this truncation, the value wraps every 1,000,000 minutes (~694 days),
|
|
114003
|
+
// so the same value can represent multiple dates.
|
|
114004
|
+
//
|
|
114005
|
+
// To reconstruct the correct datetime, we rebuild possible full values and
|
|
114006
|
+
// select the one closest to "now".
|
|
114007
|
+
const MINUTE_CYCLE = 1_000_000;
|
|
114008
|
+
const CYCLE_SEARCH_RADIUS = 1;
|
|
114009
|
+
const truncatedMinutesFromBarcode = Number(content.slice(location.start, location.end));
|
|
114010
|
+
const now = moment.tz(timezone).seconds(0).milliseconds(0);
|
|
114011
|
+
const barcodeBaseDate = moment.tz(DEFAULT_SMART_PARK_DATE_TIME, 'YYYY-MM-DD HH:mm:ss', timezone);
|
|
114012
|
+
const currentTotalMinutesFromBase = now.diff(barcodeBaseDate, 'minutes');
|
|
114013
|
+
const estimatedCycleIndex = Math.floor(currentTotalMinutesFromBase / MINUTE_CYCLE);
|
|
114014
|
+
let resolvedTicketDateTime = barcodeBaseDate.clone();
|
|
114015
|
+
let smallestDifferenceFromNow = Number.MAX_SAFE_INTEGER;
|
|
114016
|
+
for (let cycleIndex = estimatedCycleIndex - CYCLE_SEARCH_RADIUS; cycleIndex <= estimatedCycleIndex + CYCLE_SEARCH_RADIUS; cycleIndex++) {
|
|
114017
|
+
const fullMinutesFromBase = cycleIndex * MINUTE_CYCLE + truncatedMinutesFromBarcode;
|
|
114018
|
+
const candidateDateTime = barcodeBaseDate.clone().add(fullMinutesFromBase, 'minutes');
|
|
114019
|
+
const differenceFromNow = Math.abs(candidateDateTime.diff(now));
|
|
114020
|
+
if (differenceFromNow < smallestDifferenceFromNow) {
|
|
114021
|
+
smallestDifferenceFromNow = differenceFromNow;
|
|
114022
|
+
resolvedTicketDateTime = candidateDateTime;
|
|
114023
|
+
}
|
|
114024
|
+
}
|
|
114025
|
+
return resolvedTicketDateTime.diff(barcodeBaseDate, 'seconds');
|
|
114012
114026
|
}
|
|
114013
114027
|
navigateToTicket(ticketNum, guid, parkId, activatedRoute, ticketDateTimeInSeconds = null) {
|
|
114014
114028
|
const tibaKeys = this.store.selectSnapshot(FacilityState.tibaKeys);
|