diff --git a/ui/package-lock.json b/ui/package-lock.json
index b76a8a5a0..36f907771 100644
--- a/ui/package-lock.json
+++ b/ui/package-lock.json
@@ -3727,9 +3727,9 @@
}
},
"node_modules/cross-spawn": {
- "version": "7.0.3",
- "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
- "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
+ "version": "7.0.6",
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
+ "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
"dev": true,
"license": "MIT",
"dependencies": {
@@ -7356,9 +7356,9 @@
"license": "MIT"
},
"node_modules/nanoid": {
- "version": "3.3.7",
- "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz",
- "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==",
+ "version": "3.3.8",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz",
+ "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==",
"dev": true,
"funding": [
{
diff --git a/ui/src/assets/iso31661Countries.ts b/ui/src/assets/iso31661Countries.ts
new file mode 100644
index 000000000..4972d9e18
--- /dev/null
+++ b/ui/src/assets/iso31661Countries.ts
@@ -0,0 +1,251 @@
+export const iso31661Countries = [
+ { alpha2: "AF", alpha3: "AFG", code: 4, name: "Afghanistan" },
+ { alpha2: "AL", alpha3: "ALB", code: 8, name: "Albania" },
+ { alpha2: "AQ", alpha3: "ATA", code: 10, name: "Antarctica" },
+ { alpha2: "DZ", alpha3: "DZA", code: 12, name: "Algeria" },
+ { alpha2: "AS", alpha3: "ASM", code: 16, name: "American Samoa" },
+ { alpha2: "AD", alpha3: "AND", code: 20, name: "Andorra" },
+ { alpha2: "AO", alpha3: "AGO", code: 24, name: "Angola" },
+ { alpha2: "AG", alpha3: "ATG", code: 28, name: "Antigua and Barbuda" },
+ { alpha2: "AZ", alpha3: "AZE", code: 31, name: "Azerbaijan" },
+ { alpha2: "AR", alpha3: "ARG", code: 32, name: "Argentina" },
+ { alpha2: "AU", alpha3: "AUS", code: 36, name: "Australia" },
+ { alpha2: "AT", alpha3: "AUT", code: 40, name: "Austria" },
+ { alpha2: "BS", alpha3: "BHS", code: 44, name: "Bahamas" },
+ { alpha2: "BH", alpha3: "BHR", code: 48, name: "Bahrain" },
+ { alpha2: "BD", alpha3: "BGD", code: 50, name: "Bangladesh" },
+ { alpha2: "AM", alpha3: "ARM", code: 51, name: "Armenia" },
+ { alpha2: "BB", alpha3: "BRB", code: 52, name: "Barbados" },
+ { alpha2: "BE", alpha3: "BEL", code: 56, name: "Belgium" },
+ { alpha2: "BM", alpha3: "BMU", code: 60, name: "Bermuda" },
+ { alpha2: "BT", alpha3: "BTN", code: 64, name: "Bhutan" },
+ { alpha2: "BO", alpha3: "BOL", code: 68, name: "Bolivia, Plurinational State of" },
+ { alpha2: "BA", alpha3: "BIH", code: 70, name: "Bosnia and Herzegovina" },
+ { alpha2: "BW", alpha3: "BWA", code: 72, name: "Botswana" },
+ { alpha2: "BV", alpha3: "BVT", code: 74, name: "Bouvet Island" },
+ { alpha2: "BR", alpha3: "BRA", code: 76, name: "Brazil" },
+ { alpha2: "BZ", alpha3: "BLZ", code: 84, name: "Belize" },
+ { alpha2: "IO", alpha3: "IOT", code: 86, name: "British Indian Ocean Territory" },
+ { alpha2: "SB", alpha3: "SLB", code: 90, name: "Solomon Islands" },
+ { alpha2: "VG", alpha3: "VGB", code: 92, name: "Virgin Islands (British)" },
+ { alpha2: "BN", alpha3: "BRN", code: 96, name: "Brunei Darussalam" },
+ { alpha2: "BG", alpha3: "BGR", code: 100, name: "Bulgaria" },
+ { alpha2: "MM", alpha3: "MMR", code: 104, name: "Myanmar" },
+ { alpha2: "BI", alpha3: "BDI", code: 108, name: "Burundi" },
+ { alpha2: "BY", alpha3: "BLR", code: 112, name: "Belarus" },
+ { alpha2: "KH", alpha3: "KHM", code: 116, name: "Cambodia" },
+ { alpha2: "CM", alpha3: "CMR", code: 120, name: "Cameroon" },
+ { alpha2: "CA", alpha3: "CAN", code: 124, name: "Canada" },
+ { alpha2: "CV", alpha3: "CPV", code: 132, name: "Cabo Verde" },
+ { alpha2: "KY", alpha3: "CYM", code: 136, name: "Cayman Islands" },
+ { alpha2: "CF", alpha3: "CAF", code: 140, name: "Central African Republic" },
+ { alpha2: "LK", alpha3: "LKA", code: 144, name: "Sri Lanka" },
+ { alpha2: "TD", alpha3: "TCD", code: 148, name: "Chad" },
+ { alpha2: "CL", alpha3: "CHL", code: 152, name: "Chile" },
+ { alpha2: "CN", alpha3: "CHN", code: 156, name: "China" },
+ { alpha2: "TW", alpha3: "TWN", code: 158, name: "Taiwan, Province of China" },
+ { alpha2: "CX", alpha3: "CXR", code: 162, name: "Christmas Island" },
+ { alpha2: "CC", alpha3: "CCK", code: 166, name: "Cocos (Keeling) Islands" },
+ { alpha2: "CO", alpha3: "COL", code: 170, name: "Colombia" },
+ { alpha2: "KM", alpha3: "COM", code: 174, name: "Comoros" },
+ { alpha2: "YT", alpha3: "MYT", code: 175, name: "Mayotte" },
+ { alpha2: "CG", alpha3: "COG", code: 178, name: "Congo" },
+ { alpha2: "CD", alpha3: "COD", code: 180, name: "Congo, Democratic Republic of the" },
+ { alpha2: "CK", alpha3: "COK", code: 184, name: "Cook Islands" },
+ { alpha2: "CR", alpha3: "CRI", code: 188, name: "Costa Rica" },
+ { alpha2: "HR", alpha3: "HRV", code: 191, name: "Croatia" },
+ { alpha2: "CU", alpha3: "CUB", code: 192, name: "Cuba" },
+ { alpha2: "CY", alpha3: "CYP", code: 196, name: "Cyprus" },
+ { alpha2: "CZ", alpha3: "CZE", code: 203, name: "Czechia" },
+ { alpha2: "BJ", alpha3: "BEN", code: 204, name: "Benin" },
+ { alpha2: "DK", alpha3: "DNK", code: 208, name: "Denmark" },
+ { alpha2: "DM", alpha3: "DMA", code: 212, name: "Dominica" },
+ { alpha2: "DO", alpha3: "DOM", code: 214, name: "Dominican Republic" },
+ { alpha2: "EC", alpha3: "ECU", code: 218, name: "Ecuador" },
+ { alpha2: "SV", alpha3: "SLV", code: 222, name: "El Salvador" },
+ { alpha2: "GQ", alpha3: "GNQ", code: 226, name: "Equatorial Guinea" },
+ { alpha2: "ET", alpha3: "ETH", code: 231, name: "Ethiopia" },
+ { alpha2: "ER", alpha3: "ERI", code: 232, name: "Eritrea" },
+ { alpha2: "EE", alpha3: "EST", code: 233, name: "Estonia" },
+ { alpha2: "FO", alpha3: "FRO", code: 234, name: "Faroe Islands" },
+ { alpha2: "FK", alpha3: "FLK", code: 238, name: "Falkland Islands (Malvinas)" },
+ { alpha2: "GS", alpha3: "SGS", code: 239, name: "South Georgia and the South Sandwich Islands" },
+ { alpha2: "FJ", alpha3: "FJI", code: 242, name: "Fiji" },
+ { alpha2: "FI", alpha3: "FIN", code: 246, name: "Finland" },
+ { alpha2: "AX", alpha3: "ALA", code: 248, name: "Åland Islands" },
+ { alpha2: "FR", alpha3: "FRA", code: 250, name: "France" },
+ { alpha2: "GF", alpha3: "GUF", code: 254, name: "French Guiana" },
+ { alpha2: "PF", alpha3: "PYF", code: 258, name: "French Polynesia" },
+ { alpha2: "TF", alpha3: "ATF", code: 260, name: "French Southern Territories" },
+ { alpha2: "DJ", alpha3: "DJI", code: 262, name: "Djibouti" },
+ { alpha2: "GA", alpha3: "GAB", code: 266, name: "Gabon" },
+ { alpha2: "GE", alpha3: "GEO", code: 268, name: "Georgia" },
+ { alpha2: "GM", alpha3: "GMB", code: 270, name: "Gambia" },
+ { alpha2: "PS", alpha3: "PSE", code: 275, name: "Palestine, State of" },
+ { alpha2: "DE", alpha3: "DEU", code: 276, name: "Germany" },
+ { alpha2: "GH", alpha3: "GHA", code: 288, name: "Ghana" },
+ { alpha2: "GI", alpha3: "GIB", code: 292, name: "Gibraltar" },
+ { alpha2: "KI", alpha3: "KIR", code: 296, name: "Kiribati" },
+ { alpha2: "GR", alpha3: "GRC", code: 300, name: "Greece" },
+ { alpha2: "GL", alpha3: "GRL", code: 304, name: "Greenland" },
+ { alpha2: "GD", alpha3: "GRD", code: 308, name: "Grenada" },
+ { alpha2: "GP", alpha3: "GLP", code: 312, name: "Guadeloupe" },
+ { alpha2: "GU", alpha3: "GUM", code: 316, name: "Guam" },
+ { alpha2: "GT", alpha3: "GTM", code: 320, name: "Guatemala" },
+ { alpha2: "GN", alpha3: "GIN", code: 324, name: "Guinea" },
+ { alpha2: "GY", alpha3: "GUY", code: 328, name: "Guyana" },
+ { alpha2: "HT", alpha3: "HTI", code: 332, name: "Haiti" },
+ { alpha2: "HM", alpha3: "HMD", code: 334, name: "Heard Island and McDonald Islands" },
+ { alpha2: "VA", alpha3: "VAT", code: 336, name: "Holy See" },
+ { alpha2: "HN", alpha3: "HND", code: 340, name: "Honduras" },
+ { alpha2: "HK", alpha3: "HKG", code: 344, name: "Hong Kong" },
+ { alpha2: "HU", alpha3: "HUN", code: 348, name: "Hungary" },
+ { alpha2: "IS", alpha3: "ISL", code: 352, name: "Iceland" },
+ { alpha2: "IN", alpha3: "IND", code: 356, name: "India" },
+ { alpha2: "ID", alpha3: "IDN", code: 360, name: "Indonesia" },
+ { alpha2: "IR", alpha3: "IRN", code: 364, name: "Iran, Islamic Republic of" },
+ { alpha2: "IQ", alpha3: "IRQ", code: 368, name: "Iraq" },
+ { alpha2: "IE", alpha3: "IRL", code: 372, name: "Ireland" },
+ { alpha2: "IL", alpha3: "ISR", code: 376, name: "Israel" },
+ { alpha2: "IT", alpha3: "ITA", code: 380, name: "Italy" },
+ { alpha2: "CI", alpha3: "CIV", code: 384, name: "Côte d'Ivoire" },
+ { alpha2: "JM", alpha3: "JAM", code: 388, name: "Jamaica" },
+ { alpha2: "JP", alpha3: "JPN", code: 392, name: "Japan" },
+ { alpha2: "KZ", alpha3: "KAZ", code: 398, name: "Kazakhstan" },
+ { alpha2: "JO", alpha3: "JOR", code: 400, name: "Jordan" },
+ { alpha2: "KE", alpha3: "KEN", code: 404, name: "Kenya" },
+ { alpha2: "KP", alpha3: "PRK", code: 408, name: "Korea, Democratic People's Republic of" },
+ { alpha2: "KR", alpha3: "KOR", code: 410, name: "Korea, Republic of" },
+ { alpha2: "KW", alpha3: "KWT", code: 414, name: "Kuwait" },
+ { alpha2: "KG", alpha3: "KGZ", code: 417, name: "Kyrgyzstan" },
+ { alpha2: "LA", alpha3: "LAO", code: 418, name: "Lao People's Democratic Republic" },
+ { alpha2: "LB", alpha3: "LBN", code: 422, name: "Lebanon" },
+ { alpha2: "LS", alpha3: "LSO", code: 426, name: "Lesotho" },
+ { alpha2: "LV", alpha3: "LVA", code: 428, name: "Latvia" },
+ { alpha2: "LR", alpha3: "LBR", code: 430, name: "Liberia" },
+ { alpha2: "LY", alpha3: "LBY", code: 434, name: "Libya" },
+ { alpha2: "LI", alpha3: "LIE", code: 438, name: "Liechtenstein" },
+ { alpha2: "LT", alpha3: "LTU", code: 440, name: "Lithuania" },
+ { alpha2: "LU", alpha3: "LUX", code: 442, name: "Luxembourg" },
+ { alpha2: "MO", alpha3: "MAC", code: 446, name: "Macao" },
+ { alpha2: "MG", alpha3: "MDG", code: 450, name: "Madagascar" },
+ { alpha2: "MW", alpha3: "MWI", code: 454, name: "Malawi" },
+ { alpha2: "MY", alpha3: "MYS", code: 458, name: "Malaysia" },
+ { alpha2: "MV", alpha3: "MDV", code: 462, name: "Maldives" },
+ { alpha2: "ML", alpha3: "MLI", code: 466, name: "Mali" },
+ { alpha2: "MT", alpha3: "MLT", code: 470, name: "Malta" },
+ { alpha2: "MQ", alpha3: "MTQ", code: 474, name: "Martinique" },
+ { alpha2: "MR", alpha3: "MRT", code: 478, name: "Mauritania" },
+ { alpha2: "MU", alpha3: "MUS", code: 480, name: "Mauritius" },
+ { alpha2: "MX", alpha3: "MEX", code: 484, name: "Mexico" },
+ { alpha2: "MC", alpha3: "MCO", code: 492, name: "Monaco" },
+ { alpha2: "MN", alpha3: "MNG", code: 496, name: "Mongolia" },
+ { alpha2: "MD", alpha3: "MDA", code: 498, name: "Moldova, Republic of" },
+ { alpha2: "ME", alpha3: "MNE", code: 499, name: "Montenegro" },
+ { alpha2: "MS", alpha3: "MSR", code: 500, name: "Montserrat" },
+ { alpha2: "MA", alpha3: "MAR", code: 504, name: "Morocco" },
+ { alpha2: "MZ", alpha3: "MOZ", code: 508, name: "Mozambique" },
+ { alpha2: "OM", alpha3: "OMN", code: 512, name: "Oman" },
+ { alpha2: "nan", alpha3: "NAM", code: 516, name: "Namibia" },
+ { alpha2: "NR", alpha3: "NRU", code: 520, name: "Nauru" },
+ { alpha2: "NP", alpha3: "NPL", code: 524, name: "Nepal" },
+ { alpha2: "NL", alpha3: "NLD", code: 528, name: "Netherlands, Kingdom of the" },
+ { alpha2: "CW", alpha3: "CUW", code: 531, name: "Curaçao" },
+ { alpha2: "AW", alpha3: "ABW", code: 533, name: "Aruba" },
+ { alpha2: "SX", alpha3: "SXM", code: 534, name: "Sint Maarten (Dutch part)" },
+ { alpha2: "BQ", alpha3: "BES", code: 535, name: "Bonaire, Sint Eustatius and Saba" },
+ { alpha2: "NC", alpha3: "NCL", code: 540, name: "New Caledonia" },
+ { alpha2: "VU", alpha3: "VUT", code: 548, name: "Vanuatu" },
+ { alpha2: "NZ", alpha3: "NZL", code: 554, name: "New Zealand" },
+ { alpha2: "NI", alpha3: "NIC", code: 558, name: "Nicaragua" },
+ { alpha2: "NE", alpha3: "NER", code: 562, name: "Niger" },
+ { alpha2: "NG", alpha3: "NGA", code: 566, name: "Nigeria" },
+ { alpha2: "NU", alpha3: "NIU", code: 570, name: "Niue" },
+ { alpha2: "NF", alpha3: "NFK", code: 574, name: "Norfolk Island" },
+ { alpha2: "NO", alpha3: "NOR", code: 578, name: "Norway" },
+ { alpha2: "MP", alpha3: "MNP", code: 580, name: "Northern Mariana Islands" },
+ { alpha2: "UM", alpha3: "UMI", code: 581, name: "United States Minor Outlying Islands" },
+ { alpha2: "FM", alpha3: "FSM", code: 583, name: "Micronesia, Federated States of" },
+ { alpha2: "MH", alpha3: "MHL", code: 584, name: "Marshall Islands" },
+ { alpha2: "PW", alpha3: "PLW", code: 585, name: "Palau" },
+ { alpha2: "PK", alpha3: "PAK", code: 586, name: "Pakistan" },
+ { alpha2: "PA", alpha3: "PAN", code: 591, name: "Panama" },
+ { alpha2: "PG", alpha3: "PNG", code: 598, name: "Papua New Guinea" },
+ { alpha2: "PY", alpha3: "PRY", code: 600, name: "Paraguay" },
+ { alpha2: "PE", alpha3: "PER", code: 604, name: "Peru" },
+ { alpha2: "PH", alpha3: "PHL", code: 608, name: "Philippines" },
+ { alpha2: "PN", alpha3: "PCN", code: 612, name: "Pitcairn" },
+ { alpha2: "PL", alpha3: "POL", code: 616, name: "Poland" },
+ { alpha2: "PT", alpha3: "PRT", code: 620, name: "Portugal" },
+ { alpha2: "GW", alpha3: "GNB", code: 624, name: "Guinea-Bissau" },
+ { alpha2: "TL", alpha3: "TLS", code: 626, name: "Timor-Leste" },
+ { alpha2: "PR", alpha3: "PRI", code: 630, name: "Puerto Rico" },
+ { alpha2: "QA", alpha3: "QAT", code: 634, name: "Qatar" },
+ { alpha2: "RE", alpha3: "REU", code: 638, name: "Réunion" },
+ { alpha2: "RO", alpha3: "ROU", code: 642, name: "Romania" },
+ { alpha2: "RU", alpha3: "RUS", code: 643, name: "Russian Federation" },
+ { alpha2: "RW", alpha3: "RWA", code: 646, name: "Rwanda" },
+ { alpha2: "BL", alpha3: "BLM", code: 652, name: "Saint Barthélemy" },
+ { alpha2: "SH", alpha3: "SHN", code: 654, name: "Saint Helena, Ascension and Tristan da Cunha" },
+ { alpha2: "KN", alpha3: "KNA", code: 659, name: "Saint Kitts and Nevis" },
+ { alpha2: "AI", alpha3: "AIA", code: 660, name: "Anguilla" },
+ { alpha2: "LC", alpha3: "LCA", code: 662, name: "Saint Lucia" },
+ { alpha2: "MF", alpha3: "MAF", code: 663, name: "Saint Martin (French part)" },
+ { alpha2: "PM", alpha3: "SPM", code: 666, name: "Saint Pierre and Miquelon" },
+ { alpha2: "VC", alpha3: "VCT", code: 670, name: "Saint Vincent and the Grenadines" },
+ { alpha2: "SM", alpha3: "SMR", code: 674, name: "San Marino" },
+ { alpha2: "ST", alpha3: "STP", code: 678, name: "Sao Tome and Principe" },
+ { alpha2: "SA", alpha3: "SAU", code: 682, name: "Saudi Arabia" },
+ { alpha2: "SN", alpha3: "SEN", code: 686, name: "Senegal" },
+ { alpha2: "RS", alpha3: "SRB", code: 688, name: "Serbia" },
+ { alpha2: "SC", alpha3: "SYC", code: 690, name: "Seychelles" },
+ { alpha2: "SL", alpha3: "SLE", code: 694, name: "Sierra Leone" },
+ { alpha2: "SG", alpha3: "SGP", code: 702, name: "Singapore" },
+ { alpha2: "SK", alpha3: "SVK", code: 703, name: "Slovakia" },
+ { alpha2: "VN", alpha3: "VNM", code: 704, name: "Viet Nam" },
+ { alpha2: "SI", alpha3: "SVN", code: 705, name: "Slovenia" },
+ { alpha2: "SO", alpha3: "SOM", code: 706, name: "Somalia" },
+ { alpha2: "ZA", alpha3: "ZAF", code: 710, name: "South Africa" },
+ { alpha2: "ZW", alpha3: "ZWE", code: 716, name: "Zimbabwe" },
+ { alpha2: "ES", alpha3: "ESP", code: 724, name: "Spain" },
+ { alpha2: "SS", alpha3: "SSD", code: 728, name: "South Sudan" },
+ { alpha2: "SD", alpha3: "SDN", code: 729, name: "Sudan" },
+ { alpha2: "EH", alpha3: "ESH", code: 732, name: "Western Sahara" },
+ { alpha2: "SR", alpha3: "SUR", code: 740, name: "Suriname" },
+ { alpha2: "SJ", alpha3: "SJM", code: 744, name: "Svalbard and Jan Mayen" },
+ { alpha2: "SZ", alpha3: "SWZ", code: 748, name: "Eswatini" },
+ { alpha2: "SE", alpha3: "SWE", code: 752, name: "Sweden" },
+ { alpha2: "CH", alpha3: "CHE", code: 756, name: "Switzerland" },
+ { alpha2: "SY", alpha3: "SYR", code: 760, name: "Syrian Arab Republic" },
+ { alpha2: "TJ", alpha3: "TJK", code: 762, name: "Tajikistan" },
+ { alpha2: "TH", alpha3: "THA", code: 764, name: "Thailand" },
+ { alpha2: "TG", alpha3: "TGO", code: 768, name: "Togo" },
+ { alpha2: "TK", alpha3: "TKL", code: 772, name: "Tokelau" },
+ { alpha2: "TO", alpha3: "TON", code: 776, name: "Tonga" },
+ { alpha2: "TT", alpha3: "TTO", code: 780, name: "Trinidad and Tobago" },
+ { alpha2: "AE", alpha3: "ARE", code: 784, name: "United Arab Emirates" },
+ { alpha2: "TN", alpha3: "TUN", code: 788, name: "Tunisia" },
+ { alpha2: "TR", alpha3: "TUR", code: 792, name: "Türkiye" },
+ { alpha2: "TM", alpha3: "TKM", code: 795, name: "Turkmenistan" },
+ { alpha2: "TC", alpha3: "TCA", code: 796, name: "Turks and Caicos Islands" },
+ { alpha2: "TV", alpha3: "TUV", code: 798, name: "Tuvalu" },
+ { alpha2: "UG", alpha3: "UGA", code: 800, name: "Uganda" },
+ { alpha2: "UA", alpha3: "UKR", code: 804, name: "Ukraine" },
+ { alpha2: "MK", alpha3: "MKD", code: 807, name: "North Macedonia" },
+ { alpha2: "EG", alpha3: "EGY", code: 818, name: "Egypt" },
+ { alpha2: "GB", alpha3: "GBR", code: 826, name: "UK & Northern Ireland" },
+ { alpha2: "GG", alpha3: "GGY", code: 831, name: "Guernsey" },
+ { alpha2: "JE", alpha3: "JEY", code: 832, name: "Jersey" },
+ { alpha2: "IM", alpha3: "IMN", code: 833, name: "Isle of Man" },
+ { alpha2: "TZ", alpha3: "TZA", code: 834, name: "Tanzania, United Republic of" },
+ { alpha2: "US", alpha3: "USA", code: 840, name: "United States of America" },
+ { alpha2: "VI", alpha3: "VIR", code: 850, name: "Virgin Islands (U.S.)" },
+ { alpha2: "BF", alpha3: "BFA", code: 854, name: "Burkina Faso" },
+ { alpha2: "UY", alpha3: "URY", code: 858, name: "Uruguay" },
+ { alpha2: "UZ", alpha3: "UZB", code: 860, name: "Uzbekistan" },
+ { alpha2: "VE", alpha3: "VEN", code: 862, name: "Venezuela, Bolivarian Republic of" },
+ { alpha2: "WF", alpha3: "WLF", code: 876, name: "Wallis and Futuna" },
+ { alpha2: "WS", alpha3: "WSM", code: 882, name: "Samoa" },
+ { alpha2: "YE", alpha3: "YEM", code: 887, name: "Yemen" },
+ { alpha2: "ZM", alpha3: "ZMB", code: 894, name: "Zambia" },
+] as const;
diff --git a/ui/src/components/connections/CredentialsTable.tsx b/ui/src/components/connections/CredentialsTable.tsx
index 23b41b537..4d9455441 100644
--- a/ui/src/components/connections/CredentialsTable.tsx
+++ b/ui/src/components/connections/CredentialsTable.tsx
@@ -76,11 +76,17 @@ export function CredentialsTable({ userID }: { userID: string }) {
dataIndex: "schemaType",
ellipsis: { showTitle: false },
key: "schemaType",
- render: (schemaType: Credential["schemaType"]) => (
-
- {schemaType}
-
+ render: (schemaType: Credential["schemaType"], credential: Credential) => (
+
+ navigate(generatePath(ROUTES.credentialDetails.path, { credentialID: credential.id }))
+ }
+ strong
+ >
+ {schemaType}
+
),
+
sorter: ({ schemaType: a }, { schemaType: b }) => a.localeCompare(b),
title: "Credential",
},
diff --git a/ui/src/components/credentials/CredentialDetails.tsx b/ui/src/components/credentials/CredentialDetails.tsx
index 41d32de85..e693d0c29 100644
--- a/ui/src/components/credentials/CredentialDetails.tsx
+++ b/ui/src/components/credentials/CredentialDetails.tsx
@@ -202,6 +202,7 @@ export function CredentialDetails() {
);
} else {
const {
+ credentialStatus,
displayMethod,
expirationDate,
issuanceDate,
@@ -260,13 +261,20 @@ export function CredentialDetails() {
text={expirationDate ? formatDate(expirationDate) : "-"}
/>
+
+
+
+
diff --git a/ui/src/components/credentials/CredentialsTable.tsx b/ui/src/components/credentials/CredentialsTable.tsx
index a61baf69e..efd289d13 100644
--- a/ui/src/components/credentials/CredentialsTable.tsx
+++ b/ui/src/components/credentials/CredentialsTable.tsx
@@ -109,10 +109,15 @@ export function CredentialsTable() {
dataIndex: "schemaType",
ellipsis: { showTitle: false },
key: "schemaType",
- render: (schemaType: Credential["schemaType"]) => (
-
- {schemaType}
-
+ render: (schemaType: Credential["schemaType"], credential: Credential) => (
+
+ navigate(generatePath(ROUTES.credentialDetails.path, { credentialID: credential.id }))
+ }
+ strong
+ >
+ {schemaType}
+
),
sorter: {
multiple: 1,
diff --git a/ui/src/components/credentials/IssueCredentialForm.tsx b/ui/src/components/credentials/IssueCredentialForm.tsx
index 85aa8fa1c..93c83dd65 100644
--- a/ui/src/components/credentials/IssueCredentialForm.tsx
+++ b/ui/src/components/credentials/IssueCredentialForm.tsx
@@ -35,6 +35,7 @@ import {
import IconBack from "src/assets/icons/arrow-narrow-left.svg?react";
import IconRight from "src/assets/icons/arrow-narrow-right.svg?react";
import IconLink from "src/assets/icons/link-external-01.svg?react";
+import { iso31661Countries } from "src/assets/iso31661Countries";
import { InputErrors, ObjectAttributeForm } from "src/components/credentials/ObjectAttributeForm";
import { ErrorResult } from "src/components/shared/ErrorResult";
import { LoadingResult } from "src/components/shared/LoadingResult";
@@ -144,9 +145,8 @@ export function IssueCredentialForm({
}
};
- const isCountryCode = (x: number) => {
- const iso31661NumericRegex = /^\d{1,3}$/;
- return iso31661NumericRegex.test(x.toString());
+ const isCountryCode = (code: number) => {
+ return iso31661Countries.some((country) => country.code === code);
};
function isFormValid(value: Record, objectAttribute: ObjectAttribute): boolean {
diff --git a/ui/src/components/credentials/LinkDetails.tsx b/ui/src/components/credentials/LinkDetails.tsx
index ed89d3c35..45f39df5e 100644
--- a/ui/src/components/credentials/LinkDetails.tsx
+++ b/ui/src/components/credentials/LinkDetails.tsx
@@ -229,7 +229,7 @@ export function LinkDetails() {
/>
diff --git a/ui/src/components/credentials/LinksTable.tsx b/ui/src/components/credentials/LinksTable.tsx
index d9a682527..28c58ac53 100644
--- a/ui/src/components/credentials/LinksTable.tsx
+++ b/ui/src/components/credentials/LinksTable.tsx
@@ -13,7 +13,6 @@ import {
Table,
TableColumnsType,
Tag,
- Tooltip,
Typography,
} from "antd";
@@ -97,10 +96,13 @@ export function LinksTable() {
dataIndex: "schemaType",
ellipsis: true,
key: "schemaType",
- render: (schemaType: Link["schemaType"]) => (
-
- {schemaType}
-
+ render: (schemaType: Link["schemaType"], link: Link) => (
+ navigate(generatePath(ROUTES.linkDetails.path, { linkID: link.id }))}
+ strong
+ >
+ {schemaType}
+
),
sorter: ({ schemaType: a }, { schemaType: b }) => a.localeCompare(b),
title: "Credential",
diff --git a/ui/src/components/identities/IdentitiesTable.tsx b/ui/src/components/identities/IdentitiesTable.tsx
index 8bef8b7dc..4f62204a1 100644
--- a/ui/src/components/identities/IdentitiesTable.tsx
+++ b/ui/src/components/identities/IdentitiesTable.tsx
@@ -8,11 +8,14 @@ import {
Table,
TableColumnsType,
Tag,
+ Tooltip,
Typography,
} from "antd";
import { useCallback, useEffect, useState } from "react";
import { generatePath, useNavigate, useSearchParams } from "react-router-dom";
import IconIssuers from "src/assets/icons/building-08.svg?react";
+import IconCheckMark from "src/assets/icons/check.svg?react";
+import IconCopy from "src/assets/icons/copy-01.svg?react";
import IconDots from "src/assets/icons/dots-vertical.svg?react";
import IconInfoCircle from "src/assets/icons/info-circle.svg?react";
import IconPlus from "src/assets/icons/plus.svg?react";
@@ -81,9 +84,18 @@ export function IdentitiesTable({ handleAddIdentity }: { handleAddIdentity: () =
dataIndex: "displayName",
key: "displayName",
render: (displayName: Identity["displayName"], identity: Identity) => (
-
+
+ navigate(
+ generatePath(ROUTES.identityDetails.path, {
+ identityID: identity.identifier,
+ })
+ )
+ }
+ strong
+ >
{displayName || formatIdentifier(identity.identifier, { short: true })}
-
+
),
sorter: (a, b) =>
(a.displayName || a.identifier).localeCompare(b.displayName || b.identifier),
@@ -93,16 +105,36 @@ export function IdentitiesTable({ handleAddIdentity }: { handleAddIdentity: () =
dataIndex: "identifier",
key: "identifier",
render: (identifier: Identity["identifier"]) => (
- {formatIdentifier(identifier)}
+
+ , ],
+ }}
+ ellipsis={{
+ suffix: identifier.slice(-5),
+ }}
+ >
+ {identifier}
+
+
),
sorter: ({ identifier: a }, { identifier: b }) => a.localeCompare(b),
title: "DID",
},
+ {
+ dataIndex: "credentialStatusType",
+ key: "credentialStatusType",
+ render: (credentialStatusType: Identity["credentialStatusType"]) => (
+ {credentialStatusType}
+ ),
+ sorter: ({ identifier: a }, { identifier: b }) => a.localeCompare(b),
+ title: "Credential status",
+ },
{
dataIndex: "blockchain",
key: "blockchain",
render: (blockchain: Identity["blockchain"]) => (
- {blockchain}
+ {blockchain}
),
sorter: ({ blockchain: a }, { blockchain: b }) => a.localeCompare(b),
title: "Blockchain",
@@ -110,7 +142,7 @@ export function IdentitiesTable({ handleAddIdentity }: { handleAddIdentity: () =
{
dataIndex: "network",
key: "network",
- render: (network: Identity["network"]) => {network},
+ render: (network: Identity["network"]) => {network},
sorter: ({ network: a }, { network: b }) => a.localeCompare(b),
title: "Network",
},
diff --git a/ui/src/components/identities/Identity.tsx b/ui/src/components/identities/Identity.tsx
index 808b019dd..90487a705 100644
--- a/ui/src/components/identities/Identity.tsx
+++ b/ui/src/components/identities/Identity.tsx
@@ -1,25 +1,19 @@
-import { App, Button, Card, Divider, Flex, Form, Input, Space } from "antd";
+import { App, Button, Card, Divider, Flex, Form, Input, Space, Typography } from "antd";
import { useCallback, useEffect, useState } from "react";
import { useParams } from "react-router-dom";
-import { useIdentityContext } from "../../contexts/Identity";
+
import { getIdentity, updateIdentityDisplayName } from "src/adapters/api/identities";
import { IdentityDetailsFormData } from "src/adapters/parsers/view";
-import CheckIcon from "src/assets/icons/check.svg?react";
import EditIcon from "src/assets/icons/edit-02.svg?react";
-import CloseIcon from "src/assets/icons/x-close.svg?react";
import { IdentityAuthCredentials } from "src/components/identities/IdentityAuthCredentials";
import { Detail } from "src/components/shared/Detail";
+import { EditModal } from "src/components/shared/EditModal";
import { ErrorResult } from "src/components/shared/ErrorResult";
import { LoadingResult } from "src/components/shared/LoadingResult";
import { SiderLayoutContent } from "src/components/shared/SiderLayoutContent";
import { useEnvContext } from "src/contexts/Env";
import { AppError, IdentityDetails } from "src/domain";
-import {
- AsyncTask,
- hasAsyncTaskFailed,
- isAsyncTaskDataAvailable,
- isAsyncTaskStarting,
-} from "src/utils/async";
+import { AsyncTask, hasAsyncTaskFailed, isAsyncTaskStarting } from "src/utils/async";
import { isAbortedError, makeRequestAbortable } from "src/utils/browser";
import { IDENTITY_DETAILS, VALUE_REQUIRED } from "src/utils/constants";
import { formatIdentifier } from "src/utils/forms";
@@ -29,9 +23,9 @@ export function Identity() {
const [identity, setIdentity] = useState>({
status: "pending",
});
- const { fetchIdentities, identityList } = useIdentityContext();
- const [displayNameEditable, setDisplayNameEditable] = useState(false);
+ const [isEditModalOpen, setIsEditModalOpen] = useState(false);
+
const { message } = App.useApp();
const [form] = Form.useForm();
@@ -70,27 +64,16 @@ export function Identity() {
return ;
}
- const handleEditDisplayName = (formValues: IdentityDetailsFormData) => {
- const isUnique =
- isAsyncTaskDataAvailable(identityList) &&
- !identityList.data.some(
- (identity) =>
- identity.identifier !== identifier && identity.displayName === formValues.displayName
- );
-
- if (!isUnique) {
- return void message.error(`${formValues.displayName} already exists`);
- }
-
- return void updateIdentityDisplayName({
- displayName: formValues.displayName.trim(),
+ const handleEdit = () => {
+ const { displayName } = form.getFieldsValue();
+ void updateIdentityDisplayName({
+ displayName,
env,
identifier,
}).then((response) => {
+ setIsEditModalOpen(false);
if (response.success) {
void fetchIdentity().then(() => {
- setDisplayNameEditable(false);
- makeRequestAbortable(fetchIdentities);
void message.success("Identity edited successfully");
});
} else {
@@ -130,44 +113,19 @@ export function Identity() {
<>
- {displayNameEditable ? (
-
- ) : (
- <>
- {identity.data.displayName}
- }
- onClick={() => setDisplayNameEditable(true)}
- size="small"
- type="text"
- />
- >
- )}
+
+
+ {identity.data.displayName}
+
+
+ }
+ onClick={() => setIsEditModalOpen(true)}
+ style={{ flexShrink: 0 }}
+ type="text"
+ />
+
}
>
@@ -195,8 +153,31 @@ export function Identity() {
{identity.data.authCredentialsIDs.length && (
-
+
)}
+
+ setIsEditModalOpen(false)}
+ onSubmit={handleEdit}
+ open={isEditModalOpen}
+ title="Edit identity"
+ >
+
+
+
+
+
>
);
}
diff --git a/ui/src/components/identities/IdentityAuthCredentials.tsx b/ui/src/components/identities/IdentityAuthCredentials.tsx
index b4cf048cd..d842cb05e 100644
--- a/ui/src/components/identities/IdentityAuthCredentials.tsx
+++ b/ui/src/components/identities/IdentityAuthCredentials.tsx
@@ -28,7 +28,6 @@ import { CredentialRevokeModal } from "src/components/shared/CredentialRevokeMod
import { TableCard } from "src/components/shared/TableCard";
import { useEnvContext } from "src/contexts/Env";
-import { useIdentityContext } from "src/contexts/Identity";
import { AppError, AuthCredential } from "src/domain";
import { ROUTES } from "src/routes";
import { AsyncTask, isAsyncTaskDataAvailable, isAsyncTaskStarting } from "src/utils/async";
@@ -36,9 +35,14 @@ import { isAbortedError, makeRequestAbortable } from "src/utils/browser";
import { DOTS_DROPDOWN_WIDTH, ISSUE_DATE, REVOCATION, REVOKE } from "src/utils/constants";
import { formatDate } from "src/utils/forms";
-export function IdentityAuthCredentials({ IDs }: { IDs: Array }) {
+export function IdentityAuthCredentials({
+ identityID,
+ IDs,
+}: {
+ IDs: Array;
+ identityID: string;
+}) {
const env = useEnvContext();
- const { identifier } = useIdentityContext();
const navigate = useNavigate();
const [credentials, setCredentials] = useState>({
@@ -57,7 +61,7 @@ export function IdentityAuthCredentials({ IDs }: { IDs: Array }) {
const response = await getAuthCredentialsByIDs({
env,
- identifier,
+ identifier: identityID,
IDs,
signal,
});
@@ -74,7 +78,7 @@ export function IdentityAuthCredentials({ IDs }: { IDs: Array }) {
}
}
},
- [env, identifier, IDs]
+ [env, identityID, IDs]
);
const tableColumns: TableColumnsType = [
diff --git a/ui/src/components/schemas/SchemasTable.tsx b/ui/src/components/schemas/SchemasTable.tsx
index ce181a3d1..16ee0d4aa 100644
--- a/ui/src/components/schemas/SchemasTable.tsx
+++ b/ui/src/components/schemas/SchemasTable.tsx
@@ -13,7 +13,7 @@ import {
} from "antd";
import { useCallback, useEffect, useState } from "react";
-import { Link, generatePath, useSearchParams } from "react-router-dom";
+import { Link, generatePath, useNavigate, useSearchParams } from "react-router-dom";
import { getApiSchemas } from "src/adapters/api/schemas";
import { notifyErrors } from "src/adapters/parsers";
@@ -40,7 +40,7 @@ import { formatDate } from "src/utils/forms";
export function SchemasTable() {
const env = useEnvContext();
const { identifier } = useIdentityContext();
-
+ const navigate = useNavigate();
const [apiSchemas, setApiSchemas] = useState>({
status: "pending",
});
@@ -56,12 +56,23 @@ export function SchemasTable() {
dataIndex: "type",
ellipsis: { showTitle: false },
key: "type",
- render: (type: ApiSchema["type"], { description, title }: ApiSchema) => (
+ render: (type: ApiSchema["type"], { description, id, title }: ApiSchema) => (
- {type}
+
+ navigate(
+ generatePath(ROUTES.schemaDetails.path, {
+ schemaID: id,
+ })
+ )
+ }
+ strong
+ >
+ {type}
+
),
sorter: {
@@ -74,7 +85,7 @@ export function SchemasTable() {
dataIndex: "version",
key: "version",
render: (version: ApiSchema["version"]) => (
- {version || "-"}
+ {version || "-"}
),
sorter: {
compare: ({ version: a }, { version: b }) => (a && b ? a.localeCompare(b) : 0),
diff --git a/ui/src/components/shared/SiderMenu.tsx b/ui/src/components/shared/SiderMenu.tsx
index d046adbcb..4c88977d8 100644
--- a/ui/src/components/shared/SiderMenu.tsx
+++ b/ui/src/components/shared/SiderMenu.tsx
@@ -51,7 +51,7 @@ export function SiderMenu({
const schemasPath = ROUTES.schemas.path;
const identitiesPath = ROUTES.identities.path;
const displayMethodsPath = ROUTES.displayMethods.path;
- const keysPath = ROUTES.keys.path;
+ const keysPath = ROUTES.keys.path;
const getSelectedKey = (): string[] => {
if (
@@ -106,12 +106,12 @@ export function SiderMenu({
) {
return [displayMethodsPath];
} else if (
- matchRoutes(
- [{ path: keysPath }, { path: ROUTES.keyDetails.path }, { path: ROUTES.createKey.path }],
- pathname
- )
+ matchRoutes(
+ [{ path: keysPath }, { path: ROUTES.keyDetails.path }, { path: ROUTES.createKey.path }],
+ pathname
+ )
) {
- return [keysPath];
+ return [keysPath];
}
return [];
@@ -184,13 +184,7 @@ export function SiderMenu({
onClick: () => onMenuClick(issuerStatePath),
title: "",
},
- {
- icon: ,
- key: identitiesPath,
- label: IDENTITIES,
- onClick: () => onMenuClick(identitiesPath),
- title: "",
- },
+
{
icon: ,
key: keysPath,
@@ -207,6 +201,13 @@ export function SiderMenu({
,
+ key: identitiesPath,
+ label: IDENTITIES,
+ onClick: () => onMenuClick(identitiesPath),
+ title: "",
+ },
{
icon: ,
key: "settings",
@@ -236,6 +237,7 @@ export function SiderMenu({
},
]}
selectable={false}
+ selectedKeys={getSelectedKey()}
/>
{isBreakpoint && (
diff --git a/ui/src/styles/index.scss b/ui/src/styles/index.scss
index fdee55dae..c23895457 100644
--- a/ui/src/styles/index.scss
+++ b/ui/src/styles/index.scss
@@ -347,6 +347,10 @@
}
}
+.ant-layout-sider-children {
+ overflow: auto;
+}
+
/* CUSTOM */
.background-grey {