navigo.c 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696
  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. char* get_token(char* psrc, const char* delimit, void* psave) {
  188. static char sret[512];
  189. register char* ptr = psave;
  190. memset(sret, 0, sizeof(sret));
  191. if(psrc != NULL) strcpy(ptr, psrc);
  192. if(ptr == NULL) return NULL;
  193. int i = 0, nlength = strlen(ptr);
  194. for(i = 0; i < nlength; i++) {
  195. if(ptr[i] == delimit[0]) break;
  196. if(ptr[i] == delimit[1]) {
  197. ptr = NULL;
  198. break;
  199. }
  200. sret[i] = ptr[i];
  201. }
  202. if(ptr != NULL) strcpy(ptr, &ptr[i + 1]);
  203. return sret;
  204. }
  205. char* get_navigo_station(
  206. int station_group_id,
  207. int station_id,
  208. int station_sub_id,
  209. int service_provider) {
  210. switch(service_provider) {
  211. case NAVIGO_PROVIDER_SNCF: {
  212. if(station_group_id < 77 && station_id < 19) {
  213. const char* sncf_stations_path = APP_ASSETS_PATH("navigo/sncf_stations.txt");
  214. Storage* storage = furi_record_open(RECORD_STORAGE);
  215. Stream* stream = file_stream_alloc(storage);
  216. FuriString* line = furi_string_alloc();
  217. char* found_station_name = NULL;
  218. if(file_stream_open(stream, sncf_stations_path, FSAM_READ, FSOM_OPEN_EXISTING)) {
  219. while(stream_read_line(stream, line)) {
  220. // file is in csv format: station_group_id,station_id,station_sub_id,station_name
  221. // search for the station
  222. furi_string_replace_all(line, "\r", "");
  223. furi_string_replace_all(line, "\n", "");
  224. const char* string_line = furi_string_get_cstr(line);
  225. char* string_line_copy = strdup(string_line);
  226. if(!string_line_copy) {
  227. return "Unknown";
  228. }
  229. int line_station_group_id =
  230. atoi(get_token(string_line_copy, ",", string_line_copy));
  231. int line_station_id = atoi(get_token(string_line_copy, ",", string_line_copy));
  232. int line_station_sub_id =
  233. atoi(get_token(string_line_copy, ",", string_line_copy));
  234. if(line_station_group_id == station_group_id &&
  235. line_station_id == station_id && line_station_sub_id == station_sub_id) {
  236. found_station_name =
  237. strdup(get_token(string_line_copy, ",", string_line_copy));
  238. break;
  239. }
  240. free(string_line_copy);
  241. }
  242. } else {
  243. FURI_LOG_E("Metroflip:Scene:Calypso", "Failed to open sncf_stations.txt");
  244. }
  245. furi_string_free(line);
  246. file_stream_close(stream);
  247. stream_free(stream);
  248. if(found_station_name) {
  249. return found_station_name;
  250. }
  251. }
  252. // cast station_group_id-station_id-station_sub_id to a string
  253. char* station = malloc(12 * sizeof(char));
  254. if(!station) {
  255. return "Unknown";
  256. }
  257. snprintf(station, 10, "%d-%d-%d", station_group_id, station_id, station_sub_id);
  258. return station;
  259. }
  260. case NAVIGO_PROVIDER_RATP:
  261. case NAVIGO_PROVIDER_ORA: {
  262. if(station_group_id < 32 && station_id < 16) {
  263. const char* ratp_stations_path = APP_ASSETS_PATH("navigo/ratp_stations.txt");
  264. Storage* storage = furi_record_open(RECORD_STORAGE);
  265. Stream* stream = file_stream_alloc(storage);
  266. FuriString* line = furi_string_alloc();
  267. char* found_station_name = NULL;
  268. if(file_stream_open(stream, ratp_stations_path, FSAM_READ, FSOM_OPEN_EXISTING)) {
  269. while(stream_read_line(stream, line)) {
  270. // file is in csv format: station_group_id,station_id,station_name
  271. // search for the station
  272. furi_string_replace_all(line, "\r", "");
  273. furi_string_replace_all(line, "\n", "");
  274. const char* string_line = furi_string_get_cstr(line);
  275. char* string_line_copy = strdup(string_line);
  276. if(!string_line_copy) {
  277. return "Unknown";
  278. }
  279. int line_station_group_id =
  280. atoi(get_token(string_line_copy, ",", string_line_copy));
  281. int line_station_id = atoi(get_token(string_line_copy, ",", string_line_copy));
  282. if(line_station_group_id == station_group_id &&
  283. line_station_id == station_id) {
  284. found_station_name =
  285. strdup(get_token(string_line_copy, ",", string_line_copy));
  286. break;
  287. }
  288. free(string_line_copy);
  289. }
  290. } else {
  291. FURI_LOG_E("Metroflip:Scene:Calypso", "Failed to open ratp_stations.txt");
  292. }
  293. furi_string_free(line);
  294. file_stream_close(stream);
  295. stream_free(stream);
  296. if(found_station_name) {
  297. return found_station_name;
  298. }
  299. }
  300. // cast station_group_id-station_id to a string
  301. char* station = malloc(12 * sizeof(char));
  302. if(!station) {
  303. return "Unknown";
  304. }
  305. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  306. return station;
  307. }
  308. default: {
  309. // cast station_group_id-station_id to a string
  310. char* station = malloc(12 * sizeof(char));
  311. if(!station) {
  312. return "Unknown";
  313. }
  314. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  315. return station;
  316. }
  317. }
  318. }
  319. const char* get_navigo_sncf_train_line(int station_group_id) {
  320. if(station_group_id < 77) {
  321. return NAVIGO_SNCF_TRAIN_LINES_LIST[station_group_id];
  322. }
  323. return "Unknown";
  324. }
  325. const char* get_navigo_tram_line(int route_number) {
  326. switch(route_number) {
  327. case 1:
  328. return "T3a";
  329. case 9:
  330. return "T9";
  331. case 16:
  332. return "T6";
  333. case 18:
  334. return "T8";
  335. default: {
  336. char* line = malloc(5 * sizeof(char));
  337. if(!line) {
  338. return "Unknown";
  339. }
  340. snprintf(line, 5, "?%d?", route_number);
  341. return line;
  342. }
  343. }
  344. }
  345. void show_navigo_event_info(
  346. NavigoCardEvent* event,
  347. NavigoCardContract* contracts,
  348. FuriString* parsed_data) {
  349. if(event->used_contract == 0) {
  350. furi_string_cat_printf(parsed_data, "No event data\n");
  351. return;
  352. }
  353. char* station = get_navigo_station(
  354. event->station_group_id, event->station_id, event->station_sub_id, event->service_provider);
  355. char* sector = get_navigo_station(event->station_group_id, 0, 0, event->service_provider);
  356. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  357. event->transport_type == METRO || event->transport_type == TRAM) {
  358. if(event->route_number_available) {
  359. if(event->transport_type == METRO && event->route_number == 103) {
  360. furi_string_cat_printf(
  361. parsed_data,
  362. "%s 3 bis\n%s\n",
  363. get_navigo_transport_type(event->transport_type),
  364. get_intercode_string_transition_type(event->transition));
  365. } else if(event->transport_type == TRAM) {
  366. furi_string_cat_printf(
  367. parsed_data,
  368. "%s %s\n%s\n",
  369. get_navigo_transport_type(event->transport_type),
  370. get_navigo_tram_line(event->route_number),
  371. get_intercode_string_transition_type(event->transition));
  372. } else {
  373. furi_string_cat_printf(
  374. parsed_data,
  375. "%s %d\n%s\n",
  376. get_navigo_transport_type(event->transport_type),
  377. event->route_number,
  378. get_intercode_string_transition_type(event->transition));
  379. }
  380. } else {
  381. furi_string_cat_printf(
  382. parsed_data,
  383. "%s\n%s\n",
  384. get_navigo_transport_type(event->transport_type),
  385. get_intercode_string_transition_type(event->transition));
  386. }
  387. furi_string_cat_printf(
  388. parsed_data,
  389. "Transporter: %s\n",
  390. get_navigo_service_provider(event->service_provider));
  391. furi_string_cat_printf(parsed_data, "Station: %s\nSector: %s\n", station, sector);
  392. if(event->location_gate_available) {
  393. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  394. }
  395. if(event->device_available) {
  396. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS) {
  397. const char* side = event->side == 0 ? "right" : "left";
  398. furi_string_cat_printf(parsed_data, "Door: %d\nSide: %s\n", event->door, side);
  399. } else {
  400. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  401. }
  402. }
  403. if(event->mission_available) {
  404. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  405. }
  406. if(event->vehicle_id_available) {
  407. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  408. }
  409. if(event->used_contract_available) {
  410. furi_string_cat_printf(
  411. parsed_data,
  412. "Contract: %d - %s\n",
  413. event->used_contract,
  414. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  415. }
  416. locale_format_datetime_cat(parsed_data, &event->date, true);
  417. furi_string_cat_printf(parsed_data, "\n");
  418. } else if(event->transport_type == COMMUTER_TRAIN) {
  419. if(event->route_number_available) {
  420. furi_string_cat_printf(
  421. parsed_data,
  422. "RER %c\n%s\n",
  423. (65 + event->route_number - 17),
  424. get_intercode_string_transition_type(event->transition));
  425. } else {
  426. furi_string_cat_printf(
  427. parsed_data,
  428. "%s %s\n%s\n",
  429. get_navigo_transport_type(event->transport_type),
  430. get_navigo_sncf_train_line(event->station_group_id),
  431. get_intercode_string_transition_type(event->transition));
  432. }
  433. furi_string_cat_printf(
  434. parsed_data,
  435. "Transporter: %s\n",
  436. get_navigo_service_provider(event->service_provider));
  437. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  438. if(event->location_gate_available) {
  439. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  440. }
  441. if(event->device_available) {
  442. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  443. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  444. } else {
  445. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  446. }
  447. }
  448. if(event->mission_available) {
  449. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  450. }
  451. if(event->vehicle_id_available) {
  452. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  453. }
  454. if(event->used_contract_available) {
  455. furi_string_cat_printf(
  456. parsed_data,
  457. "Contract: %d - %s\n",
  458. event->used_contract,
  459. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  460. }
  461. locale_format_datetime_cat(parsed_data, &event->date, true);
  462. furi_string_cat_printf(parsed_data, "\n");
  463. } else {
  464. furi_string_cat_printf(
  465. parsed_data,
  466. "%s - %s\n",
  467. get_navigo_transport_type(event->transport_type),
  468. get_intercode_string_transition_type(event->transition));
  469. furi_string_cat_printf(
  470. parsed_data,
  471. "Transporter: %s\n",
  472. get_navigo_service_provider(event->service_provider));
  473. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  474. if(event->location_gate_available) {
  475. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  476. }
  477. if(event->device_available) {
  478. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  479. }
  480. if(event->mission_available) {
  481. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  482. }
  483. if(event->vehicle_id_available) {
  484. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  485. }
  486. if(event->used_contract_available) {
  487. furi_string_cat_printf(
  488. parsed_data,
  489. "Contract: %d - %s\n",
  490. event->used_contract,
  491. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  492. }
  493. locale_format_datetime_cat(parsed_data, &event->date, true);
  494. furi_string_cat_printf(parsed_data, "\n");
  495. }
  496. free(station);
  497. free(sector);
  498. }
  499. void show_navigo_special_event_info(NavigoCardSpecialEvent* event, FuriString* parsed_data) {
  500. char* station = get_navigo_station(
  501. event->station_group_id, event->station_id, event->station_sub_id, event->service_provider);
  502. char* sector = get_navigo_station(event->station_group_id, 0, 0, event->service_provider);
  503. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  504. event->transport_type == METRO || event->transport_type == TRAM) {
  505. if(event->route_number_available) {
  506. if(event->transport_type == METRO && event->route_number == 103) {
  507. furi_string_cat_printf(
  508. parsed_data,
  509. "%s 3 bis\n%s\n",
  510. get_navigo_transport_type(event->transport_type),
  511. get_intercode_string_transition_type(event->transition));
  512. } else if(event->transport_type == TRAM) {
  513. furi_string_cat_printf(
  514. parsed_data,
  515. "%s %s\n%s\n",
  516. get_navigo_transport_type(event->transport_type),
  517. get_navigo_tram_line(event->route_number),
  518. get_intercode_string_transition_type(event->transition));
  519. } else {
  520. furi_string_cat_printf(
  521. parsed_data,
  522. "%s %d\n%s\n",
  523. get_navigo_transport_type(event->transport_type),
  524. event->route_number,
  525. get_intercode_string_transition_type(event->transition));
  526. }
  527. } else {
  528. furi_string_cat_printf(
  529. parsed_data,
  530. "%s\n%s\n",
  531. get_navigo_transport_type(event->transport_type),
  532. get_intercode_string_transition_type(event->transition));
  533. }
  534. furi_string_cat_printf(
  535. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  536. furi_string_cat_printf(
  537. parsed_data,
  538. "Transporter: %s\n",
  539. get_navigo_service_provider(event->service_provider));
  540. furi_string_cat_printf(parsed_data, "Station: %s\nSector: %s\n", station, sector);
  541. if(event->device_available) {
  542. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  543. }
  544. locale_format_datetime_cat(parsed_data, &event->date, true);
  545. furi_string_cat_printf(parsed_data, "\n");
  546. } else if(event->transport_type == COMMUTER_TRAIN) {
  547. if(event->route_number_available) {
  548. furi_string_cat_printf(
  549. parsed_data,
  550. "RER %c\n%s\n",
  551. (65 + event->route_number - 17),
  552. get_intercode_string_transition_type(event->transition));
  553. } else {
  554. furi_string_cat_printf(
  555. parsed_data,
  556. "%s %s\n%s\n",
  557. get_navigo_transport_type(event->transport_type),
  558. get_navigo_sncf_train_line(event->station_group_id),
  559. get_intercode_string_transition_type(event->transition));
  560. }
  561. furi_string_cat_printf(
  562. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  563. furi_string_cat_printf(
  564. parsed_data,
  565. "Transporter: %s\n",
  566. get_navigo_service_provider(event->service_provider));
  567. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  568. if(event->device_available) {
  569. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  570. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  571. } else {
  572. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  573. }
  574. }
  575. locale_format_datetime_cat(parsed_data, &event->date, true);
  576. furi_string_cat_printf(parsed_data, "\n");
  577. } else {
  578. furi_string_cat_printf(
  579. parsed_data,
  580. "%s - %s\n",
  581. get_navigo_transport_type(event->transport_type),
  582. get_intercode_string_transition_type(event->transition));
  583. furi_string_cat_printf(
  584. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  585. furi_string_cat_printf(
  586. parsed_data,
  587. "Transporter: %s\n",
  588. get_navigo_service_provider(event->service_provider));
  589. furi_string_cat_printf(parsed_data, "Station: %s\n", station);
  590. if(event->device_available) {
  591. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  592. }
  593. locale_format_datetime_cat(parsed_data, &event->date, true);
  594. furi_string_cat_printf(parsed_data, "\n");
  595. }
  596. free(station);
  597. free(sector);
  598. }
  599. void show_navigo_contract_info(NavigoCardContract* contract, FuriString* parsed_data) {
  600. // Core type and ticket info
  601. furi_string_cat_printf(parsed_data, "Type: %s\n", get_navigo_tariff(contract->tariff));
  602. if(contract->counter_present) {
  603. furi_string_cat_printf(parsed_data, "Remaining Tickets: %d\n", contract->counter.count);
  604. furi_string_cat_printf(parsed_data, "Last load: %d\n", contract->counter.last_load);
  605. }
  606. // Validity period
  607. furi_string_cat_printf(parsed_data, "Valid from: ");
  608. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  609. furi_string_cat_printf(parsed_data, "\n");
  610. if(contract->end_date_available) {
  611. furi_string_cat_printf(parsed_data, "to: ");
  612. locale_format_datetime_cat(parsed_data, &contract->end_date, false);
  613. furi_string_cat_printf(parsed_data, "\n");
  614. }
  615. // Serial number (if available)
  616. if(contract->serial_number_available) {
  617. furi_string_cat_printf(parsed_data, "TCN Number: %d\n", contract->serial_number);
  618. }
  619. if(contract->price_amount_available) {
  620. furi_string_cat_printf(parsed_data, "Amount: %.2f EUR\n", contract->price_amount);
  621. }
  622. if(contract->pay_method_available) {
  623. furi_string_cat_printf(
  624. parsed_data, "Payment Method: %s\n", get_pay_method(contract->pay_method));
  625. }
  626. if(contract->zones_available) {
  627. furi_string_cat_printf(parsed_data, "%s\n", get_zones(contract->zones));
  628. }
  629. furi_string_cat_printf(parsed_data, "Sold on: ");
  630. locale_format_datetime_cat(parsed_data, &contract->sale_date, false);
  631. furi_string_cat_printf(parsed_data, "\n");
  632. furi_string_cat_printf(
  633. parsed_data, "Sales Agent: %s\n", get_navigo_service_provider(contract->sale_agent));
  634. furi_string_cat_printf(parsed_data, "Sales Terminal: %d\n", contract->sale_device);
  635. furi_string_cat_printf(
  636. parsed_data, "Status: %s\n", get_intercode_string_contract_status(contract->status));
  637. furi_string_cat_printf(parsed_data, "Authenticity Code: %d\n", contract->authenticator);
  638. }
  639. void show_navigo_environment_info(
  640. NavigoCardEnv* environment,
  641. NavigoCardHolder* holder,
  642. FuriString* parsed_data) {
  643. furi_string_cat_printf(
  644. parsed_data, "Card status: %s\n", get_intercode_string_holder_type(holder->card_status));
  645. if(is_intercode_string_holder_linked(holder->card_status)) {
  646. furi_string_cat_printf(parsed_data, "Linked to an organization\n");
  647. }
  648. furi_string_cat_printf(
  649. parsed_data,
  650. "App Version: %s - v%d\n",
  651. get_intercode_string_version(environment->app_version),
  652. get_intercode_string_subversion(environment->app_version));
  653. furi_string_cat_printf(
  654. parsed_data, "Country: %s\n", get_country_string(environment->country_num));
  655. furi_string_cat_printf(
  656. parsed_data,
  657. "Network: %s\n",
  658. get_network_string(guess_card_type(environment->country_num, environment->network_num)));
  659. furi_string_cat_printf(parsed_data, "End of validity:\n");
  660. locale_format_datetime_cat(parsed_data, &environment->end_dt, false);
  661. furi_string_cat_printf(parsed_data, "\n");
  662. }