@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.
@@ -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 ticketDateTime;
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
- ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14
51
- // Location of ticket date; from 14 to 20
52
- // Location of ticket time; from 20 to 24
53
- ticketDateTime = this.getTicketDateTimeByContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });
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
- ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.
57
- // Location of ticket date; from 16 to 22
58
- // Location of ticket time; from 22 to 28
59
- ticketDateTime = this.getTicketDateTimeByContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });
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
- ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 1, end: 7 });
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
- ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 7, end: 13 });
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
- getTicketDateTimeByContent(content, date, time) {
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
- getTicketDateTimeByTicketTime(content, location) {
111
- // multiple 60000:
112
- // getTime method return date in miliseconds
113
- // to convert to seconds multiple by 10000
114
- // to convert seconds to minutes multiple by 60
115
- const ticketTime = +content.slice(location.start, location.end);
116
- var curTimeDateTime = new Date();
117
- curTimeDateTime.setSeconds(0);
118
- let defaultSmartParkDateTime = new Date(2000, 0, 1, 0, 0, 0).getTime();
119
- // The default smart park date time is always: Sat Jan 01 2000 00:00:00 GMT+0200 (Israel Standard Time)
120
- // If the current time date is in DST (GMT+0300) then to avoid incorrect results we need to remove 1 hour
121
- if (moment(curTimeDateTime).isDST()) {
122
- // 60 Minutes ( 1 hour ) in miliseconds equal 3600000
123
- defaultSmartParkDateTime -= 3600000;
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
- var curTimeInMinutes = (curTimeDateTime.getTime() - defaultSmartParkDateTime) / 60000;
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 ticketDateTime;
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
- ticketNum = content.slice(6, 14); // Location of ticket number; from 7 to 14
113932
- // Location of ticket date; from 14 to 20
113933
- // Location of ticket time; from 20 to 24
113934
- ticketDateTime = this.getTicketDateTimeByContent(content, { start: 14, end: 20 }, { start: 20, end: 24 });
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
- ticketNum = content.slice(7, 16); // Location of ticket number; from 8 to 16.
113938
- // Location of ticket date; from 16 to 22
113939
- // Location of ticket time; from 22 to 28
113940
- ticketDateTime = this.getTicketDateTimeByContent(content, { start: 16, end: 22 }, { start: 22, end: 28 });
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
- ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 1, end: 7 });
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
- ticketDateTime = this.getTicketDateTimeByTicketTime(content, { start: 7, end: 13 });
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
- getTicketDateTimeByContent(content, date, time) {
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
- getTicketDateTimeByTicketTime(content, location) {
113992
- // multiple 60000:
113993
- // getTime method return date in miliseconds
113994
- // to convert to seconds multiple by 10000
113995
- // to convert seconds to minutes multiple by 60
113996
- const ticketTime = +content.slice(location.start, location.end);
113997
- var curTimeDateTime = new Date();
113998
- curTimeDateTime.setSeconds(0);
113999
- let defaultSmartParkDateTime = new Date(2000, 0, 1, 0, 0, 0).getTime();
114000
- // The default smart park date time is always: Sat Jan 01 2000 00:00:00 GMT+0200 (Israel Standard Time)
114001
- // If the current time date is in DST (GMT+0300) then to avoid incorrect results we need to remove 1 hour
114002
- if (moment(curTimeDateTime).isDST()) {
114003
- // 60 Minutes ( 1 hour ) in miliseconds equal 3600000
114004
- defaultSmartParkDateTime -= 3600000;
114005
- }
114006
- var curTimeInMinutes = (curTimeDateTime.getTime() - defaultSmartParkDateTime) / 60000;
114007
- var curTimeTicketFormat = curTimeInMinutes % 1000000;
114008
- if (ticketTime > curTimeTicketFormat)
114009
- curTimeTicketFormat += 1000000;
114010
- const ticketDateTimeInMinutes = (defaultSmartParkDateTime / 60000) + curTimeInMinutes - (curTimeTicketFormat - ticketTime);
114011
- return moment(new Date(ticketDateTimeInMinutes * 60000));
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);