navigo.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  1. #include "navigo.h"
  2. #include "navigo_lists.h"
  3. #include "../../../metroflip_i.h"
  4. const char* get_navigo_transport_type(int type) {
  5. switch(type) {
  6. case URBAN_BUS:
  7. return "Urban Bus";
  8. case INTERURBAN_BUS:
  9. return "Interurban Bus";
  10. case METRO:
  11. return "Metro";
  12. case TRAM:
  13. return "Tram";
  14. case COMMUTER_TRAIN:
  15. return "Train";
  16. case PARKING:
  17. return "Parking";
  18. case EXPRESS_COMMUTER_TRAIN:
  19. return "TER";
  20. default:
  21. return "Unknown";
  22. }
  23. }
  24. const char* get_navigo_service_provider(int provider) {
  25. switch(provider) {
  26. case NAVIGO_PROVIDER_SNCF:
  27. return "SNCF";
  28. case NAVIGO_PROVIDER_RATP:
  29. return "RATP";
  30. case 4:
  31. case 10:
  32. return "IDF Mobilites";
  33. case NAVIGO_PROVIDER_ORA:
  34. return "ORA";
  35. case NAVIGO_PROVIDER_VEOLIA_CSO:
  36. return "CSO (VEOLIA)";
  37. case NAVIGO_PROVIDER_VEOLIA_RBUS:
  38. return "R'Bus (VEOLIA)";
  39. case NAVIGO_PROVIDER_PHEBUS:
  40. return "Phebus";
  41. case NAVIGO_PROVIDER_RATP_VEOLIA_NANTERRE:
  42. return "RATP (Veolia Transport Nanterre)";
  43. default: {
  44. char* provider_str = malloc(6 * sizeof(char));
  45. snprintf(provider_str, 6, "%d", provider);
  46. return provider_str;
  47. }
  48. }
  49. }
  50. const char* get_navigo_type(int type) {
  51. switch(type) {
  52. case NAVIGO_EASY:
  53. return "Navigo Easy";
  54. case NAVIGO_DECOUVERTE:
  55. return "Navigo Decouverte";
  56. case NAVIGO_STANDARD:
  57. return "Navigo Standard";
  58. case NAVIGO_INTEGRAL:
  59. return "Navigo Integral";
  60. case IMAGINE_R:
  61. return "Imagine R";
  62. default:
  63. return "Navigo";
  64. }
  65. }
  66. const char* get_navigo_tariff(int tariff) {
  67. switch(tariff) {
  68. case 0x0000:
  69. return "Navigo Mois";
  70. case 0x0001:
  71. return "Navigo Semaine";
  72. case 0x0002:
  73. return "Navigo Annuel";
  74. case 0x0003:
  75. return "Navigo Jour";
  76. case 0x0004:
  77. return "Imagine R Junior";
  78. case 0x0005:
  79. return "Imagine R Etudiant";
  80. case 0x000D:
  81. return "Navigo Jeunes Week-end";
  82. case 0x0015:
  83. return "Paris-Visite"; // Theoric
  84. case 0x1000:
  85. return "Navigo Liberte+";
  86. case 0x4000:
  87. return "Navigo Mois 75%%";
  88. case 0x4001:
  89. return "Navigo Semaine 75%%";
  90. case 0x4015:
  91. return "Paris-Visite (Enfant)"; // Theoric
  92. case 0x5000:
  93. return "Tickets T+";
  94. case 0x5004:
  95. return "Tickets OrlyBus"; // Theoric
  96. case 0x5005:
  97. return "Tickets RoissyBus"; // Theoric
  98. case 0x5006:
  99. return "Bus-Tram"; // Theoric
  100. case 0x5008:
  101. return "Metro-Train-RER"; // Theoric
  102. case 0x500b:
  103. return "Paris <> Aeroports"; // Theoric
  104. case 0x5010:
  105. return "Tickets T+ (Reduit)"; // Theoric
  106. case 0x5016:
  107. return "Bus-Tram (Reduit)"; // Theoric
  108. case 0x5018:
  109. return "Metro-Train-RER (Reduit)"; // Theoric
  110. case 0x501b:
  111. return "Paris <> Aeroports (Reduit)"; // Theoric
  112. case 0x8003:
  113. return "Navigo Solidarite Gratuit";
  114. default: {
  115. char* tariff_str = malloc(6 * sizeof(char));
  116. snprintf(tariff_str, 6, "%d", tariff);
  117. return tariff_str;
  118. }
  119. }
  120. }
  121. bool is_ticket_count_available(int tariff) {
  122. return tariff >= 0x5000 && tariff <= 0x501b;
  123. }
  124. const char* get_pay_method(int pay_method) {
  125. switch(pay_method) {
  126. case 0x30:
  127. return "Apple Pay/Google Pay";
  128. case 0x80:
  129. return "Debit PME";
  130. case 0x90:
  131. return "Cash";
  132. case 0xA0:
  133. return "Mobility Check";
  134. case 0xB3:
  135. return "Payment Card";
  136. case 0xA4:
  137. return "Check";
  138. case 0xA5:
  139. return "Vacation Check";
  140. case 0xB7:
  141. return "Telepayment";
  142. case 0xD0:
  143. return "Remote Payment";
  144. case 0xD7:
  145. return "Voucher, Prepayment, Exchange Voucher, Travel Voucher";
  146. case 0xD9:
  147. return "Discount Voucher";
  148. default:
  149. return "Unknown";
  150. }
  151. }
  152. const char* get_zones(int* zones) {
  153. if(zones[0] && zones[4]) {
  154. return "All Zones (1-5)";
  155. } else if(zones[0] && zones[3]) {
  156. return "Zones 1-4";
  157. } else if(zones[0] && zones[2]) {
  158. return "Zones 1-3";
  159. } else if(zones[0] && zones[1]) {
  160. return "Zones 1-2";
  161. } else if(zones[0]) {
  162. return "Zone 1";
  163. } else if(zones[1] && zones[4]) {
  164. return "Zones 2-5";
  165. } else if(zones[1] && zones[3]) {
  166. return "Zones 2-4";
  167. } else if(zones[1] && zones[2]) {
  168. return "Zones 2-3";
  169. } else if(zones[1]) {
  170. return "Zone 2";
  171. } else if(zones[2] && zones[4]) {
  172. return "Zones 3-5";
  173. } else if(zones[2] && zones[3]) {
  174. return "Zones 3-4";
  175. } else if(zones[2]) {
  176. return "Zone 3";
  177. } else if(zones[3] && zones[4]) {
  178. return "Zones 4-5";
  179. } else if(zones[3]) {
  180. return "Zone 4";
  181. } else if(zones[4]) {
  182. return "Zone 5";
  183. } else {
  184. return "Unknown";
  185. }
  186. }
  187. const char* get_intercode_version(int version) {
  188. // version is a 6 bits int
  189. // if the first 3 bits are 000, it's a 1.x version
  190. // if the first 3 bits are 001, it's a 2.x version
  191. // else, it's unknown
  192. int major = (version >> 3) & 0x07;
  193. if(major == 0) {
  194. return "Intercode I";
  195. } else if(major == 1) {
  196. return "Intercode II";
  197. }
  198. return "Unknown";
  199. }
  200. int get_intercode_subversion(int version) {
  201. // subversion is a 3 bits int
  202. return version & 0x07;
  203. }
  204. char* get_token(char* psrc, const char* delimit, void* psave) {
  205. static char sret[512];
  206. register char* ptr = psave;
  207. memset(sret, 0, sizeof(sret));
  208. if(psrc != NULL) strcpy(ptr, psrc);
  209. if(ptr == NULL) return NULL;
  210. int i = 0, nlength = strlen(ptr);
  211. for(i = 0; i < nlength; i++) {
  212. if(ptr[i] == delimit[0]) break;
  213. if(ptr[i] == delimit[1]) {
  214. ptr = NULL;
  215. break;
  216. }
  217. sret[i] = ptr[i];
  218. }
  219. if(ptr != NULL) strcpy(ptr, &ptr[i + 1]);
  220. return sret;
  221. }
  222. char* get_navigo_station(
  223. int station_group_id,
  224. int station_id,
  225. int station_sub_id,
  226. int service_provider) {
  227. switch(service_provider) {
  228. case NAVIGO_PROVIDER_SNCF: {
  229. if(station_group_id < 77 && station_id < 19) {
  230. const char* station_group_name =
  231. NAVIGO_SNCF_LOCATION_LIST[station_group_id][station_id];
  232. if(station_group_name) {
  233. // split station_name by '|' and return the station_sub_id - 1 element
  234. char* station_name = strdup(station_group_name);
  235. if(!station_name) {
  236. return "Unknown";
  237. }
  238. char* token = get_token(station_name, "|", station_name);
  239. for(int i = 0; i < station_sub_id - 1; i++) {
  240. token = get_token(station_name, "|", station_name);
  241. if(!token) {
  242. break;
  243. }
  244. }
  245. free(station_name);
  246. if(token) {
  247. return token;
  248. }
  249. }
  250. }
  251. // cast station_group_id-station_id-station_sub_id to a string
  252. char* station = malloc(12 * sizeof(char));
  253. if(!station) {
  254. return "Unknown";
  255. }
  256. snprintf(station, 10, "%d-%d-%d", station_group_id, station_id, station_sub_id);
  257. return station;
  258. }
  259. case NAVIGO_PROVIDER_RATP:
  260. case NAVIGO_PROVIDER_ORA: {
  261. if(station_group_id < 32 && station_id < 16) {
  262. const char* station_name = NAVIGO_RATP_LOCATION_LIST[station_group_id][station_id];
  263. if(station_name) {
  264. char* station;
  265. if((station = strdup(station_name)) != NULL) {
  266. return station;
  267. }
  268. }
  269. }
  270. // cast station_group_id-station_id to a string
  271. char* station = malloc(12 * sizeof(char));
  272. if(!station) {
  273. return "Unknown";
  274. }
  275. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  276. return station;
  277. }
  278. default: {
  279. // cast station_group_id-station_id to a string
  280. char* station = malloc(12 * sizeof(char));
  281. if(!station) {
  282. return "Unknown";
  283. }
  284. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  285. return station;
  286. }
  287. }
  288. }
  289. const char* get_navigo_sncf_train_line(int station_group_id) {
  290. if(station_group_id < 77) {
  291. return NAVIGO_SNCF_TRAIN_LINES_LIST[station_group_id];
  292. }
  293. return "Unknown";
  294. }
  295. const char* get_navigo_tram_line(int route_number) {
  296. switch(route_number) {
  297. case 1:
  298. return "T3a";
  299. case 9:
  300. return "T9";
  301. case 16:
  302. return "T6";
  303. case 18:
  304. return "T8";
  305. default: {
  306. char* line = malloc(5 * sizeof(char));
  307. if(!line) {
  308. return "Unknown";
  309. }
  310. snprintf(line, 5, "?%d?", route_number);
  311. return line;
  312. }
  313. }
  314. }
  315. void show_navigo_event_info(
  316. NavigoCardEvent* event,
  317. NavigoCardContract* contracts,
  318. FuriString* parsed_data) {
  319. if(event->used_contract == 0) {
  320. furi_string_cat_printf(parsed_data, "No event data\n");
  321. return;
  322. }
  323. char* station = get_navigo_station(
  324. event->station_group_id, event->station_id, event->station_sub_id, event->service_provider);
  325. char* sector = get_navigo_station(event->station_group_id, 0, 0, event->service_provider);
  326. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  327. event->transport_type == METRO || event->transport_type == TRAM) {
  328. if(event->route_number_available) {
  329. if(event->transport_type == METRO && event->route_number == 103) {
  330. furi_string_cat_printf(
  331. parsed_data,
  332. "%s 3 bis\n%s\n",
  333. get_navigo_transport_type(event->transport_type),
  334. get_intercode_string_transition_type(event->transition));
  335. } else if(event->transport_type == TRAM) {
  336. furi_string_cat_printf(
  337. parsed_data,
  338. "%s %s\n%s\n",
  339. get_navigo_transport_type(event->transport_type),
  340. get_navigo_tram_line(event->route_number),
  341. get_intercode_string_transition_type(event->transition));
  342. } else {
  343. furi_string_cat_printf(
  344. parsed_data,
  345. "%s %d\n%s\n",
  346. get_navigo_transport_type(event->transport_type),
  347. event->route_number,
  348. get_intercode_string_transition_type(event->transition));
  349. }
  350. } else {
  351. furi_string_cat_printf(
  352. parsed_data,
  353. "%s\n%s\n",
  354. get_navigo_transport_type(event->transport_type),
  355. get_intercode_string_transition_type(event->transition));
  356. }
  357. furi_string_cat_printf(
  358. parsed_data,
  359. "Transporter: %s\n",
  360. get_navigo_service_provider(event->service_provider));
  361. furi_string_cat_printf(parsed_data, "Station: %s\nSector: %s\n", station, sector);
  362. if(event->location_gate_available) {
  363. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  364. }
  365. if(event->device_available) {
  366. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS) {
  367. const char* side = event->side == 0 ? "right" : "left";
  368. furi_string_cat_printf(parsed_data, "Door: %d\nSide: %s\n", event->door, side);
  369. } else {
  370. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  371. }
  372. }
  373. if(event->mission_available) {
  374. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  375. }
  376. if(event->vehicle_id_available) {
  377. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  378. }
  379. if(event->used_contract_available) {
  380. furi_string_cat_printf(
  381. parsed_data,
  382. "Contract: %d - %s\n",
  383. event->used_contract,
  384. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  385. }
  386. locale_format_datetime_cat(parsed_data, &event->date, true);
  387. furi_string_cat_printf(parsed_data, "\n");
  388. } else if(event->transport_type == COMMUTER_TRAIN) {
  389. if(event->route_number_available) {
  390. furi_string_cat_printf(
  391. parsed_data,
  392. "RER %c\n%s\n",
  393. (65 + event->route_number - 17),
  394. get_intercode_string_transition_type(event->transition));
  395. } else {
  396. furi_string_cat_printf(
  397. parsed_data,
  398. "%s %s\n%s\n",
  399. get_navigo_transport_type(event->transport_type),
  400. get_navigo_sncf_train_line(event->station_group_id),
  401. get_intercode_string_transition_type(event->transition));
  402. }
  403. furi_string_cat_printf(
  404. parsed_data,
  405. "Transporter: %s\n",
  406. get_navigo_service_provider(event->service_provider));
  407. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  408. if(event->location_gate_available) {
  409. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  410. }
  411. if(event->device_available) {
  412. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  413. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  414. } else {
  415. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  416. }
  417. }
  418. if(event->mission_available) {
  419. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  420. }
  421. if(event->vehicle_id_available) {
  422. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  423. }
  424. if(event->used_contract_available) {
  425. furi_string_cat_printf(
  426. parsed_data,
  427. "Contract: %d - %s\n",
  428. event->used_contract,
  429. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  430. }
  431. locale_format_datetime_cat(parsed_data, &event->date, true);
  432. furi_string_cat_printf(parsed_data, "\n");
  433. } else {
  434. furi_string_cat_printf(
  435. parsed_data,
  436. "%s - %s\n",
  437. get_navigo_transport_type(event->transport_type),
  438. get_intercode_string_transition_type(event->transition));
  439. furi_string_cat_printf(
  440. parsed_data,
  441. "Transporter: %s\n",
  442. get_navigo_service_provider(event->service_provider));
  443. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  444. if(event->location_gate_available) {
  445. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  446. }
  447. if(event->device_available) {
  448. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  449. }
  450. if(event->mission_available) {
  451. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  452. }
  453. if(event->vehicle_id_available) {
  454. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  455. }
  456. if(event->used_contract_available) {
  457. furi_string_cat_printf(
  458. parsed_data,
  459. "Contract: %d - %s\n",
  460. event->used_contract,
  461. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  462. }
  463. locale_format_datetime_cat(parsed_data, &event->date, true);
  464. furi_string_cat_printf(parsed_data, "\n");
  465. }
  466. free(station);
  467. free(sector);
  468. }
  469. void show_navigo_special_event_info(NavigoCardSpecialEvent* event, FuriString* parsed_data) {
  470. char* station = get_navigo_station(
  471. event->station_group_id, event->station_id, event->station_sub_id, event->service_provider);
  472. char* sector = get_navigo_station(event->station_group_id, 0, 0, event->service_provider);
  473. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  474. event->transport_type == METRO || event->transport_type == TRAM) {
  475. if(event->route_number_available) {
  476. if(event->transport_type == METRO && event->route_number == 103) {
  477. furi_string_cat_printf(
  478. parsed_data,
  479. "%s 3 bis\n%s\n",
  480. get_navigo_transport_type(event->transport_type),
  481. get_intercode_string_transition_type(event->transition));
  482. } else if(event->transport_type == TRAM) {
  483. furi_string_cat_printf(
  484. parsed_data,
  485. "%s %s\n%s\n",
  486. get_navigo_transport_type(event->transport_type),
  487. get_navigo_tram_line(event->route_number),
  488. get_intercode_string_transition_type(event->transition));
  489. } else {
  490. furi_string_cat_printf(
  491. parsed_data,
  492. "%s %d\n%s\n",
  493. get_navigo_transport_type(event->transport_type),
  494. event->route_number,
  495. get_intercode_string_transition_type(event->transition));
  496. }
  497. } else {
  498. furi_string_cat_printf(
  499. parsed_data,
  500. "%s\n%s\n",
  501. get_navigo_transport_type(event->transport_type),
  502. get_intercode_string_transition_type(event->transition));
  503. }
  504. furi_string_cat_printf(
  505. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  506. furi_string_cat_printf(
  507. parsed_data,
  508. "Transporter: %s\n",
  509. get_navigo_service_provider(event->service_provider));
  510. furi_string_cat_printf(parsed_data, "Station: %s\nSector: %s\n", station, sector);
  511. if(event->device_available) {
  512. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  513. }
  514. locale_format_datetime_cat(parsed_data, &event->date, true);
  515. furi_string_cat_printf(parsed_data, "\n");
  516. } else if(event->transport_type == COMMUTER_TRAIN) {
  517. if(event->route_number_available) {
  518. furi_string_cat_printf(
  519. parsed_data,
  520. "RER %c\n%s\n",
  521. (65 + event->route_number - 17),
  522. get_intercode_string_transition_type(event->transition));
  523. } else {
  524. furi_string_cat_printf(
  525. parsed_data,
  526. "%s %s\n%s\n",
  527. get_navigo_transport_type(event->transport_type),
  528. get_navigo_sncf_train_line(event->station_group_id),
  529. get_intercode_string_transition_type(event->transition));
  530. }
  531. furi_string_cat_printf(
  532. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  533. furi_string_cat_printf(
  534. parsed_data,
  535. "Transporter: %s\n",
  536. get_navigo_service_provider(event->service_provider));
  537. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  538. if(event->device_available) {
  539. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  540. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  541. } else {
  542. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  543. }
  544. }
  545. locale_format_datetime_cat(parsed_data, &event->date, true);
  546. furi_string_cat_printf(parsed_data, "\n");
  547. } else {
  548. furi_string_cat_printf(
  549. parsed_data,
  550. "%s - %s\n",
  551. get_navigo_transport_type(event->transport_type),
  552. get_intercode_string_transition_type(event->transition));
  553. furi_string_cat_printf(
  554. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  555. furi_string_cat_printf(
  556. parsed_data,
  557. "Transporter: %s\n",
  558. get_navigo_service_provider(event->service_provider));
  559. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  560. if(event->device_available) {
  561. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  562. }
  563. locale_format_datetime_cat(parsed_data, &event->date, true);
  564. furi_string_cat_printf(parsed_data, "\n");
  565. }
  566. free(station);
  567. free(sector);
  568. }
  569. void show_navigo_contract_info(NavigoCardContract* contract, FuriString* parsed_data) {
  570. furi_string_cat_printf(parsed_data, "Type: %s\n", get_navigo_tariff(contract->tariff));
  571. if(is_ticket_count_available(contract->tariff)) {
  572. furi_string_cat_printf(parsed_data, "Remaining Tickets: %d\n", contract->counter.count);
  573. }
  574. if(contract->serial_number_available) {
  575. furi_string_cat_printf(parsed_data, "TCN Number: %d\n", contract->serial_number);
  576. }
  577. if(contract->pay_method_available) {
  578. furi_string_cat_printf(
  579. parsed_data, "Payment Method: %s\n", get_pay_method(contract->pay_method));
  580. }
  581. if(contract->price_amount_available) {
  582. furi_string_cat_printf(parsed_data, "Amount: %.2f EUR\n", contract->price_amount);
  583. }
  584. if(contract->end_date_available) {
  585. furi_string_cat_printf(parsed_data, "Valid\nfrom: ");
  586. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  587. furi_string_cat_printf(parsed_data, "\nto: ");
  588. locale_format_datetime_cat(parsed_data, &contract->end_date, false);
  589. furi_string_cat_printf(parsed_data, "\n");
  590. } else {
  591. furi_string_cat_printf(parsed_data, "Valid from\n");
  592. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  593. furi_string_cat_printf(parsed_data, "\n");
  594. }
  595. if(contract->zones_available) {
  596. furi_string_cat_printf(parsed_data, "%s\n", get_zones(contract->zones));
  597. }
  598. furi_string_cat_printf(parsed_data, "Sold on: ");
  599. locale_format_datetime_cat(parsed_data, &contract->sale_date, false);
  600. furi_string_cat_printf(parsed_data, "\n");
  601. furi_string_cat_printf(
  602. parsed_data, "Sales Agent: %s\n", get_navigo_service_provider(contract->sale_agent));
  603. furi_string_cat_printf(parsed_data, "Sales Terminal: %d\n", contract->sale_device);
  604. if(contract->status == 1) {
  605. furi_string_cat_printf(parsed_data, "Status: OK\n");
  606. } else {
  607. furi_string_cat_printf(parsed_data, "Status: Unknown (%d)\n", contract->status);
  608. }
  609. furi_string_cat_printf(parsed_data, "Authenticity Code: %d\n", contract->authenticator);
  610. }
  611. void show_navigo_environment_info(NavigoCardEnv* environment, FuriString* parsed_data) {
  612. furi_string_cat_printf(
  613. parsed_data,
  614. "App Version: %s - v%d\n",
  615. get_intercode_version(environment->app_version),
  616. get_intercode_subversion(environment->app_version));
  617. furi_string_cat_printf(
  618. parsed_data, "Country: %s\n", get_country_string(environment->country_num));
  619. furi_string_cat_printf(
  620. parsed_data,
  621. "Network: %s\n",
  622. get_network_string(guess_card_type(environment->country_num, environment->network_num)));
  623. furi_string_cat_printf(parsed_data, "End of validity:\n");
  624. locale_format_datetime_cat(parsed_data, &environment->end_dt, false);
  625. furi_string_cat_printf(parsed_data, "\n");
  626. }