navigo.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608
  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. const char* get_navigo_station(int station_group_id, int station_id, int service_provider) {
  205. switch(service_provider) {
  206. case NAVIGO_PROVIDER_SNCF: {
  207. if(station_group_id < 77 && station_id < 19) {
  208. const char* station_name = NAVIGO_SNCF_LOCATION_LIST[station_group_id][station_id];
  209. if(station_name) {
  210. return station_name;
  211. }
  212. }
  213. // cast station_group_id-station_id to a string
  214. char* station = malloc(12 * sizeof(char));
  215. if(!station) {
  216. return "Unknown";
  217. }
  218. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  219. return station;
  220. }
  221. case NAVIGO_PROVIDER_RATP:
  222. case NAVIGO_PROVIDER_ORA: {
  223. if(station_group_id < 32 && station_id < 16) {
  224. const char* station_name = NAVIGO_RATP_LOCATION_LIST[station_group_id][station_id];
  225. if(station_name) {
  226. return station_name;
  227. }
  228. }
  229. // cast station_group_id-station_id to a string
  230. char* station = malloc(12 * sizeof(char));
  231. if(!station) {
  232. return "Unknown";
  233. }
  234. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  235. return station;
  236. }
  237. default: {
  238. // cast station_group_id-station_id to a string
  239. char* station = malloc(12 * sizeof(char));
  240. if(!station) {
  241. return "Unknown";
  242. }
  243. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  244. return station;
  245. }
  246. }
  247. }
  248. const char* get_navigo_sncf_train_line(int station_group_id) {
  249. if(station_group_id < 77) {
  250. return NAVIGO_SNCF_TRAIN_LINES_LIST[station_group_id];
  251. }
  252. return "Unknown";
  253. }
  254. const char* get_navigo_tram_line(int route_number) {
  255. switch(route_number) {
  256. case 1:
  257. return "T3a";
  258. case 16:
  259. return "T6";
  260. default: {
  261. char* line = malloc(5 * sizeof(char));
  262. if(!line) {
  263. return "Unknown";
  264. }
  265. snprintf(line, 5, "?%d?", route_number);
  266. return line;
  267. }
  268. }
  269. }
  270. void show_navigo_event_info(
  271. NavigoCardEvent* event,
  272. NavigoCardContract* contracts,
  273. FuriString* parsed_data) {
  274. if(event->used_contract == 0) {
  275. furi_string_cat_printf(parsed_data, "No event data\n");
  276. return;
  277. }
  278. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  279. event->transport_type == METRO || event->transport_type == TRAM) {
  280. if(event->route_number_available) {
  281. if(event->transport_type == METRO && event->route_number == 103) {
  282. furi_string_cat_printf(
  283. parsed_data,
  284. "%s 3 bis\n%s\n",
  285. get_navigo_transport_type(event->transport_type),
  286. get_intercode_string_transition_type(event->transition));
  287. } else if(event->transport_type == TRAM) {
  288. furi_string_cat_printf(
  289. parsed_data,
  290. "%s %s\n%s\n",
  291. get_navigo_transport_type(event->transport_type),
  292. get_navigo_tram_line(event->route_number),
  293. get_intercode_string_transition_type(event->transition));
  294. } else {
  295. furi_string_cat_printf(
  296. parsed_data,
  297. "%s %d\n%s\n",
  298. get_navigo_transport_type(event->transport_type),
  299. event->route_number,
  300. get_intercode_string_transition_type(event->transition));
  301. }
  302. } else {
  303. furi_string_cat_printf(
  304. parsed_data,
  305. "%s\n%s\n",
  306. get_navigo_transport_type(event->transport_type),
  307. get_intercode_string_transition_type(event->transition));
  308. }
  309. furi_string_cat_printf(
  310. parsed_data,
  311. "Transporter: %s\n",
  312. get_navigo_service_provider(event->service_provider));
  313. furi_string_cat_printf(
  314. parsed_data,
  315. "Station: %s\nSector: %s\n",
  316. get_navigo_station(event->station_group_id, event->station_id, event->service_provider),
  317. get_navigo_station(event->station_group_id, 0, event->service_provider));
  318. if(event->location_gate_available) {
  319. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  320. }
  321. if(event->device_available) {
  322. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS) {
  323. const char* side = event->side == 0 ? "right" : "left";
  324. furi_string_cat_printf(parsed_data, "Door: %d\nSide: %s\n", event->door, side);
  325. } else {
  326. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  327. }
  328. }
  329. if(event->mission_available) {
  330. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  331. }
  332. if(event->vehicle_id_available) {
  333. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  334. }
  335. if(event->used_contract_available) {
  336. furi_string_cat_printf(
  337. parsed_data,
  338. "Contract: %d - %s\n",
  339. event->used_contract,
  340. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  341. }
  342. locale_format_datetime_cat(parsed_data, &event->date, true);
  343. furi_string_cat_printf(parsed_data, "\n");
  344. } else if(event->transport_type == COMMUTER_TRAIN) {
  345. if(event->route_number_available) {
  346. furi_string_cat_printf(
  347. parsed_data,
  348. "RER %c\n%s\n",
  349. (65 + event->route_number - 17),
  350. get_intercode_string_transition_type(event->transition));
  351. } else {
  352. furi_string_cat_printf(
  353. parsed_data,
  354. "%s %s\n%s\n",
  355. get_navigo_transport_type(event->transport_type),
  356. get_navigo_sncf_train_line(event->station_group_id),
  357. get_intercode_string_transition_type(event->transition));
  358. }
  359. furi_string_cat_printf(
  360. parsed_data,
  361. "Transporter: %s\n",
  362. get_navigo_service_provider(event->service_provider));
  363. furi_string_cat_printf(
  364. parsed_data,
  365. "Station: %s\n",
  366. get_navigo_station(
  367. event->station_group_id, event->station_id, event->service_provider));
  368. /* if(event->route_number_available) {
  369. furi_string_cat_printf(parsed_data, "Route: %d\n", event->route_number);
  370. } */
  371. if(event->location_gate_available) {
  372. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  373. }
  374. if(event->device_available) {
  375. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  376. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  377. } else {
  378. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  379. }
  380. }
  381. if(event->mission_available) {
  382. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  383. }
  384. if(event->vehicle_id_available) {
  385. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  386. }
  387. if(event->used_contract_available) {
  388. furi_string_cat_printf(
  389. parsed_data,
  390. "Contract: %d - %s\n",
  391. event->used_contract,
  392. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  393. }
  394. locale_format_datetime_cat(parsed_data, &event->date, true);
  395. furi_string_cat_printf(parsed_data, "\n");
  396. } else {
  397. furi_string_cat_printf(
  398. parsed_data,
  399. "%s - %s\n",
  400. get_navigo_transport_type(event->transport_type),
  401. get_intercode_string_transition_type(event->transition));
  402. furi_string_cat_printf(
  403. parsed_data,
  404. "Transporter: %s\n",
  405. get_navigo_service_provider(event->service_provider));
  406. furi_string_cat_printf(
  407. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  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. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  413. }
  414. if(event->mission_available) {
  415. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  416. }
  417. if(event->vehicle_id_available) {
  418. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  419. }
  420. if(event->used_contract_available) {
  421. furi_string_cat_printf(
  422. parsed_data,
  423. "Contract: %d - %s\n",
  424. event->used_contract,
  425. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  426. }
  427. locale_format_datetime_cat(parsed_data, &event->date, true);
  428. furi_string_cat_printf(parsed_data, "\n");
  429. }
  430. }
  431. void show_navigo_special_event_info(NavigoCardSpecialEvent* event, FuriString* parsed_data) {
  432. if(event->transport_type == URBAN_BUS || event->transport_type == INTERURBAN_BUS ||
  433. event->transport_type == METRO || event->transport_type == TRAM) {
  434. if(event->route_number_available) {
  435. if(event->transport_type == METRO && event->route_number == 103) {
  436. furi_string_cat_printf(
  437. parsed_data,
  438. "%s 3 bis\n%s\n",
  439. get_navigo_transport_type(event->transport_type),
  440. get_intercode_string_transition_type(event->transition));
  441. } else if(event->transport_type == TRAM) {
  442. furi_string_cat_printf(
  443. parsed_data,
  444. "%s %s\n%s\n",
  445. get_navigo_transport_type(event->transport_type),
  446. get_navigo_tram_line(event->route_number),
  447. get_intercode_string_transition_type(event->transition));
  448. } else {
  449. furi_string_cat_printf(
  450. parsed_data,
  451. "%s %d\n%s\n",
  452. get_navigo_transport_type(event->transport_type),
  453. event->route_number,
  454. get_intercode_string_transition_type(event->transition));
  455. }
  456. } else {
  457. furi_string_cat_printf(
  458. parsed_data,
  459. "%s\n%s\n",
  460. get_navigo_transport_type(event->transport_type),
  461. get_intercode_string_transition_type(event->transition));
  462. }
  463. furi_string_cat_printf(
  464. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  465. furi_string_cat_printf(
  466. parsed_data,
  467. "Transporter: %s\n",
  468. get_navigo_service_provider(event->service_provider));
  469. furi_string_cat_printf(
  470. parsed_data,
  471. "Station: %s\nSector: %s\n",
  472. get_navigo_station(event->station_group_id, event->station_id, event->service_provider),
  473. get_navigo_station(event->station_group_id, 0, event->service_provider));
  474. if(event->device_available) {
  475. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  476. }
  477. locale_format_datetime_cat(parsed_data, &event->date, true);
  478. furi_string_cat_printf(parsed_data, "\n");
  479. } else if(event->transport_type == COMMUTER_TRAIN) {
  480. if(event->route_number_available) {
  481. furi_string_cat_printf(
  482. parsed_data,
  483. "RER %c\n%s\n",
  484. (65 + event->route_number - 17),
  485. get_intercode_string_transition_type(event->transition));
  486. } else {
  487. furi_string_cat_printf(
  488. parsed_data,
  489. "%s %s\n%s\n",
  490. get_navigo_transport_type(event->transport_type),
  491. get_navigo_sncf_train_line(event->station_group_id),
  492. get_intercode_string_transition_type(event->transition));
  493. }
  494. furi_string_cat_printf(
  495. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  496. furi_string_cat_printf(
  497. parsed_data,
  498. "Transporter: %s\n",
  499. get_navigo_service_provider(event->service_provider));
  500. furi_string_cat_printf(
  501. parsed_data,
  502. "Station: %s\n",
  503. get_navigo_station(
  504. event->station_group_id, event->station_id, event->service_provider));
  505. if(event->device_available) {
  506. if(event->service_provider == NAVIGO_PROVIDER_SNCF) {
  507. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  508. } else {
  509. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  510. }
  511. }
  512. locale_format_datetime_cat(parsed_data, &event->date, true);
  513. furi_string_cat_printf(parsed_data, "\n");
  514. } else {
  515. furi_string_cat_printf(
  516. parsed_data,
  517. "%s - %s\n",
  518. get_navigo_transport_type(event->transport_type),
  519. get_intercode_string_transition_type(event->transition));
  520. furi_string_cat_printf(
  521. parsed_data, "Result: %s\n", get_intercode_string_event_result(event->result));
  522. furi_string_cat_printf(
  523. parsed_data,
  524. "Transporter: %s\n",
  525. get_navigo_service_provider(event->service_provider));
  526. furi_string_cat_printf(
  527. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  528. if(event->device_available) {
  529. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  530. }
  531. locale_format_datetime_cat(parsed_data, &event->date, true);
  532. furi_string_cat_printf(parsed_data, "\n");
  533. }
  534. }
  535. void show_navigo_contract_info(NavigoCardContract* contract, FuriString* parsed_data) {
  536. furi_string_cat_printf(parsed_data, "Type: %s\n", get_navigo_tariff(contract->tariff));
  537. if(is_ticket_count_available(contract->tariff)) {
  538. furi_string_cat_printf(parsed_data, "Remaining Tickets: %d\n", contract->counter.count);
  539. }
  540. if(contract->serial_number_available) {
  541. furi_string_cat_printf(parsed_data, "TCN Number: %d\n", contract->serial_number);
  542. }
  543. if(contract->pay_method_available) {
  544. furi_string_cat_printf(
  545. parsed_data, "Payment Method: %s\n", get_pay_method(contract->pay_method));
  546. }
  547. if(contract->price_amount_available) {
  548. furi_string_cat_printf(parsed_data, "Amount: %.2f EUR\n", contract->price_amount);
  549. }
  550. if(contract->end_date_available) {
  551. furi_string_cat_printf(parsed_data, "Valid\nfrom: ");
  552. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  553. furi_string_cat_printf(parsed_data, "\nto: ");
  554. locale_format_datetime_cat(parsed_data, &contract->end_date, false);
  555. furi_string_cat_printf(parsed_data, "\n");
  556. } else {
  557. furi_string_cat_printf(parsed_data, "Valid from\n");
  558. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  559. furi_string_cat_printf(parsed_data, "\n");
  560. }
  561. if(contract->zones_available) {
  562. furi_string_cat_printf(parsed_data, "%s\n", get_zones(contract->zones));
  563. }
  564. furi_string_cat_printf(parsed_data, "Sold on: ");
  565. locale_format_datetime_cat(parsed_data, &contract->sale_date, false);
  566. furi_string_cat_printf(parsed_data, "\n");
  567. furi_string_cat_printf(
  568. parsed_data, "Sales Agent: %s\n", get_navigo_service_provider(contract->sale_agent));
  569. furi_string_cat_printf(parsed_data, "Sales Terminal: %d\n", contract->sale_device);
  570. if(contract->status == 1) {
  571. furi_string_cat_printf(parsed_data, "Status: OK\n");
  572. } else {
  573. furi_string_cat_printf(parsed_data, "Status: Unknown (%d)\n", contract->status);
  574. }
  575. furi_string_cat_printf(parsed_data, "Authenticity Code: %d\n", contract->authenticator);
  576. }
  577. void show_navigo_environment_info(NavigoCardEnv* environment, FuriString* parsed_data) {
  578. furi_string_cat_printf(
  579. parsed_data,
  580. "App Version: %s - v%d\n",
  581. get_intercode_version(environment->app_version),
  582. get_intercode_subversion(environment->app_version));
  583. furi_string_cat_printf(
  584. parsed_data, "Country: %s\n", get_country_string(environment->country_num));
  585. furi_string_cat_printf(
  586. parsed_data,
  587. "Network: %s\n",
  588. get_network_string(guess_card_type(environment->country_num, environment->network_num)));
  589. furi_string_cat_printf(parsed_data, "End of validity:\n");
  590. locale_format_datetime_cat(parsed_data, &environment->end_dt, false);
  591. furi_string_cat_printf(parsed_data, "\n");
  592. }