navigo.c 26 KB

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