navigo.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530
  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 BUS_URBAIN:
  7. return "Bus Urbain";
  8. case BUS_INTERURBAIN:
  9. return "Bus Interurbain";
  10. case METRO:
  11. return "Metro";
  12. case TRAM:
  13. return "Tram";
  14. case TRAIN:
  15. return "Train";
  16. case PARKING:
  17. return "Parking";
  18. default:
  19. return "Unknown";
  20. }
  21. }
  22. const char* get_navigo_service_provider(int provider) {
  23. switch(provider) {
  24. case 2:
  25. return "SNCF";
  26. case 3:
  27. return "RATP";
  28. case 4:
  29. return "IDF Mobilites";
  30. case 10:
  31. return "IDF Mobilites";
  32. case 115:
  33. return "CSO (VEOLIA)";
  34. case 116:
  35. return "R'Bus (VEOLIA)";
  36. case 156:
  37. return "Phebus";
  38. case 175:
  39. return "RATP (Veolia Transport Nanterre)";
  40. default:
  41. return "Unknown";
  42. }
  43. }
  44. const char* get_transition_type(int transition) {
  45. switch(transition) {
  46. case 1:
  47. return "Entry";
  48. case 2:
  49. return "Exit";
  50. case 4:
  51. return "Controle volant (a bord)";
  52. case 5:
  53. return "Test validation";
  54. case 6:
  55. return "Interchange - Entry";
  56. case 7:
  57. return "Interchange - Exit";
  58. case 9:
  59. return "Validation cancelled";
  60. case 10:
  61. return "Entry";
  62. case 11:
  63. return "Exit";
  64. case 13:
  65. return "Distribution";
  66. case 15:
  67. return "Invalidation";
  68. default: {
  69. char* transition_str = malloc(6 * sizeof(char));
  70. snprintf(transition_str, 6, "%d", transition);
  71. return transition_str;
  72. }
  73. }
  74. }
  75. const char* get_navigo_type(int type) {
  76. switch(type) {
  77. case NAVIGO_EASY:
  78. return "Navigo Easy";
  79. case NAVIGO_DECOUVERTE:
  80. return "Navigo Decouverte";
  81. case NAVIGO_STANDARD:
  82. return "Navigo Standard";
  83. case NAVIGO_INTEGRAL:
  84. return "Navigo Integral";
  85. case IMAGINE_R:
  86. return "Imagine R";
  87. default:
  88. return "Navigo";
  89. }
  90. }
  91. const char* get_navigo_tariff(int tariff) {
  92. switch(tariff) {
  93. case 0x0000:
  94. return "Navigo Mois";
  95. case 0x0001:
  96. return "Navigo Semaine";
  97. case 0x0002:
  98. return "Navigo Annuel";
  99. case 0x0003:
  100. return "Navigo Jour";
  101. case 0x0004:
  102. return "Imagine R Junior";
  103. case 0x0005:
  104. return "Imagine R Etudiant";
  105. case 0x000D:
  106. return "Navigo Jeunes Week-end";
  107. case 0x0015:
  108. return "Paris-Visite"; // Theoric
  109. case 0x1000:
  110. return "Navigo Liberte+";
  111. case 0x4000:
  112. return "Navigo Mois 75%%";
  113. case 0x4001:
  114. return "Navigo Semaine 75%%";
  115. case 0x4015:
  116. return "Paris-Visite (Enfant)"; // Theoric
  117. case 0x5000:
  118. return "Tickets T+";
  119. case 0x5004:
  120. return "Tickets OrlyBus"; // Theoric
  121. case 0x5005:
  122. return "Tickets RoissyBus"; // Theoric
  123. case 0x5006:
  124. return "Bus-Tram"; // Theoric
  125. case 0x5008:
  126. return "Metro-Train-RER"; // Theoric
  127. case 0x500b:
  128. return "Paris <> Aeroports"; // Theoric
  129. case 0x5010:
  130. return "Tickets T+ (Reduit)"; // Theoric
  131. case 0x5016:
  132. return "Bus-Tram (Reduit)"; // Theoric
  133. case 0x5018:
  134. return "Metro-Train-RER (Reduit)"; // Theoric
  135. case 0x501b:
  136. return "Paris <> Aeroports (Reduit)"; // Theoric
  137. case 0x8003:
  138. return "Navigo Solidarite Gratuit";
  139. default: {
  140. char* tariff_str = malloc(6 * sizeof(char));
  141. snprintf(tariff_str, 6, "%d", tariff);
  142. return tariff_str;
  143. }
  144. }
  145. }
  146. bool is_ticket_count_available(int tariff) {
  147. return tariff >= 0x5000 && tariff <= 0x501b;
  148. }
  149. const char* get_pay_method(int pay_method) {
  150. switch(pay_method) {
  151. case 0x30:
  152. return "Apple Pay";
  153. case 0x80:
  154. return "Debit PME";
  155. case 0x90:
  156. return "Cash";
  157. case 0xA0:
  158. return "Mobility Check";
  159. case 0xB3:
  160. return "Payment Card";
  161. case 0xA4:
  162. return "Check";
  163. case 0xA5:
  164. return "Vacation Check";
  165. case 0xB7:
  166. return "Telepayment";
  167. case 0xD0:
  168. return "Remote Payment";
  169. case 0xD7:
  170. return "Voucher, Prepayment, Exchange Voucher, Travel Voucher";
  171. case 0xD9:
  172. return "Discount Voucher";
  173. default:
  174. return "Unknown";
  175. }
  176. }
  177. const char* get_zones(int* zones) {
  178. if(zones[0] && zones[4]) {
  179. return "All Zones (1-5)";
  180. } else if(zones[0] && zones[3]) {
  181. return "Zones 1-4";
  182. } else if(zones[0] && zones[2]) {
  183. return "Zones 1-3";
  184. } else if(zones[0] && zones[1]) {
  185. return "Zones 1-2";
  186. } else if(zones[0]) {
  187. return "Zone 1";
  188. } else if(zones[1] && zones[4]) {
  189. return "Zones 2-5";
  190. } else if(zones[1] && zones[3]) {
  191. return "Zones 2-4";
  192. } else if(zones[1] && zones[2]) {
  193. return "Zones 2-3";
  194. } else if(zones[1]) {
  195. return "Zone 2";
  196. } else if(zones[2] && zones[4]) {
  197. return "Zones 3-5";
  198. } else if(zones[2] && zones[3]) {
  199. return "Zones 3-4";
  200. } else if(zones[2]) {
  201. return "Zone 3";
  202. } else if(zones[3] && zones[4]) {
  203. return "Zones 4-5";
  204. } else if(zones[3]) {
  205. return "Zone 4";
  206. } else if(zones[4]) {
  207. return "Zone 5";
  208. } else {
  209. return "Unknown";
  210. }
  211. }
  212. const char* get_intercode_version(int version) {
  213. // version is a 6 bits int
  214. // if the first 3 bits are 000, it's a 1.x version
  215. // if the first 3 bits are 001, it's a 2.x version
  216. // else, it's unknown
  217. int major = (version >> 3) & 0x07;
  218. if(major == 0) {
  219. return "Intercode I";
  220. } else if(major == 1) {
  221. return "Intercode II";
  222. }
  223. return "Unknown";
  224. }
  225. int get_intercode_subversion(int version) {
  226. // subversion is a 3 bits int
  227. return version & 0x07;
  228. }
  229. const char* get_navigo_metro_station(int station_group_id, int station_id) {
  230. // Use NAVIGO_H constants
  231. if(station_group_id < 32 && station_id < 16) {
  232. return NAVIGO_METRO_STATION_LIST[station_group_id][station_id];
  233. }
  234. // cast station_group_id-station_id to a string
  235. char* station = malloc(12 * sizeof(char));
  236. if(!station) {
  237. return "Unknown";
  238. }
  239. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  240. return station;
  241. }
  242. const char* get_navigo_train_line(int station_group_id) {
  243. if(station_group_id < 77) {
  244. return NAVIGO_TRAIN_LINES_LIST[station_group_id];
  245. }
  246. return "Unknown";
  247. }
  248. const char* get_navigo_train_station(int station_group_id, int station_id) {
  249. if(station_group_id < 77 && station_id < 19) {
  250. return NAVIGO_TRAIN_STATION_LIST[station_group_id][station_id];
  251. }
  252. // cast station_group_id-station_id to a string
  253. char* station = malloc(12 * sizeof(char));
  254. if(!station) {
  255. return "Unknown";
  256. }
  257. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  258. return station;
  259. }
  260. const char* get_navigo_tram_line(int route_number) {
  261. switch(route_number) {
  262. case 1:
  263. return "T3a";
  264. case 16:
  265. return "T6";
  266. default: {
  267. char* line = malloc(5 * sizeof(char));
  268. if(!line) {
  269. return "Unknown";
  270. }
  271. snprintf(line, 5, "?%d?", route_number);
  272. return line;
  273. }
  274. }
  275. }
  276. void show_navigo_event_info(
  277. NavigoCardEvent* event,
  278. NavigoCardContract* contracts,
  279. FuriString* parsed_data) {
  280. if(event->used_contract == 0) {
  281. furi_string_cat_printf(parsed_data, "No event data\n");
  282. return;
  283. }
  284. if(event->transport_type == BUS_URBAIN || event->transport_type == BUS_INTERURBAIN ||
  285. event->transport_type == METRO || event->transport_type == TRAM) {
  286. if(event->route_number_available) {
  287. if(event->transport_type == METRO && event->route_number == 103) {
  288. furi_string_cat_printf(
  289. parsed_data,
  290. "%s 3 bis\n%s\n",
  291. get_navigo_transport_type(event->transport_type),
  292. get_transition_type(event->transition));
  293. } else if(event->transport_type == TRAM) {
  294. furi_string_cat_printf(
  295. parsed_data,
  296. "%s %s\n%s\n",
  297. get_navigo_transport_type(event->transport_type),
  298. get_navigo_tram_line(event->route_number),
  299. get_transition_type(event->transition));
  300. } else {
  301. furi_string_cat_printf(
  302. parsed_data,
  303. "%s %d\n%s\n",
  304. get_navigo_transport_type(event->transport_type),
  305. event->route_number,
  306. get_transition_type(event->transition));
  307. }
  308. } else {
  309. furi_string_cat_printf(
  310. parsed_data,
  311. "%s\n%s\n",
  312. get_navigo_transport_type(event->transport_type),
  313. get_transition_type(event->transition));
  314. }
  315. furi_string_cat_printf(
  316. parsed_data,
  317. "Transporter: %s\n",
  318. get_navigo_service_provider(event->service_provider));
  319. if(event->transport_type == METRO) {
  320. furi_string_cat_printf(
  321. parsed_data,
  322. "Station: %s\nSector: %s\n",
  323. get_navigo_metro_station(event->station_group_id, event->station_id),
  324. get_navigo_metro_station(event->station_group_id, 0));
  325. } else {
  326. furi_string_cat_printf(
  327. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  328. }
  329. if(event->location_gate_available) {
  330. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  331. }
  332. if(event->device_available) {
  333. if(event->transport_type == BUS_URBAIN || event->transport_type == BUS_INTERURBAIN) {
  334. const char* side = event->side == 0 ? "right" : "left";
  335. furi_string_cat_printf(parsed_data, "Door: %d\nSide: %s\n", event->door, side);
  336. } else {
  337. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  338. }
  339. }
  340. if(event->mission_available) {
  341. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  342. }
  343. if(event->vehicle_id_available) {
  344. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  345. }
  346. if(event->used_contract_available) {
  347. furi_string_cat_printf(
  348. parsed_data,
  349. "Contract: %d - %s\n",
  350. event->used_contract,
  351. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  352. }
  353. locale_format_datetime_cat(parsed_data, &event->date, true);
  354. furi_string_cat_printf(parsed_data, "\n");
  355. } else if(event->transport_type == TRAIN) {
  356. if(event->route_number_available) {
  357. furi_string_cat_printf(
  358. parsed_data,
  359. "RER %c\n%s\n",
  360. (65 + event->route_number - 17),
  361. get_transition_type(event->transition));
  362. } else {
  363. furi_string_cat_printf(
  364. parsed_data,
  365. "%s %s\n%s\n",
  366. get_navigo_transport_type(event->transport_type),
  367. get_navigo_train_line(event->station_group_id),
  368. get_transition_type(event->transition));
  369. }
  370. furi_string_cat_printf(
  371. parsed_data,
  372. "Transporter: %s\n",
  373. get_navigo_service_provider(event->service_provider));
  374. if(event->service_provider == 2) {
  375. furi_string_cat_printf(
  376. parsed_data,
  377. "Station: %s\n",
  378. get_navigo_train_station(event->station_group_id, event->station_id));
  379. } else if(event->service_provider == 3) {
  380. furi_string_cat_printf(
  381. parsed_data,
  382. "Station: %s\n",
  383. get_navigo_metro_station(event->station_group_id, event->station_id));
  384. }
  385. /* if(event->route_number_available) {
  386. furi_string_cat_printf(parsed_data, "Route: %d\n", event->route_number);
  387. } */
  388. if(event->location_gate_available) {
  389. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  390. }
  391. if(event->device_available) {
  392. if(event->service_provider == 2) {
  393. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  394. } else {
  395. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  396. }
  397. }
  398. if(event->mission_available) {
  399. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  400. }
  401. if(event->vehicle_id_available) {
  402. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  403. }
  404. if(event->used_contract_available) {
  405. furi_string_cat_printf(
  406. parsed_data,
  407. "Contract: %d - %s\n",
  408. event->used_contract,
  409. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  410. }
  411. locale_format_datetime_cat(parsed_data, &event->date, true);
  412. furi_string_cat_printf(parsed_data, "\n");
  413. } else {
  414. furi_string_cat_printf(
  415. parsed_data,
  416. "%s - %s\n",
  417. get_navigo_transport_type(event->transport_type),
  418. get_transition_type(event->transition));
  419. furi_string_cat_printf(
  420. parsed_data,
  421. "Transporter: %s\n",
  422. get_navigo_service_provider(event->service_provider));
  423. furi_string_cat_printf(
  424. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  425. if(event->location_gate_available) {
  426. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  427. }
  428. if(event->device_available) {
  429. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  430. }
  431. if(event->mission_available) {
  432. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  433. }
  434. if(event->vehicle_id_available) {
  435. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  436. }
  437. if(event->used_contract_available) {
  438. furi_string_cat_printf(
  439. parsed_data,
  440. "Contract: %d - %s\n",
  441. event->used_contract,
  442. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  443. }
  444. locale_format_datetime_cat(parsed_data, &event->date, true);
  445. furi_string_cat_printf(parsed_data, "\n");
  446. }
  447. }
  448. void show_navigo_contract_info(NavigoCardContract* contract, FuriString* parsed_data) {
  449. // Core type and ticket info
  450. furi_string_cat_printf(parsed_data, "Type: %s\n", get_navigo_tariff(contract->tariff));
  451. if(is_ticket_count_available(contract->tariff)) {
  452. furi_string_cat_printf(parsed_data, "Remaining Tickets: %d\n", contract->counter.count);
  453. }
  454. // Validity period
  455. furi_string_cat_printf(parsed_data, "Valid from: ");
  456. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  457. furi_string_cat_printf(parsed_data, "\n");
  458. if(contract->end_date_available) {
  459. furi_string_cat_printf(parsed_data, "\nto: ");
  460. locale_format_datetime_cat(parsed_data, &contract->end_date, false);
  461. furi_string_cat_printf(parsed_data, "\n");
  462. }
  463. // Serial number (if available)
  464. if(contract->serial_number_available) {
  465. furi_string_cat_printf(parsed_data, "TCN Number: %d\n", contract->serial_number);
  466. }
  467. // Payment and pricing details
  468. if(contract->pay_method_available) {
  469. furi_string_cat_printf(
  470. parsed_data, "Payment Method: %s\n", get_pay_method(contract->pay_method));
  471. }
  472. if(contract->price_amount_available) {
  473. furi_string_cat_printf(parsed_data, "Amount: %.2f EUR\n", contract->price_amount);
  474. }
  475. // Zone and sales details
  476. if(contract->zones_available) {
  477. furi_string_cat_printf(parsed_data, "%s\n", get_zones(contract->zones));
  478. }
  479. furi_string_cat_printf(parsed_data, "Sold on: ");
  480. locale_format_datetime_cat(parsed_data, &contract->sale_date, false);
  481. furi_string_cat_printf(parsed_data, "\n");
  482. furi_string_cat_printf(
  483. parsed_data, "Sales Agent: %s\n", get_navigo_service_provider(contract->sale_agent));
  484. furi_string_cat_printf(parsed_data, "Sales Terminal: %d\n", contract->sale_device);
  485. // Status and authenticity
  486. if(contract->status == 1) {
  487. furi_string_cat_printf(parsed_data, "Status: OK\n");
  488. } else {
  489. furi_string_cat_printf(parsed_data, "Status: Unknown (%d)\n", contract->status);
  490. }
  491. furi_string_cat_printf(parsed_data, "Authenticity Code: %d\n", contract->authenticator);
  492. }
  493. void show_navigo_environment_info(NavigoCardEnv* environment, FuriString* parsed_data) {
  494. furi_string_cat_printf(
  495. parsed_data,
  496. "App Version: %s - v%d\n",
  497. get_intercode_version(environment->app_version),
  498. get_intercode_subversion(environment->app_version));
  499. furi_string_cat_printf(
  500. parsed_data, "Country: %s\n", get_country_string(environment->country_num));
  501. furi_string_cat_printf(
  502. parsed_data,
  503. "Network: %s\n",
  504. get_network_string(guess_card_type(environment->country_num, environment->network_num)));
  505. furi_string_cat_printf(parsed_data, "End of validity:\n");
  506. locale_format_datetime_cat(parsed_data, &environment->end_dt, false);
  507. furi_string_cat_printf(parsed_data, "\n");
  508. }