diff --git a/package-lock.json b/package-lock.json
index 8e3eb38a..b2b6da12 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -17,6 +17,7 @@
"@mantine/core": "^7.14.1",
"@mantine/hooks": "^7.14.1",
"@radix-ui/react-accordion": "^1.1.2",
+ "@radix-ui/react-checkbox": "^1.3.1",
"@radix-ui/react-context-menu": "^2.2.1",
"@radix-ui/react-dialog": "^1.0.5",
"@radix-ui/react-dropdown-menu": "^2.0.6",
@@ -30,10 +31,12 @@
"@radix-ui/react-slot": "^1.0.2",
"@radix-ui/react-switch": "^1.0.3",
"@radix-ui/react-tabs": "^1.1.1",
+ "@radix-ui/react-toast": "^1.2.13",
"@radix-ui/react-toggle": "^1.1.0",
"@radix-ui/react-toggle-group": "^1.1.0",
"@radix-ui/react-tooltip": "^1.0.7",
"@tanstack/react-router": "^1.46.4",
+ "@tanstack/react-table": "^8.21.3",
"@uidotdev/usehooks": "^2.4.1",
"axios": "^1.7.7",
"axios-retry": "^4.5.0",
@@ -47,6 +50,7 @@
"i18next": "^24.1.2",
"immer": "^10.1.1",
"just-random": "^3.2.0",
+ "just-split": "^3.2.0",
"lucide-react": "^0.372.0",
"modern-screenshot": "^4.6.0",
"next-themes": "^0.3.0",
@@ -2774,6 +2778,204 @@
}
}
},
+ "node_modules/@radix-ui/react-checkbox": {
+ "version": "1.3.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-checkbox/-/react-checkbox-1.3.1.tgz",
+ "integrity": "sha512-xTaLKAO+XXMPK/BpVTSaAAhlefmvMSACjIhK9mGsImvX2ljcTDm8VGR1CuS1uYcNdR5J+oiOhoJZc5un6bh3VQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/primitive": "1.1.2",
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-context": "1.1.2",
+ "@radix-ui/react-presence": "1.1.4",
+ "@radix-ui/react-primitive": "2.1.2",
+ "@radix-ui/react-use-controllable-state": "1.2.2",
+ "@radix-ui/react-use-previous": "1.1.1",
+ "@radix-ui/react-use-size": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/primitive": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/primitive/-/primitive-1.1.2.tgz",
+ "integrity": "sha512-XnbHrrprsNqZKQhStrSwgRUQzoCI1glLzdw79xiZPoofhGICeZRSQ3dIxAKH1gb3OHfNf4d6f+vAv3kil2eggA==",
+ "license": "MIT"
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-compose-refs": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz",
+ "integrity": "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-context": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-context/-/react-context-1.1.2.tgz",
+ "integrity": "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-presence": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-presence/-/react-presence-1.1.4.tgz",
+ "integrity": "sha512-ueDqRbdc4/bkaQT3GIpLQssRlFgWaL/U2z/S31qRwwLWoxHLgry3SIfCwhxeQNbirEUXFa+lq3RL3oBYXtcmIA==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-primitive": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-primitive/-/react-primitive-2.1.2.tgz",
+ "integrity": "sha512-uHa+l/lKfxuDD2zjN/0peM/RhhSmRjr5YWdk/37EnSv1nJ88uvG85DPexSm8HdFQROd2VdERJ6ynXbkCFi+APw==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-slot": "1.2.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-slot": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.2.tgz",
+ "integrity": "sha512-y7TBO4xN4Y94FvcWIOIh18fM4R1A8S4q1jhoz4PNzOoHsFcN8pogcFmZrTYAm4F9VRUrWP/Mw7xSKybIeRI+CQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-compose-refs": "1.1.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-use-controllable-state": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-controllable-state/-/react-use-controllable-state-1.2.2.tgz",
+ "integrity": "sha512-BjasUjixPFdS+NKkypcyyN5Pmg83Olst0+c6vGov0diwTEo6mgdqVR6hxcEgFuh4QrAs7Rc+9KuGJ9TVCj0Zzg==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-use-effect-event": "0.0.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-use-layout-effect": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.1.tgz",
+ "integrity": "sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-use-previous": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-previous/-/react-use-previous-1.1.1.tgz",
+ "integrity": "sha512-2dHfToCj/pzca2Ck724OZ5L0EVrr3eHRNsG/b3xQJLA2hZpVCS99bLAX+hm1IHXDEnzU6by5z/5MIY794/a8NQ==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-checkbox/node_modules/@radix-ui/react-use-size": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-size/-/react-use-size-1.1.1.tgz",
+ "integrity": "sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
"node_modules/@radix-ui/react-collapsible": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/@radix-ui/react-collapsible/-/react-collapsible-1.0.3.tgz",
@@ -4976,6 +5178,308 @@
}
}
},
+ "node_modules/@radix-ui/react-toast": {
+ "version": "1.2.13",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-toast/-/react-toast-1.2.13.tgz",
+ "integrity": "sha512-e/e43mQAwgYs8BY4y9l99xTK6ig1bK2uXsFLOMn9IZ16lAgulSTsotcPHVT2ZlSb/ye6Sllq7IgyDB8dGhpeXQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/primitive": "1.1.2",
+ "@radix-ui/react-collection": "1.1.6",
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-context": "1.1.2",
+ "@radix-ui/react-dismissable-layer": "1.1.9",
+ "@radix-ui/react-portal": "1.1.8",
+ "@radix-ui/react-presence": "1.1.4",
+ "@radix-ui/react-primitive": "2.1.2",
+ "@radix-ui/react-use-callback-ref": "1.1.1",
+ "@radix-ui/react-use-controllable-state": "1.2.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1",
+ "@radix-ui/react-visually-hidden": "1.2.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/primitive": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/primitive/-/primitive-1.1.2.tgz",
+ "integrity": "sha512-XnbHrrprsNqZKQhStrSwgRUQzoCI1glLzdw79xiZPoofhGICeZRSQ3dIxAKH1gb3OHfNf4d6f+vAv3kil2eggA==",
+ "license": "MIT"
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-collection": {
+ "version": "1.1.6",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-collection/-/react-collection-1.1.6.tgz",
+ "integrity": "sha512-PbhRFK4lIEw9ADonj48tiYWzkllz81TM7KVYyyMMw2cwHO7D5h4XKEblL8NlaRisTK3QTe6tBEhDccFUryxHBQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-context": "1.1.2",
+ "@radix-ui/react-primitive": "2.1.2",
+ "@radix-ui/react-slot": "1.2.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-compose-refs": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.2.tgz",
+ "integrity": "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-context": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-context/-/react-context-1.1.2.tgz",
+ "integrity": "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-dismissable-layer": {
+ "version": "1.1.9",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.1.9.tgz",
+ "integrity": "sha512-way197PiTvNp+WBP7svMJasHl+vibhWGQDb6Mgf5mhEWJkgb85z7Lfl9TUdkqpWsf8GRNmoopx9ZxCyDzmgRMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/primitive": "1.1.2",
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-primitive": "2.1.2",
+ "@radix-ui/react-use-callback-ref": "1.1.1",
+ "@radix-ui/react-use-escape-keydown": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-portal": {
+ "version": "1.1.8",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-portal/-/react-portal-1.1.8.tgz",
+ "integrity": "sha512-hQsTUIn7p7fxCPvao/q6wpbxmCwgLrlz+nOrJgC+RwfZqWY/WN+UMqkXzrtKbPrF82P43eCTl3ekeKuyAQbFeg==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-primitive": "2.1.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-presence": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-presence/-/react-presence-1.1.4.tgz",
+ "integrity": "sha512-ueDqRbdc4/bkaQT3GIpLQssRlFgWaL/U2z/S31qRwwLWoxHLgry3SIfCwhxeQNbirEUXFa+lq3RL3oBYXtcmIA==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-compose-refs": "1.1.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-primitive": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-primitive/-/react-primitive-2.1.2.tgz",
+ "integrity": "sha512-uHa+l/lKfxuDD2zjN/0peM/RhhSmRjr5YWdk/37EnSv1nJ88uvG85DPexSm8HdFQROd2VdERJ6ynXbkCFi+APw==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-slot": "1.2.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-slot": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-slot/-/react-slot-1.2.2.tgz",
+ "integrity": "sha512-y7TBO4xN4Y94FvcWIOIh18fM4R1A8S4q1jhoz4PNzOoHsFcN8pogcFmZrTYAm4F9VRUrWP/Mw7xSKybIeRI+CQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-compose-refs": "1.1.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-use-callback-ref": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-callback-ref/-/react-use-callback-ref-1.1.1.tgz",
+ "integrity": "sha512-FkBMwD+qbGQeMu1cOHnuGB6x4yzPjho8ap5WtbEJ26umhgqVXbhekKUQO+hZEL1vU92a3wHwdp0HAcqAUF5iDg==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-use-controllable-state": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-controllable-state/-/react-use-controllable-state-1.2.2.tgz",
+ "integrity": "sha512-BjasUjixPFdS+NKkypcyyN5Pmg83Olst0+c6vGov0diwTEo6mgdqVR6hxcEgFuh4QrAs7Rc+9KuGJ9TVCj0Zzg==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-use-effect-event": "0.0.2",
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-use-escape-keydown": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.1.1.tgz",
+ "integrity": "sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-use-callback-ref": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-use-layout-effect": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.1.tgz",
+ "integrity": "sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-toast/node_modules/@radix-ui/react-visually-hidden": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-visually-hidden/-/react-visually-hidden-1.2.2.tgz",
+ "integrity": "sha512-ORCmRUbNiZIv6uV5mhFrhsIKw4UX/N3syZtyqvry61tbGm4JlgQuSn0hk5TwCARsCjkcnuRkSdCE3xfb+ADHew==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-primitive": "2.1.2"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "@types/react-dom": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "@types/react-dom": {
+ "optional": true
+ }
+ }
+ },
"node_modules/@radix-ui/react-toggle": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/@radix-ui/react-toggle/-/react-toggle-1.1.0.tgz",
@@ -5403,6 +5907,39 @@
}
}
},
+ "node_modules/@radix-ui/react-use-effect-event": {
+ "version": "0.0.2",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-effect-event/-/react-use-effect-event-0.0.2.tgz",
+ "integrity": "sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==",
+ "license": "MIT",
+ "dependencies": {
+ "@radix-ui/react-use-layout-effect": "1.1.1"
+ },
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@radix-ui/react-use-effect-event/node_modules/@radix-ui/react-use-layout-effect": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/@radix-ui/react-use-layout-effect/-/react-use-layout-effect-1.1.1.tgz",
+ "integrity": "sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==",
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "*",
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ }
+ }
+ },
"node_modules/@radix-ui/react-use-escape-keydown": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/@radix-ui/react-use-escape-keydown/-/react-use-escape-keydown-1.0.3.tgz",
@@ -6005,6 +6542,26 @@
"react-dom": "^17.0.0 || ^18.0.0"
}
},
+ "node_modules/@tanstack/react-table": {
+ "version": "8.21.3",
+ "resolved": "https://registry.npmjs.org/@tanstack/react-table/-/react-table-8.21.3.tgz",
+ "integrity": "sha512-5nNMTSETP4ykGegmVkhjcS8tTLW6Vl4axfEGQN3v0zdHYbK4UfoqfPChclTrJ4EoK9QynqAu9oUf8VEmrpZ5Ww==",
+ "license": "MIT",
+ "dependencies": {
+ "@tanstack/table-core": "8.21.3"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ },
+ "peerDependencies": {
+ "react": ">=16.8",
+ "react-dom": ">=16.8"
+ }
+ },
"node_modules/@tanstack/router-generator": {
"version": "1.30.0",
"resolved": "https://registry.npmjs.org/@tanstack/router-generator/-/router-generator-1.30.0.tgz",
@@ -6062,6 +6619,19 @@
"url": "https://github.com/sponsors/tannerlinsley"
}
},
+ "node_modules/@tanstack/table-core": {
+ "version": "8.21.3",
+ "resolved": "https://registry.npmjs.org/@tanstack/table-core/-/table-core-8.21.3.tgz",
+ "integrity": "sha512-ldZXEhOBb8Is7xLs01fR3YEc3DERiz5silj8tnGkFZytt1abEvl/GhUmCE0PMLaMPTa3Jk4HbKmRlHmu+gCftg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ }
+ },
"node_modules/@tootallnate/once": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-2.0.0.tgz",
@@ -11070,6 +11640,12 @@
"integrity": "sha512-RMf8vbtCfLIbAEHvIPu2FwMkpB/JudGyk/VPfqPItcRgt7k8QnV+Aa7s7kRFPo+bavQkUi8Yg1x/ooW6Ttyb9A==",
"license": "MIT"
},
+ "node_modules/just-split": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/just-split/-/just-split-3.2.0.tgz",
+ "integrity": "sha512-hh57dN5koTBkmg3T6gBFISVVaW5bgZ6Ct1W5KODD5M7hQJKqGzTKkfMwOil8MBxyztLQEjh/v6UGXE8cP5tnqQ==",
+ "license": "MIT"
+ },
"node_modules/keyv": {
"version": "4.5.4",
"resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
diff --git a/package.json b/package.json
index a988e0be..aa9f8d6f 100644
--- a/package.json
+++ b/package.json
@@ -37,6 +37,7 @@
"@mantine/core": "^7.14.1",
"@mantine/hooks": "^7.14.1",
"@radix-ui/react-accordion": "^1.1.2",
+ "@radix-ui/react-checkbox": "^1.3.1",
"@radix-ui/react-context-menu": "^2.2.1",
"@radix-ui/react-dialog": "^1.0.5",
"@radix-ui/react-dropdown-menu": "^2.0.6",
@@ -50,10 +51,12 @@
"@radix-ui/react-slot": "^1.0.2",
"@radix-ui/react-switch": "^1.0.3",
"@radix-ui/react-tabs": "^1.1.1",
+ "@radix-ui/react-toast": "^1.2.13",
"@radix-ui/react-toggle": "^1.1.0",
"@radix-ui/react-toggle-group": "^1.1.0",
"@radix-ui/react-tooltip": "^1.0.7",
"@tanstack/react-router": "^1.46.4",
+ "@tanstack/react-table": "^8.21.3",
"@uidotdev/usehooks": "^2.4.1",
"axios": "^1.7.7",
"axios-retry": "^4.5.0",
@@ -67,6 +70,7 @@
"i18next": "^24.1.2",
"immer": "^10.1.1",
"just-random": "^3.2.0",
+ "just-split": "^3.2.0",
"lucide-react": "^0.372.0",
"modern-screenshot": "^4.6.0",
"next-themes": "^0.3.0",
diff --git a/src/app.tsx b/src/app.tsx
index e2de9bc7..32a9ccbe 100644
--- a/src/app.tsx
+++ b/src/app.tsx
@@ -24,7 +24,8 @@ import { LoadGroups } from './bootstrap/components/load-groups'
import { LoadSettings } from './bootstrap/components/load-settings'
import { LoadTags } from './bootstrap/components/load-tags'
-import { Toaster } from './components/ui/sonner'
+import { SonnerToaster } from './components/ui/sonner'
+import { Toaster } from './components/ui/toaster'
import { ThemeProvider } from './components/theme-provider'
import 'dayjs/locale/es'
@@ -65,7 +66,8 @@ root.render(
defaultNotFoundComponent={IndexComponent}
/>
-
+
+
)
diff --git a/src/bootstrap/components/load-friends.tsx b/src/bootstrap/components/load-friends.tsx
index 13e18b43..0d4e248b 100644
--- a/src/bootstrap/components/load-friends.tsx
+++ b/src/bootstrap/components/load-friends.tsx
@@ -1,9 +1,20 @@
import { useEffect } from 'react'
import { usePartyFriendsForm } from '../../hooks/stw-operations/party'
+import { useFriendsManagementActions } from '../../hooks/management/friends'
+
+import { toast } from '../../lib/notifications'
export function LoadFriends() {
const { syncFriends } = usePartyFriendsForm()
+ const {
+ removeFriends,
+ syncBlocklist,
+ syncIncoming,
+ syncOutgoing,
+ syncSummary,
+ updateLoading,
+ } = useFriendsManagementActions()
useEffect(() => {
const listener = window.electronAPI.notificationLoadFriends(
@@ -19,5 +30,99 @@ export function LoadFriends() {
}
}, [])
+ useEffect(() => {
+ const summaryListener = window.electronAPI.notificationFriendsSummary(
+ async (accountId, summary) => {
+ updateLoading(false)
+ syncSummary(accountId, summary)
+ }
+ )
+ const blockFriendsListener =
+ window.electronAPI.notificationBlockFriends(
+ async (account, blocklist, context) => {
+ if (context === undefined) {
+ removeFriends(
+ account.accountId,
+ blocklist.map((item) => item.accountId)
+ )
+ } else if (context === 'incoming') {
+ syncIncoming('remove', account.accountId, blocklist)
+ } else if (context === 'outgoing') {
+ syncOutgoing('remove', account.accountId, blocklist)
+ }
+
+ syncBlocklist('add', account.accountId, blocklist)
+
+ toast(`Total bloqueados: ${blocklist.length}`)
+ }
+ )
+ const unblockFriendsListener =
+ window.electronAPI.notificationUnblockFriends(
+ async (account, unblocklist) => {
+ syncBlocklist(
+ 'remove',
+ account.accountId,
+ unblocklist === 'full' ? undefined : unblocklist
+ )
+
+ toast(
+ unblocklist === 'full'
+ ? 'Se desbloquearon a todos'
+ : `Total desbloqueados: ${unblocklist.length}`
+ )
+ }
+ )
+ const addFriendsListener = window.electronAPI.notificationAddFriends(
+ async (account, friends, context) => {
+ if (context === undefined) {
+ //
+ } else if (context === 'incoming') {
+ syncIncoming('remove', account.accountId, friends)
+ }
+
+ toast(`Total aƱadidos: ${friends.length}`)
+ }
+ )
+ const removeFriendsListener =
+ window.electronAPI.notificationRemoveFriends(
+ async (account, friends, context) => {
+ const isFull = friends === 'full'
+
+ if (context === undefined) {
+ removeFriends(
+ account.accountId,
+ isFull ? undefined : friends.map((item) => item.accountId)
+ )
+ } else if (context === 'incoming') {
+ syncIncoming(
+ 'remove',
+ account.accountId,
+ isFull ? undefined : friends
+ )
+ } else if (context === 'outgoing') {
+ syncOutgoing(
+ 'remove',
+ account.accountId,
+ isFull ? undefined : friends
+ )
+ }
+
+ toast(
+ isFull
+ ? 'Se eliminaron todos los amigos'
+ : `Total eliminados: ${friends.length}`
+ )
+ }
+ )
+
+ return () => {
+ summaryListener.removeListener()
+ blockFriendsListener.removeListener()
+ unblockFriendsListener.removeListener()
+ addFriendsListener.removeListener()
+ removeFriendsListener.removeListener()
+ }
+ }, [])
+
return null
}
diff --git a/src/components/menu/sidebar.tsx b/src/components/menu/sidebar.tsx
index 5e45e6f8..c9c1f21d 100644
--- a/src/components/menu/sidebar.tsx
+++ b/src/components/menu/sidebar.tsx
@@ -270,6 +270,23 @@ export function SidebarMenu({
)}
+ {getMenuOptionVisibility('friendsManagement') && (
+
+
+ {t(
+ 'account-management.options.friends-management'
+ )}
+
+
+ )}
{getMenuOptionVisibility('redeemCodes') && (
,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+
+
+
+
+))
+Checkbox.displayName = CheckboxPrimitive.Root.displayName
+
+export { Checkbox }
diff --git a/src/components/ui/extended/form/table.tsx b/src/components/ui/extended/form/table.tsx
new file mode 100644
index 00000000..7bd691d6
--- /dev/null
+++ b/src/components/ui/extended/form/table.tsx
@@ -0,0 +1,298 @@
+import type {
+ ColumnDef,
+ Table as TableDefinition,
+} from '@tanstack/react-table'
+
+import {
+ // ColumnFiltersState,
+ // Row,
+ // SortingState,
+ // VisibilityState,
+ flexRender,
+ getCoreRowModel,
+ getFacetedRowModel,
+ getFacetedUniqueValues,
+ getFilteredRowModel,
+ getPaginationRowModel,
+ getSortedRowModel,
+ useReactTable,
+} from '@tanstack/react-table'
+import {
+ ChevronLeftIcon,
+ ChevronRightIcon,
+ ChevronsLeftIcon,
+ ChevronsRightIcon,
+} from 'lucide-react'
+import { useState } from 'react'
+
+import { Button } from '../../button'
+import { Input } from '../../input'
+import {
+ Select,
+ SelectContent,
+ SelectItem,
+ SelectTrigger,
+ SelectValue,
+} from '../../select'
+import {
+ Table,
+ TableBody,
+ TableCell,
+ TableHead,
+ TableHeader,
+ TableRow,
+} from '../../table'
+
+import { cn } from '../../../../lib/utils'
+
+const defaultPerPageList = [10, 20, 30, 40, 50, 100]
+
+export function useTableConfig>({
+ columns,
+ data,
+ perPageList = defaultPerPageList,
+ rowId,
+
+ defaultPageSize = 50,
+}: {
+ columns: Array>
+ data: Array
+ perPageList?: Array
+ rowId: keyof Data
+
+ defaultPageSize?: number
+}) {
+ const [globalFilter, setGlobalFilter] = useState('')
+ const [rowSelection, setRowSelection] = useState({})
+ const [pagination, setPagination] = useState({
+ pageIndex: 0,
+ pageSize: defaultPageSize,
+ })
+
+ const table = useReactTable({
+ data,
+ columns,
+ enableRowSelection: true,
+ globalFilterFn: 'includesString',
+ state: {
+ // columnFilters,
+ // columnVisibility,
+ globalFilter,
+ pagination,
+ rowSelection,
+ // sorting,
+ },
+
+ getRowId: (row) => row[rowId] as string,
+ getFacetedRowModel: getFacetedRowModel(),
+ getFacetedUniqueValues: getFacetedUniqueValues(),
+ getCoreRowModel: getCoreRowModel(),
+ getFilteredRowModel: getFilteredRowModel(),
+ getPaginationRowModel: getPaginationRowModel(),
+ getSortedRowModel: getSortedRowModel(),
+
+ // onColumnFiltersChange: setColumnFilters,
+ // onColumnVisibilityChange: setColumnVisibility,
+ onGlobalFilterChange: setGlobalFilter,
+ onPaginationChange: setPagination,
+ onRowSelectionChange: setRowSelection,
+ // onSortingChange: setSorting,
+ })
+
+ return {
+ table,
+ perPageList,
+ }
+}
+
+export function CustomTable>({
+ table,
+}: {
+ table: TableDefinition
+}) {
+ return (
+
+
+ {table.getHeaderGroups().map((headerGroup) => (
+ th:first-child]:w-10',
+ '[&>th:nth-child(2)]:w-44'
+ )}
+ key={headerGroup.id}
+ >
+ {headerGroup.headers.map((header) => {
+ return (
+
+ {header.isPlaceholder
+ ? null
+ : flexRender(
+ header.column.columnDef.header,
+ header.getContext()
+ )}
+
+ )
+ })}
+
+ ))}
+
+
+ {table.getRowModel().rows?.length ? (
+ table.getRowModel().rows.map((row, index) => (
+
+ {row.getVisibleCells().map((cell) => (
+
+ {flexRender(
+ cell.column.columnDef.cell,
+ cell.getContext()
+ )}
+
+ ))}
+
+ ))
+ ) : (
+
+
+ No results.
+
+
+ )}
+
+
+ )
+}
+
+export function CustomSearch>({
+ isDisabled,
+ table,
+}: {
+ isDisabled?: boolean
+ table: TableDefinition
+}) {
+ return (
+
+ table.setGlobalFilter(String(event.target.value).trim())
+ }
+ disabled={isDisabled}
+ />
+ )
+}
+
+export function CustomTablePagination<
+ Data extends Record,
+>({
+ perPageList = defaultPerPageList,
+ table,
+}: {
+ perPageList?: Array
+ table: TableDefinition
+}) {
+ if (table.getPageCount() <= 0) {
+ return null
+ }
+
+ return (
+
+ {/*
+ {table.getFilteredSelectedRowModel().rows.length} of{' '}
+ {table.getFilteredRowModel().rows.length} row(s) selected.
+
*/}
+
+
+ {/* */}
+
+
+
+ Page {table.getState().pagination.pageIndex + 1} of{' '}
+ {table.getPageCount()}
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/components/ui/sonner.tsx b/src/components/ui/sonner.tsx
index 430ab4fb..f489ba01 100644
--- a/src/components/ui/sonner.tsx
+++ b/src/components/ui/sonner.tsx
@@ -4,7 +4,7 @@ import { Toaster as Sonner } from 'sonner'
type ToasterProps = React.ComponentProps
-const Toaster = ({ ...props }: ToasterProps) => {
+const SonnerToaster = ({ ...props }: ToasterProps) => {
const { theme = 'light' } = useTheme()
return (
@@ -24,4 +24,4 @@ const Toaster = ({ ...props }: ToasterProps) => {
)
}
-export { Toaster }
+export { SonnerToaster }
diff --git a/src/components/ui/table.tsx b/src/components/ui/table.tsx
new file mode 100644
index 00000000..8a6681c9
--- /dev/null
+++ b/src/components/ui/table.tsx
@@ -0,0 +1,130 @@
+import type {
+ HTMLAttributes,
+ TdHTMLAttributes,
+ ThHTMLAttributes,
+} from 'react'
+
+import { forwardRef } from 'react'
+
+import { cn } from '../../lib/utils'
+
+const Table = forwardRef<
+ HTMLTableElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+
+))
+Table.displayName = 'Table'
+
+const TableHeader = forwardRef<
+ HTMLTableSectionElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+
+))
+TableHeader.displayName = 'TableHeader'
+
+const TableBody = forwardRef<
+ HTMLTableSectionElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+
+))
+TableBody.displayName = 'TableBody'
+
+const TableFooter = forwardRef<
+ HTMLTableSectionElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+ tr]:last:border-b-0',
+ className
+ )}
+ {...props}
+ />
+))
+TableFooter.displayName = 'TableFooter'
+
+const TableRow = forwardRef<
+ HTMLTableRowElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+
+))
+TableRow.displayName = 'TableRow'
+
+const TableHead = forwardRef<
+ HTMLTableCellElement,
+ ThHTMLAttributes
+>(({ className, ...props }, ref) => (
+ |
+))
+TableHead.displayName = 'TableHead'
+
+const TableCell = forwardRef<
+ HTMLTableCellElement,
+ TdHTMLAttributes
+>(({ className, ...props }, ref) => (
+ |
+))
+TableCell.displayName = 'TableCell'
+
+const TableCaption = forwardRef<
+ HTMLTableCaptionElement,
+ HTMLAttributes
+>(({ className, ...props }, ref) => (
+
+))
+TableCaption.displayName = 'TableCaption'
+
+export {
+ Table,
+ TableHeader,
+ TableBody,
+ TableFooter,
+ TableHead,
+ TableRow,
+ TableCell,
+ TableCaption,
+}
diff --git a/src/components/ui/toast.tsx b/src/components/ui/toast.tsx
new file mode 100644
index 00000000..ea43d6ec
--- /dev/null
+++ b/src/components/ui/toast.tsx
@@ -0,0 +1,134 @@
+import type { VariantProps } from 'class-variance-authority'
+import type {
+ ComponentPropsWithoutRef,
+ ElementRef,
+ ReactElement,
+} from 'react'
+
+import * as ToastPrimitives from '@radix-ui/react-toast'
+import { cva } from 'class-variance-authority'
+import { X } from 'lucide-react'
+import { forwardRef } from 'react'
+
+import { cn } from '../../lib/utils'
+
+const ToastProvider = ToastPrimitives.Provider
+
+const ToastViewport = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+))
+ToastViewport.displayName = ToastPrimitives.Viewport.displayName
+
+const toastVariants = cva(
+ 'group pointer-events-auto relative flex w-full items-center justify-between space-x-4 overflow-hidden rounded-md border p-6 pr-8 shadow-lg transition-all data-[swipe=cancel]:translate-x-0 data-[swipe=end]:translate-x-[var(--radix-toast-swipe-end-x)] data-[swipe=move]:translate-x-[var(--radix-toast-swipe-move-x)] data-[swipe=move]:transition-none data-[state=open]:animate-in data-[state=closed]:animate-out data-[swipe=end]:animate-out data-[state=closed]:fade-out-80 data-[state=closed]:slide-out-to-right-full data-[state=open]:slide-in-from-top-full data-[state=open]:sm:slide-in-from-bottom-full',
+ {
+ variants: {
+ variant: {
+ default: 'border bg-background text-foreground',
+ destructive:
+ 'destructive group border-destructive bg-destructive text-destructive-foreground',
+ },
+ },
+ defaultVariants: {
+ variant: 'default',
+ },
+ }
+)
+
+const Toast = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef &
+ VariantProps
+>(({ className, variant, ...props }, ref) => {
+ return (
+
+ )
+})
+Toast.displayName = ToastPrimitives.Root.displayName
+
+const ToastAction = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+))
+ToastAction.displayName = ToastPrimitives.Action.displayName
+
+const ToastClose = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+
+
+))
+ToastClose.displayName = ToastPrimitives.Close.displayName
+
+const ToastTitle = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+))
+ToastTitle.displayName = ToastPrimitives.Title.displayName
+
+const ToastDescription = forwardRef<
+ ElementRef,
+ ComponentPropsWithoutRef
+>(({ className, ...props }, ref) => (
+
+))
+ToastDescription.displayName = ToastPrimitives.Description.displayName
+
+type ToastProps = ComponentPropsWithoutRef
+
+type ToastActionElement = ReactElement
+
+export {
+ type ToastProps,
+ type ToastActionElement,
+ ToastProvider,
+ ToastViewport,
+ Toast,
+ ToastTitle,
+ ToastDescription,
+ ToastClose,
+ ToastAction,
+}
diff --git a/src/components/ui/toaster.tsx b/src/components/ui/toaster.tsx
new file mode 100644
index 00000000..d07af34d
--- /dev/null
+++ b/src/components/ui/toaster.tsx
@@ -0,0 +1,36 @@
+import { useToast } from '../../hooks/use-toast'
+import {
+ Toast,
+ ToastClose,
+ ToastDescription,
+ ToastProvider,
+ ToastTitle,
+ ToastViewport,
+} from '../../components/ui/toast'
+
+export function Toaster() {
+ const { toasts } = useToast()
+
+ return (
+
+ {toasts.map(function ({ id, title, description, action, ...props }) {
+ return (
+
+
+ {title && {title}}
+ {description && (
+ {description}
+ )}
+
+ {action}
+
+
+ )
+ })}
+
+
+ )
+}
diff --git a/src/config/constants/main-process.ts b/src/config/constants/main-process.ts
index 83fb8ab4..a4e2b925 100644
--- a/src/config/constants/main-process.ts
+++ b/src/config/constants/main-process.ts
@@ -230,6 +230,21 @@ export enum ElectronAPIEventKeys {
VBucksInformationRequest = 'vbucks-information:request:data',
VBucksInformationResponseData = 'vbucks-information:response:data',
+ /**
+ * Friends Management
+ */
+
+ GetFriendsSummary = 'friends:summary:request',
+ GetFriendsSummaryResponse = 'friends:summary:response',
+ BlockFriends = 'friends:block',
+ BlockFriendsResponse = 'friends:block:response',
+ UnblockFriends = 'friends:unblock',
+ UnblockFriendsResponse = 'friends:unblock:response',
+ AddFriends = 'friends:add',
+ AddFriendsResponse = 'friends:add:response',
+ RemoveFriends = 'friends:remove',
+ RemoveFriendsResponse = 'friends:remove:response',
+
/**
* Redeem Codes
*/
diff --git a/src/hooks/management/friends.ts b/src/hooks/management/friends.ts
new file mode 100644
index 00000000..baf4eabd
--- /dev/null
+++ b/src/hooks/management/friends.ts
@@ -0,0 +1,71 @@
+import { useShallow } from 'zustand/react/shallow'
+
+import {
+ defaultFriendsSummary,
+ useFriendsStore,
+} from '../../state/management/friends'
+
+import { useGetAccounts } from '../accounts'
+
+export function useFriendsManagement() {
+ const { accounts, selected, changeSelection } = useFriendsStore(
+ useShallow((state) => ({
+ accounts: state.accounts,
+ selected: state.selected,
+
+ changeSelection: state.changeSelection,
+ }))
+ )
+
+ return {
+ accounts,
+ defaultFriendsSummary,
+ selected,
+
+ changeSelection,
+ }
+}
+
+export function useFriendsManagementActions() {
+ const { accountList } = useGetAccounts()
+ const {
+ isLoading,
+ selected,
+ getSummary,
+ removeFriends,
+ syncBlocklist,
+ syncIncoming,
+ syncOutgoing,
+ syncSummary,
+ updateLoading,
+ } = useFriendsStore(
+ useShallow((state) => ({
+ isLoading: state.isLoading,
+ selected: state.selected,
+
+ getSummary: state.getSummary,
+ removeFriends: state.removeFriends,
+ syncBlocklist: state.syncBlocklist,
+ syncIncoming: state.syncIncoming,
+ syncOutgoing: state.syncOutgoing,
+ syncSummary: state.syncSummary,
+ updateLoading: state.updateLoading,
+ }))
+ )
+
+ const account = selected ? accountList[selected] ?? null : null
+
+ return {
+ account,
+ isLoading,
+ selected,
+
+ getSummary,
+ removeFriends,
+ syncBlocklist,
+ syncIncoming,
+ syncOutgoing,
+ syncSummary,
+ updateLoading,
+ }
+}
diff --git a/src/hooks/use-toast.ts b/src/hooks/use-toast.ts
new file mode 100644
index 00000000..bcb65d2f
--- /dev/null
+++ b/src/hooks/use-toast.ts
@@ -0,0 +1,196 @@
+'use client'
+
+// Inspired by react-hot-toast library
+import type { ReactNode } from 'react'
+import type {
+ ToastActionElement,
+ ToastProps,
+} from '../components/ui/toast'
+
+import { useEffect, useState } from 'react'
+
+const TOAST_LIMIT = 1
+const TOAST_REMOVE_DELAY = 1000000
+
+type ToasterToast = ToastProps & {
+ id: string
+ title?: ReactNode
+ description?: ReactNode
+ action?: ToastActionElement
+}
+
+const actionTypes = {
+ ADD_TOAST: 'ADD_TOAST',
+ UPDATE_TOAST: 'UPDATE_TOAST',
+ DISMISS_TOAST: 'DISMISS_TOAST',
+ REMOVE_TOAST: 'REMOVE_TOAST',
+} as const
+
+let count = 0
+
+function genId() {
+ count = (count + 1) % Number.MAX_SAFE_INTEGER
+ return count.toString()
+}
+
+type ActionType = typeof actionTypes
+
+type Action =
+ | {
+ type: ActionType['ADD_TOAST']
+ toast: ToasterToast
+ }
+ | {
+ type: ActionType['UPDATE_TOAST']
+ toast: Partial
+ }
+ | {
+ type: ActionType['DISMISS_TOAST']
+ toastId?: ToasterToast['id']
+ }
+ | {
+ type: ActionType['REMOVE_TOAST']
+ toastId?: ToasterToast['id']
+ }
+
+interface State {
+ toasts: ToasterToast[]
+}
+
+const toastTimeouts = new Map>()
+
+const addToRemoveQueue = (toastId: string) => {
+ if (toastTimeouts.has(toastId)) {
+ return
+ }
+
+ const timeout = setTimeout(() => {
+ toastTimeouts.delete(toastId)
+ dispatch({
+ type: 'REMOVE_TOAST',
+ toastId: toastId,
+ })
+ }, TOAST_REMOVE_DELAY)
+
+ toastTimeouts.set(toastId, timeout)
+}
+
+export const reducer = (state: State, action: Action): State => {
+ switch (action.type) {
+ case 'ADD_TOAST':
+ return {
+ ...state,
+ toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
+ }
+
+ case 'UPDATE_TOAST':
+ return {
+ ...state,
+ toasts: state.toasts.map((t) =>
+ t.id === action.toast.id ? { ...t, ...action.toast } : t
+ ),
+ }
+
+ case 'DISMISS_TOAST': {
+ const { toastId } = action
+
+ // ! Side effects ! - This could be extracted into a dismissToast() action,
+ // but I'll keep it here for simplicity
+ if (toastId) {
+ addToRemoveQueue(toastId)
+ } else {
+ state.toasts.forEach((toast) => {
+ addToRemoveQueue(toast.id)
+ })
+ }
+
+ return {
+ ...state,
+ toasts: state.toasts.map((t) =>
+ t.id === toastId || toastId === undefined
+ ? {
+ ...t,
+ open: false,
+ }
+ : t
+ ),
+ }
+ }
+ case 'REMOVE_TOAST':
+ if (action.toastId === undefined) {
+ return {
+ ...state,
+ toasts: [],
+ }
+ }
+ return {
+ ...state,
+ toasts: state.toasts.filter((t) => t.id !== action.toastId),
+ }
+ }
+}
+
+const listeners: Array<(state: State) => void> = []
+
+let memoryState: State = { toasts: [] }
+
+function dispatch(action: Action) {
+ memoryState = reducer(memoryState, action)
+ listeners.forEach((listener) => {
+ listener(memoryState)
+ })
+}
+
+type Toast = Omit
+
+function toast({ ...props }: Toast) {
+ const id = genId()
+
+ const update = (props: ToasterToast) =>
+ dispatch({
+ type: 'UPDATE_TOAST',
+ toast: { ...props, id },
+ })
+ const dismiss = () => dispatch({ type: 'DISMISS_TOAST', toastId: id })
+
+ dispatch({
+ type: 'ADD_TOAST',
+ toast: {
+ ...props,
+ id,
+ open: true,
+ onOpenChange: (open) => {
+ if (!open) dismiss()
+ },
+ },
+ })
+
+ return {
+ id: id,
+ dismiss,
+ update,
+ }
+}
+
+function useToast() {
+ const [state, setState] = useState(memoryState)
+
+ useEffect(() => {
+ listeners.push(setState)
+ return () => {
+ const index = listeners.indexOf(setState)
+ if (index > -1) {
+ listeners.splice(index, 1)
+ }
+ }
+ }, [state])
+
+ return {
+ ...state,
+ toast,
+ dismiss: (toastId?: string) =>
+ dispatch({ type: 'DISMISS_TOAST', toastId }),
+ }
+}
+
+export { useToast, toast }
diff --git a/src/kernel/core/friends.ts b/src/kernel/core/friends.ts
new file mode 100644
index 00000000..07be3db8
--- /dev/null
+++ b/src/kernel/core/friends.ts
@@ -0,0 +1,287 @@
+import type {
+ EpicFriend,
+ FriendBlock,
+ FriendIncoming,
+ FriendOutgoing,
+ FriendsSummary,
+} from '../../types/services/friends'
+import type { AccountData } from '../../types/accounts'
+
+import split from 'just-split'
+
+import { ElectronAPIEventKeys } from '../../config/constants/main-process'
+
+import { MainWindow } from '../startup/windows/main'
+import { Authentication } from './authentication'
+
+import { getFriendsSummary } from '../../services/endpoints/friends'
+import { queryAccountsByIds } from '../../services/endpoints/lookup'
+
+import { getRawDateWithTZ } from '../../lib/dates'
+import { localeCompareForSorting } from '../../lib/utils'
+
+export class FriendsManagement {
+ static async getSummary(account: AccountData) {
+ const notificationKey = ElectronAPIEventKeys.GetFriendsSummaryResponse
+
+ try {
+ const accessToken = await Authentication.verifyAccessToken(account)
+
+ if (!accessToken) {
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account.accountId,
+ null
+ )
+
+ return
+ }
+
+ // const response = preloadedResults
+ const response = await getFriendsSummary({
+ accessToken,
+ accountId: account.accountId,
+ })
+
+ const friends = toObject(response.data.friends, 'accountId')
+ const incoming = toObject(response.data.incoming, 'accountId')
+ const outgoing = toObject(response.data.outgoing, 'accountId')
+ const blocklist = toObject(response.data.blocklist, 'accountId')
+
+ await Promise.allSettled([
+ queryIds(
+ Object.keys(friends),
+ accessToken,
+ (accountId, displayName) => {
+ if (friends[accountId] !== undefined) {
+ friends[accountId].displayName = displayName
+ }
+ }
+ ),
+ queryIds(
+ Object.keys(incoming),
+ accessToken,
+ (accountId, displayName) => {
+ if (incoming[accountId] !== undefined) {
+ incoming[accountId].displayName = displayName
+ }
+ }
+ ),
+ queryIds(
+ Object.keys(outgoing),
+ accessToken,
+ (accountId, displayName) => {
+ if (outgoing[accountId] !== undefined) {
+ outgoing[accountId].displayName = displayName
+ }
+ }
+ ),
+ queryIds(
+ Object.keys(blocklist),
+ accessToken,
+ (accountId, displayName) => {
+ if (blocklist[accountId] !== undefined) {
+ blocklist[accountId].displayName = displayName
+ }
+ }
+ ),
+ ])
+
+ const results: FriendsSummary = {
+ blocklist: sortValues(Object.values(blocklist)),
+ friends: sortValues(Object.values(friends)),
+ incoming: sortValues(Object.values(incoming)),
+ outgoing: sortValues(Object.values(outgoing)),
+ suggested: [],
+ limitsReached: response.data.limitsReached ?? null,
+ settings: response.data.settings ?? null,
+ }
+
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account.accountId,
+ results
+ )
+
+ return
+
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ } catch (error) {
+ //
+ }
+
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account.accountId,
+ null
+ )
+ }
+
+ static async blockFriends(
+ account: AccountData,
+ blocklist: Array,
+ origin?: 'incoming' | 'outgoing'
+ ) {
+ const notificationKey = ElectronAPIEventKeys.BlockFriendsResponse
+
+ FriendsManagement.checkAccessToken(
+ account,
+ notificationKey,
+ async () => {
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account,
+ blocklist.map((friend) => {
+ return {
+ accountId: friend.accountId,
+ created: getRawDateWithTZ().toISOString(),
+ displayName: friend.displayName,
+ } as FriendBlock
+ }),
+ origin
+ )
+ }
+ )
+ }
+
+ static async unblock(
+ account: AccountData,
+ unblocklist: 'full' | Array
+ ) {
+ const notificationKey = ElectronAPIEventKeys.UnblockFriendsResponse
+
+ FriendsManagement.checkAccessToken(
+ account,
+ notificationKey,
+ async () => {
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account,
+ unblocklist
+ )
+ }
+ )
+ }
+
+ static async addFriends(
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming'
+ ) {
+ const notificationKey = ElectronAPIEventKeys.AddFriendsResponse
+
+ FriendsManagement.checkAccessToken(
+ account,
+ notificationKey,
+ async () => {
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account,
+ friends,
+ context
+ )
+ }
+ )
+ }
+
+ static async removeFriends(
+ account: AccountData,
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+ ) {
+ const notificationKey = ElectronAPIEventKeys.RemoveFriendsResponse
+
+ FriendsManagement.checkAccessToken(
+ account,
+ notificationKey,
+ async () => {
+ MainWindow.instance.webContents.send(
+ notificationKey,
+ account,
+ friends,
+ context
+ )
+ }
+ )
+ }
+
+ private static async checkAccessToken(
+ account: AccountData,
+ notificationKey: ElectronAPIEventKeys,
+ callback: (accessToken: string) => Promise
+ ) {
+ try {
+ const accessToken = await Authentication.verifyAccessToken(account)
+
+ if (!accessToken) {
+ MainWindow.instance.webContents.send(notificationKey, account, [])
+
+ return
+ }
+
+ await callback(accessToken)
+
+ return
+ } catch (error) {
+ //
+ }
+
+ MainWindow.instance.webContents.send(notificationKey, account, [])
+
+ return
+ }
+}
+
+const toObject = >(
+ data: Array,
+ key: keyof Data
+) =>
+ data.reduce(
+ (accumulator, current) => {
+ if (
+ typeof current[key] === 'string' &&
+ current[key].trim().length > 0
+ ) {
+ accumulator[current[key]] = {
+ ...current,
+ displayName: current[key],
+ }
+ }
+
+ return accumulator
+ },
+ {} as Record
+ )
+const queryIds = async (
+ ids: Array,
+ accessToken: string,
+ callback: (accoundId: string, displayName: string) => void
+) => {
+ const queryFriends = await Promise.allSettled(
+ split(ids, 100).map((ids) =>
+ queryAccountsByIds({
+ accessToken,
+ ids,
+ })
+ )
+ )
+
+ queryFriends.map((response) => {
+ if (response.status === 'fulfilled') {
+ response.value.data.forEach((item) => {
+ const displayName =
+ item.displayName ??
+ item.externalAuths?.xbl?.externalDisplayName ??
+ item.externalAuths?.psn?.externalDisplayName ??
+ item.id
+
+ callback?.(item.id, displayName)
+ })
+ }
+ })
+}
+// eslint-disable-next-line @typescript-eslint/no-explicit-any
+const sortValues = (data: Array>): Data =>
+ data.toSorted((itemA, itemB) =>
+ localeCompareForSorting(itemA.displayName!, itemB.displayName!)
+ ) as Data
diff --git a/src/kernel/main.ts b/src/kernel/main.ts
index 0287f20f..547f0b72 100644
--- a/src/kernel/main.ts
+++ b/src/kernel/main.ts
@@ -2,6 +2,12 @@ import type {
SaveWorldInfoData,
WorldInfoFileData,
} from '../types/data/advanced-mode/world-info'
+import type {
+ EpicFriend,
+ FriendBlock,
+ FriendIncoming,
+ FriendOutgoing,
+} from '../types/services/friends'
import type {
AccountBasicInfo,
AccountData,
@@ -36,8 +42,10 @@ import { AlertsDone } from './core/alerts'
// import { AntiCheatProvider } from './core/anti-cheat-provider'
import { Authentication } from './core/authentication'
import { ClaimRewards } from './core/claim-rewards'
+import { CustomProcess } from './core/custom-process'
import { DevicesAuthManager } from './core/devices-auth'
import { EULATracking } from './core/eula-tracking'
+import { FriendsManagement } from './core/friends'
import { FortniteLauncher } from './core/launcher'
import { MCPClientQuestLogin, MCPHomebaseName } from './core/mcp'
import { MatchmakingTrack } from './core/matchmaking-track'
@@ -75,7 +83,6 @@ import {
} from '../state/stw-operations/auto/llamas'
import { Language } from '../locales/resources'
-import { CustomProcess } from './core/custom-process'
dayjs.extend(localizedFormat)
dayjs.extend(relativeTime)
@@ -703,6 +710,64 @@ const gotTheLock = app.requestSingleInstanceLock()
}
)
+ /**
+ * Friends Management
+ */
+
+ ipcMain.on(
+ ElectronAPIEventKeys.GetFriendsSummary,
+ async (_, account: AccountData) => {
+ await FriendsManagement.getSummary(account)
+ }
+ )
+
+ ipcMain.on(
+ ElectronAPIEventKeys.BlockFriends,
+ async (
+ _,
+ account: AccountData,
+ friends: Array,
+ origin?: 'incoming' | 'outgoing'
+ ) => {
+ await FriendsManagement.blockFriends(account, friends, origin)
+ }
+ )
+
+ ipcMain.on(
+ ElectronAPIEventKeys.UnblockFriends,
+ async (
+ _,
+ account: AccountData,
+ unblocklist: 'full' | Array
+ ) => {
+ await FriendsManagement.unblock(account, unblocklist)
+ }
+ )
+
+ ipcMain.on(
+ ElectronAPIEventKeys.AddFriends,
+ async (
+ _,
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming'
+ ) => {
+ await FriendsManagement.addFriends(account, friends, context)
+ }
+ )
+
+ ipcMain.on(
+ ElectronAPIEventKeys.RemoveFriends,
+ async (
+ _,
+ account: AccountData,
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+ ) => {
+ await FriendsManagement.removeFriends(account, friends, context)
+ }
+ )
+
/**
* Redeem Codes
*/
diff --git a/src/kernel/preload-actions/friends.ts b/src/kernel/preload-actions/friends.ts
new file mode 100644
index 00000000..36a21899
--- /dev/null
+++ b/src/kernel/preload-actions/friends.ts
@@ -0,0 +1,209 @@
+import type { IpcRendererEvent } from 'electron'
+import type { AccountData } from '../../types/accounts'
+import type {
+ EpicAddFriend,
+ EpicFriend,
+ FriendBlock,
+ FriendIncoming,
+ FriendOutgoing,
+ FriendsSummary,
+} from '../../types/services/friends'
+
+import { ipcRenderer } from 'electron'
+
+import { ElectronAPIEventKeys } from '../../config/constants/main-process'
+
+export function requestFriendsSummary(account: AccountData) {
+ ipcRenderer.send(ElectronAPIEventKeys.GetFriendsSummary, account)
+}
+
+export function requestBlockFriends(
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming' | 'outgoing'
+) {
+ ipcRenderer.send(
+ ElectronAPIEventKeys.BlockFriends,
+ account,
+ friends,
+ context
+ )
+}
+
+export function requestUnblockFriends(
+ account: AccountData,
+ unblocklist: 'full' | Array
+) {
+ ipcRenderer.send(
+ ElectronAPIEventKeys.UnblockFriends,
+ account,
+ unblocklist
+ )
+}
+
+export function requestRemoveFriends(
+ account: AccountData,
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+) {
+ ipcRenderer.send(
+ ElectronAPIEventKeys.RemoveFriends,
+ account,
+ friends,
+ context
+ )
+}
+
+export function requestAddFriends(
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming'
+) {
+ ipcRenderer.send(
+ ElectronAPIEventKeys.AddFriends,
+ account,
+ friends,
+ context
+ )
+}
+
+export function notificationFriendsSummary(
+ callback: (
+ accountId: string,
+ value: FriendsSummary | null
+ ) => Promise
+) {
+ const customCallback = (
+ _: IpcRendererEvent,
+ accountId: string,
+ value: FriendsSummary | null
+ ) => {
+ callback(accountId, value).catch(() => {})
+ }
+ const rendererInstance = ipcRenderer.on(
+ ElectronAPIEventKeys.GetFriendsSummaryResponse,
+ customCallback
+ )
+
+ return {
+ removeListener: () =>
+ rendererInstance.removeListener(
+ ElectronAPIEventKeys.GetFriendsSummaryResponse,
+ customCallback
+ ),
+ }
+}
+
+export function notificationBlockFriends(
+ callback: (
+ account: AccountData,
+ blocklist: Array,
+ context?: 'incoming' | 'outgoing'
+ ) => Promise
+) {
+ const customCallback = (
+ _: IpcRendererEvent,
+ account: AccountData,
+ blocklist: Array,
+ context?: 'incoming' | 'outgoing'
+ ) => {
+ callback(account, blocklist, context).catch(() => {})
+ }
+ const rendererInstance = ipcRenderer.on(
+ ElectronAPIEventKeys.BlockFriendsResponse,
+ customCallback
+ )
+
+ return {
+ removeListener: () =>
+ rendererInstance.removeListener(
+ ElectronAPIEventKeys.BlockFriendsResponse,
+ customCallback
+ ),
+ }
+}
+
+export function notificationUnblockFriends(
+ callback: (
+ account: AccountData,
+ unblocklist: 'full' | Array
+ ) => Promise
+) {
+ const customCallback = (
+ _: IpcRendererEvent,
+ account: AccountData,
+ unblocklist: 'full' | Array
+ ) => {
+ callback(account, unblocklist).catch(() => {})
+ }
+ const rendererInstance = ipcRenderer.on(
+ ElectronAPIEventKeys.UnblockFriendsResponse,
+ customCallback
+ )
+
+ return {
+ removeListener: () =>
+ rendererInstance.removeListener(
+ ElectronAPIEventKeys.UnblockFriendsResponse,
+ customCallback
+ ),
+ }
+}
+
+export function notificationAddFriends(
+ callback: (
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming'
+ ) => Promise
+) {
+ const customCallback = (
+ _: IpcRendererEvent,
+ account: AccountData,
+ friends: Array,
+ context?: 'incoming'
+ ) => {
+ callback(account, friends, context).catch(() => {})
+ }
+ const rendererInstance = ipcRenderer.on(
+ ElectronAPIEventKeys.AddFriendsResponse,
+ customCallback
+ )
+
+ return {
+ removeListener: () =>
+ rendererInstance.removeListener(
+ ElectronAPIEventKeys.AddFriendsResponse,
+ customCallback
+ ),
+ }
+}
+
+export function notificationRemoveFriends(
+ callback: (
+ account: AccountData,
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+ ) => Promise
+) {
+ const customCallback = (
+ _: IpcRendererEvent,
+ account: AccountData,
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+ ) => {
+ callback(account, friends, context).catch(() => {})
+ }
+ const rendererInstance = ipcRenderer.on(
+ ElectronAPIEventKeys.RemoveFriendsResponse,
+ customCallback
+ )
+
+ return {
+ removeListener: () =>
+ rendererInstance.removeListener(
+ ElectronAPIEventKeys.RemoveFriendsResponse,
+ customCallback
+ ),
+ }
+}
diff --git a/src/kernel/preload.ts b/src/kernel/preload.ts
index 6dfa8867..73b2d0b0 100644
--- a/src/kernel/preload.ts
+++ b/src/kernel/preload.ts
@@ -12,6 +12,7 @@ import * as automationsActions from './preload-actions/automation'
import * as customizableMenuActions from './preload-actions/customizable-menu'
import * as devicesAuthActions from './preload-actions/devices-auth'
import * as eventActions from './preload-actions/events'
+import * as friendsActions from './preload-actions/friends'
import * as generalActions from './preload-actions/general'
import * as launcherActions from './preload-actions/launcher'
import * as matchmakingActions from './preload-actions/matchmaking'
@@ -35,6 +36,7 @@ export const availableElectronAPIs = {
...autoPinUrnsActions,
...devicesAuthActions,
...eventActions,
+ ...friendsActions,
...generalActions,
...launcherActions,
...partyActions,
diff --git a/src/kernel/startup/auto-llamas.ts b/src/kernel/startup/auto-llamas.ts
index 4145c5a4..53873bdc 100644
--- a/src/kernel/startup/auto-llamas.ts
+++ b/src/kernel/startup/auto-llamas.ts
@@ -3,7 +3,11 @@ import type {
AutoLlamasData,
AutoLlamasRecord,
} from '../../types/auto-llamas'
-import type { MCPQueryProfileProfileChangesPrerollData } from '../../types/services/mcp'
+import type {
+ MCPQueryProfileChanges,
+ MCPQueryProfileProfileChangesPrerollData,
+} from '../../types/services/mcp'
+import type { StorefrontCatalogResponse } from '../../types/services/storefront'
import type { RewardsNotification } from '../../types/notifications'
import { Collection } from '@discordjs/collection'
@@ -292,53 +296,33 @@ export class ProcessAutoLlamas {
}
}
- // eslint-disable-next-line no-constant-condition
- while (true) {
- let success = true
+ const accountId = account.accountId
- try {
- const accessToken =
- await Authentication.verifyAccessToken(account)
+ if (type === ProcessLlamaType.Survivor) {
+ // eslint-disable-next-line no-constant-condition
+ while (true) {
+ try {
+ const accessToken =
+ await Authentication.verifyAccessToken(account)
- if (!accessToken) {
- break
- }
+ if (!accessToken) {
+ break
+ }
- try {
- await populatePrerolledOffers({
+ await ProcessAutoLlamas.populateOffers({
accessToken,
- accountId: account.accountId,
+ accountId,
})
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- } catch (error) {
- //
- }
-
- const queryProfile = await getQueryProfile({
- accessToken,
- accountId: account.accountId,
- })
- const profileChanges =
- queryProfile.data.profileChanges[0] ?? null
-
- const xRayTickets =
- Object.values(profileChanges?.profile.items).find(
- (item) =>
- (item.templateId as string) ===
- 'AccountResource:currency_xrayllama'
- )?.quantity ?? 0
- const llamaTokens =
- Object.values(profileChanges?.profile.items).find(
- (item) =>
- (item.templateId as string) ===
- 'AccountResource:voucher_cardpack_bronze'
- )?.quantity ?? 0
+ const { llamaTokens, profile, xRayTickets } =
+ await ProcessAutoLlamas.getProfileData({
+ accessToken,
+ accountId,
+ })
- let currencyTotal: number | null = null
- let currencySubType = 'AccountResource:currency_xrayllama'
+ let currencyTotal: number | null = null
+ let currencySubType = 'AccountResource:currency_xrayllama'
- if (type === ProcessLlamaType.Survivor) {
if (current.actions.survivors) {
if (current.actions['use-token']) {
if (llamaTokens > 0) {
@@ -354,215 +338,404 @@ export class ProcessAutoLlamas {
}
}
}
- } else if (type === ProcessLlamaType.FreeUpgrade) {
- if (current.actions['free-llamas']) {
- currencyTotal = 0
+
+ const currencyIsToken =
+ currencySubType ===
+ 'AccountResource:voucher_cardpack_bronze'
+
+ if (currencyTotal === null) {
+ break
}
- }
- const currencyIsToken =
- currencySubType ===
- 'AccountResource:voucher_cardpack_bronze'
+ const cardPacks =
+ await ProcessAutoLlamas.getCurrentCatalog({
+ accessToken,
+ })
- if (currencyTotal === null) {
- break
- }
+ if (!cardPacks) {
+ break
+ }
- const catalog = await getCatalog({
- accessToken,
- })
- const cardPacks = catalog.data.storefronts.find(
- (item) => item.name === 'CardPackStorePreroll'
- )
+ const llama = cardPacks.catalogEntries.find((item) => {
+ return currencyIsToken
+ ? item.devName === 'Always.UpgradePack.02'
+ : item.devName === 'Always.UpgradePack.01'
+ })
- if (!cardPacks) {
- break
- }
+ if (!llama) {
+ break
+ }
- const llama = cardPacks.catalogEntries.find((item) => {
- if (type === ProcessLlamaType.Survivor) {
- if (currencyIsToken) {
- return item.devName === 'Always.UpgradePack.02'
- }
+ const { maxPurchases } =
+ await ProcessAutoLlamas.checkDailyLimit({
+ accessToken,
+ accountId,
+ currencyIsToken,
+ llama,
+ })
- return (
- item.devName === 'Always.UpgradePack.01' &&
- item.dailyLimit === 50 &&
- item.prices[0]?.regularPrice === 50 &&
- item.prices[0]?.finalPrice === 50
- )
+ if (maxPurchases) {
+ break
}
- return (
- (item.devName?.toLowerCase().includes('free') ||
- item.title?.toLowerCase().includes('free')) &&
- item.prices[0]?.regularPrice === 50 &&
- item.prices[0]?.finalPrice === 0
- )
- })
+ const prerollData = Object.values(
+ profile.items ?? {}
+ ).find(
+ (item) =>
+ isMCPQueryProfileChangesPrerollData(item) &&
+ item.attributes.offerId === llama.offerId
+ ) as MCPQueryProfileProfileChangesPrerollData | undefined
- if (!llama) {
- break
- }
+ if (!prerollData) {
+ break
+ }
- const mainProfile = await getQueryProfileMainProfile({
- accessToken,
- accountId: account.accountId,
- })
- const totalPurchases =
- mainProfile.data.profileChanges[0]?.profile.stats
- .attributes.daily_purchases.purchaseList[
- llama.offerId
- ] ?? 0
- const dailyLimit =
- currencyIsToken && llama.dailyLimit <= -1
- ? 10
- : llama.dailyLimit ?? 2
-
- if (totalPurchases >= dailyLimit) {
- break
- }
+ const canPurchase = prerollData.attributes.items.some(
+ ({ itemType }) => {
+ const newKey = itemType
+ .replace(
+ /_((very)?low|medium|(very)?high|extreme)$/gi,
+ ''
+ )
+ .replace('AccountResource:', '')
+ .replace('CardPack:zcp_', '')
+ const survivor = getKey(newKey, survivorsJson)
+ const mythicSurvivor = getKey(
+ newKey,
+ survivorsMythicLeadsJson
+ )
+ const isWorker = newKey.startsWith('Worker:')
+ const rarity = parseRarity(newKey)
+
+ return (
+ survivor ||
+ mythicSurvivor ||
+ (isWorker &&
+ [RarityType.Legendary, RarityType.Mythic].includes(
+ rarity.rarity
+ ))
+ )
+ }
+ )
- const prerollData = Object.values(
- profileChanges.profile.items ?? {}
- ).find(
- (item) =>
- isMCPQueryProfileChangesPrerollData(item) &&
- item.attributes.offerId === llama.offerId
- ) as MCPQueryProfileProfileChangesPrerollData | undefined
+ if (!canPurchase) {
+ break
+ }
- if (!prerollData) {
+ await ProcessAutoLlamas.buyAndNotify({
+ accessToken,
+ currencySubType,
+ profile,
+ accountId: account.accountId,
+ expectedTotalPrice: currencyTotal,
+ offerId: llama.offerId,
+ })
+
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ } catch (error) {
break
}
+ }
- const canPurchase =
- type === ProcessLlamaType.FreeUpgrade
- ? true
- : prerollData.attributes.items.some(({ itemType }) => {
- const newKey = itemType
- .replace(
- /_((very)?low|medium|(very)?high|extreme)$/gi,
- ''
- )
- .replace('AccountResource:', '')
- .replace('CardPack:zcp_', '')
- const survivor = getKey(newKey, survivorsJson)
- const mythicSurvivor = getKey(
- newKey,
- survivorsMythicLeadsJson
- )
- const isWorker = newKey.startsWith('Worker:')
- const rarity = parseRarity(newKey)
-
- return (
- survivor ||
- mythicSurvivor ||
- (isWorker &&
- [
- RarityType.Legendary,
- RarityType.Mythic,
- ].includes(rarity.rarity))
- )
- })
+ return
+ }
- if (!canPurchase) {
- break
- }
+ try {
+ const accessToken =
+ await Authentication.verifyAccessToken(account)
- const response = await purchaseCatalogEntry({
- accessToken,
- currencySubType,
- accountId: account.accountId,
- offerId: llama.offerId,
- expectedTotalPrice: currencyTotal,
- })
- const responseNotifications =
- response.data.notifications ?? []
-
- const notifications: Array<{
- itemType: string
- quantity: number
- }> = []
-
- responseNotifications?.forEach((notification) => {
- if (notification.loot) {
- if (notification.loot.items) {
- notification.loot.items.forEach((loot) => {
- notifications.push({
- itemType: loot.itemType,
- quantity: loot.quantity,
- })
- })
- } else if (notification.loot.lootGranted) {
- notification.loot.lootGranted.items.forEach((loot) => {
- notifications.push({
- itemType: loot.itemType,
- quantity: loot.quantity,
- })
- })
- }
- } else if (notification.lootGranted) {
- notification.lootGranted?.items.forEach((loot) => {
- notifications.push({
- itemType: loot.itemType,
- quantity: loot.quantity,
- })
- })
- } else if (notification.lootResult) {
- notification.lootResult?.items.forEach((loot) => {
- notifications.push({
- itemType: loot.itemType,
- quantity: loot.quantity,
- })
- })
- }
- })
+ if (!accessToken) {
+ return
+ }
- const rewards: RewardsNotification['rewards'] = {}
+ await ProcessAutoLlamas.populateOffers({
+ accessToken,
+ accountId,
+ })
- notifications.forEach(({ itemType, quantity }) => {
- if (!itemType.toLowerCase().startsWith('accolades:')) {
- const newItemType =
- itemType === 'AccountResource:campaign_event_currency'
- ? profileChanges.profile.stats.attributes
- .event_currency?.templateId ?? itemType
- : itemType
+ const { profile } = await ProcessAutoLlamas.getProfileData({
+ accessToken,
+ accountId,
+ })
- if (!rewards[newItemType]) {
- rewards[newItemType] = 0
- }
+ const cardPacks = await ProcessAutoLlamas.getCurrentCatalog({
+ accessToken,
+ })
- rewards[newItemType] += quantity
- }
- })
+ if (!cardPacks) {
+ return
+ }
- const result: RewardsNotification = {
- accolades: {
- totalMissionXPRedeemed: 0,
- totalQuestXPRedeemed: 0,
- },
- rewards,
- createdAt: getDateWithDefaultFormat(),
- id: crypto.randomUUID(),
- accountId: account.accountId,
- }
+ const llamas = cardPacks.catalogEntries.filter((item) => {
+ return item.prices[0]?.finalPrice === 0
+ })
- MainWindow.instance.webContents.send(
- ElectronAPIEventKeys.ClaimRewardsClientGlobalSyncNotification,
- [result]
- )
+ if (llamas.length <= 0) {
+ return
+ }
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- } catch (error) {
- success = false
+ const availableLlamas: StorefrontCatalogResponse['storefronts'][number]['catalogEntries'] =
+ []
+
+ for (const llama of llamas) {
+ // const { dailyLimit, maxPurchases, totalPurchases } =
+ // await ProcessAutoLlamas.checkDailyLimit({
+ // accessToken,
+ // accountId,
+ // llama,
+ // currencyIsToken: false,
+ // })
+
+ // if (maxPurchases) {
+ // continue
+ // }
+
+ // Array.from({ length: dailyLimit - totalPurchases }).forEach(
+ Array.from({
+ length: llama.dailyLimit <= -1 ? 30 : llama.dailyLimit,
+ }).forEach(() => {
+ availableLlamas.push(llama)
+ })
}
- if (!success) {
- break
+ if (availableLlamas.length <= 0) {
+ return
}
+
+ availableLlamas.forEach((llama) => {
+ ProcessAutoLlamas.buyAndNotify({
+ accessToken,
+ profile,
+ accountId: account.accountId,
+ currencySubType: 'AccountResource:currency_xrayllama',
+ expectedTotalPrice: 0,
+ offerId: llama.offerId,
+ }).catch(() => {})
+ })
+
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ } catch (error) {
+ //
}
+
+ return
}
)
})
}
+
+ static async populateOffers({
+ accessToken,
+ accountId,
+ }: {
+ accessToken: string
+ accountId: string
+ }) {
+ try {
+ await populatePrerolledOffers({
+ accessToken,
+ accountId,
+ })
+
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ } catch (error) {
+ //
+ }
+ }
+
+ static async getProfileData({
+ accessToken,
+ accountId,
+ }: {
+ accessToken: string
+ accountId: string
+ }) {
+ const queryProfile = await getQueryProfile({
+ accessToken,
+ accountId,
+ })
+ const profileChanges = queryProfile.data.profileChanges[0] ?? null
+
+ const xRayTickets =
+ Object.values(profileChanges?.profile.items).find(
+ (item) =>
+ (item.templateId as string) ===
+ 'AccountResource:currency_xrayllama'
+ )?.quantity ?? 0
+ const llamaTokens =
+ Object.values(profileChanges?.profile.items).find(
+ (item) =>
+ (item.templateId as string) ===
+ 'AccountResource:voucher_cardpack_bronze'
+ )?.quantity ?? 0
+
+ return {
+ llamaTokens,
+ xRayTickets,
+ profile: queryProfile.data.profileChanges[0]?.profile,
+ }
+ }
+
+ static async getCurrentCatalog({
+ accessToken,
+ }: {
+ accessToken: string
+ }) {
+ const catalog = await getCatalog({
+ accessToken,
+ })
+ const cardPacks = catalog.data.storefronts.find(
+ (item) => item.name === 'CardPackStorePreroll'
+ )
+
+ return cardPacks
+ }
+
+ static async checkDailyLimit({
+ accessToken,
+ accountId,
+ currencyIsToken,
+ llama,
+ }: {
+ accessToken: string
+ accountId: string
+ currencyIsToken: boolean
+ llama: StorefrontCatalogResponse['storefronts'][number]['catalogEntries'][number]
+ }) {
+ try {
+ const mainProfile = await getQueryProfileMainProfile({
+ accessToken,
+ accountId,
+ })
+ const getDenyRequirement = llama.requirements?.find(
+ (item) => item.requirementType === 'DenyOnFulfillment'
+ )
+ const totalPurchases =
+ mainProfile.data.profileChanges[0]?.profile.stats.attributes
+ .daily_purchases?.purchaseList?.[llama.offerId] ??
+ mainProfile.data.profileChanges[0]?.profile.stats.attributes
+ .in_app_purchases?.fulfillmentCounts?.[
+ getDenyRequirement?.requiredId ?? -1
+ ] ??
+ 0
+ const dailyLimit =
+ currencyIsToken && llama.dailyLimit <= -1
+ ? 30
+ : llama.dailyLimit ?? 30
+
+ return {
+ maxPurchases: totalPurchases >= dailyLimit,
+ totalPurchases,
+ dailyLimit,
+ }
+
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ } catch (error) {
+ //
+ }
+
+ return {
+ maxPurchases: false,
+ totalPurchases: 0,
+ dailyLimit: 0,
+ }
+ }
+
+ static async buyAndNotify({
+ accessToken,
+ accountId,
+ currencySubType,
+ expectedTotalPrice,
+ offerId,
+ profile,
+ }: {
+ accessToken: string
+ accountId: string
+ currencySubType?: string
+ offerId: string
+ expectedTotalPrice: number
+ profile: MCPQueryProfileChanges['profile']
+ }) {
+ const response = await purchaseCatalogEntry({
+ accessToken,
+ accountId,
+ currencySubType,
+ offerId,
+ expectedTotalPrice,
+ })
+ const responseNotifications = response.data.notifications ?? []
+
+ const notifications: Array<{
+ itemType: string
+ quantity: number
+ }> = []
+
+ responseNotifications?.forEach((notification) => {
+ if (notification.loot) {
+ if (notification.loot.items) {
+ notification.loot.items.forEach((loot) => {
+ notifications.push({
+ itemType: loot.itemType,
+ quantity: loot.quantity,
+ })
+ })
+ } else if (notification.loot.lootGranted) {
+ notification.loot.lootGranted.items.forEach((loot) => {
+ notifications.push({
+ itemType: loot.itemType,
+ quantity: loot.quantity,
+ })
+ })
+ }
+ } else if (notification.lootGranted) {
+ notification.lootGranted?.items.forEach((loot) => {
+ notifications.push({
+ itemType: loot.itemType,
+ quantity: loot.quantity,
+ })
+ })
+ } else if (notification.lootResult) {
+ notification.lootResult?.items.forEach((loot) => {
+ notifications.push({
+ itemType: loot.itemType,
+ quantity: loot.quantity,
+ })
+ })
+ }
+ })
+
+ const rewards: RewardsNotification['rewards'] = {}
+
+ notifications.forEach(({ itemType, quantity }) => {
+ if (!itemType.toLowerCase().startsWith('accolades:')) {
+ const newItemType =
+ itemType === 'AccountResource:campaign_event_currency'
+ ? profile.stats.attributes.event_currency?.templateId ??
+ itemType
+ : itemType
+
+ if (!rewards[newItemType]) {
+ rewards[newItemType] = 0
+ }
+
+ rewards[newItemType] += quantity
+ }
+ })
+
+ const result: RewardsNotification = {
+ accountId,
+ rewards,
+ accolades: {
+ totalMissionXPRedeemed: 0,
+ totalQuestXPRedeemed: 0,
+ },
+ createdAt: getDateWithDefaultFormat(),
+ id: crypto.randomUUID(),
+ }
+
+ MainWindow.instance.webContents.send(
+ ElectronAPIEventKeys.ClaimRewardsClientGlobalSyncNotification,
+ [result]
+ )
+ }
}
diff --git a/src/lib/validations/schemas/settings.ts b/src/lib/validations/schemas/settings.ts
index 6132e323..8ac4b261 100644
--- a/src/lib/validations/schemas/settings.ts
+++ b/src/lib/validations/schemas/settings.ts
@@ -60,6 +60,7 @@ export const customizableMenuSettingsSchema = z
accountManagement: z.boolean().default(true),
vbucksInformation: z.boolean().default(true),
+ friendsManagement: z.boolean().default(true),
redeemCodes: z.boolean().default(true),
devicesAuth: z.boolean().default(true),
epicGamesSettings: z.boolean().default(true),
diff --git a/src/lib/validations/schemas/storefront.ts b/src/lib/validations/schemas/storefront.ts
index da6d400b..d321a369 100644
--- a/src/lib/validations/schemas/storefront.ts
+++ b/src/lib/validations/schemas/storefront.ts
@@ -48,7 +48,18 @@ export const storefrontCatalogSchema = z.object({
monthlyLimit: z.number(),
refundable: z.boolean(),
// appStoreId: z.array(z.string()),
- // requirements: z.array(z.unknown()),
+ requirements: z
+ .array(
+ z.object({
+ requirementType: z.enum([
+ 'DenyOnFulfillment',
+ 'RequireFulfillment',
+ ]),
+ requiredId: z.string(),
+ minQuantity: z.number(),
+ })
+ )
+ .optional(),
meta: z
.object({
PurchaseLimitingEventId: z.string(),
diff --git a/src/locales/en-US/account-management/friends-management.json b/src/locales/en-US/account-management/friends-management.json
new file mode 100644
index 00000000..addc7666
--- /dev/null
+++ b/src/locales/en-US/account-management/friends-management.json
@@ -0,0 +1,3 @@
+{
+ "coming-soon": "Coming Soon!"
+}
diff --git a/src/locales/en-US/sidebar.json b/src/locales/en-US/sidebar.json
index 8e2e425d..91754bd3 100644
--- a/src/locales/en-US/sidebar.json
+++ b/src/locales/en-US/sidebar.json
@@ -20,6 +20,7 @@
"title": "Account Management",
"options": {
"vbucks-information": "V-Bucks Information",
+ "friends-management": "Friends Management",
"redeem-codes": "Redeem Codes",
"devices-auth": "Devices Auth",
"epic-settings": "Epic Games Settings"
diff --git a/src/locales/resources.ts b/src/locales/resources.ts
index 250e2fa7..92a87438 100644
--- a/src/locales/resources.ts
+++ b/src/locales/resources.ts
@@ -18,6 +18,7 @@ import enUS_stwOperations_XPBoosts from './en-US/stw-operations/xpboosts.json'
import enUS_stwOperations_Llamas from './en-US/stw-operations/llamas.json'
import enUS_stwOperations_Unlock from './en-US/stw-operations/unlock.json'
import enUS_accountManagement_VBucksInformation from './en-US/account-management/vbucks-information.json'
+import enUS_accountManagement_FriendsManagement from './en-US/account-management/friends-management.json'
import enUS_accountManagement_EULA from './en-US/account-management/eula.json'
import enUS_accountManagement_RedeemCodes from './en-US/account-management/redeem-codes.json'
import enUS_accountManagement_DevicesAuth from './en-US/account-management/devices-auth.json'
@@ -49,6 +50,7 @@ const enUS = {
},
'account-management': {
'vbucks-information': enUS_accountManagement_VBucksInformation,
+ 'friends-management': enUS_accountManagement_FriendsManagement,
'redeem-codes': enUS_accountManagement_RedeemCodes,
'devices-auth': enUS_accountManagement_DevicesAuth,
'epic-settings': enUS_accountManagement_EpicSettings,
diff --git a/src/routeTree.gen.ts b/src/routeTree.gen.ts
index aad52b32..9bd49fd3 100644
--- a/src/routeTree.gen.ts
+++ b/src/routeTree.gen.ts
@@ -27,6 +27,7 @@ import { Route as AdvancedModeMatchmakingTrackRouteImport } from './routes/advan
import { Route as AccountsRemoveRouteImport } from './routes/accounts/remove/route'
import { Route as AccountManagementVbucksInformationRouteImport } from './routes/account-management/vbucks-information/route'
import { Route as AccountManagementRedeemCodesRouteImport } from './routes/account-management/redeem-codes/route'
+import { Route as AccountManagementFriendsManagementRouteImport } from './routes/account-management/friends-management/route'
import { Route as AccountManagementEulaRouteImport } from './routes/account-management/eula/route'
import { Route as AccountManagementEpicGamesSettingsRouteImport } from './routes/account-management/epic-games-settings/route'
import { Route as AccountManagementDevicesAuthRouteImport } from './routes/account-management/devices-auth/route'
@@ -125,6 +126,12 @@ const AccountManagementRedeemCodesRouteRoute =
getParentRoute: () => rootRoute,
} as any)
+const AccountManagementFriendsManagementRouteRoute =
+ AccountManagementFriendsManagementRouteImport.update({
+ path: '/account-management/friends-management',
+ getParentRoute: () => rootRoute,
+ } as any)
+
const AccountManagementEulaRouteRoute = AccountManagementEulaRouteImport.update(
{
path: '/account-management/eula',
@@ -173,6 +180,10 @@ declare module '@tanstack/react-router' {
preLoaderRoute: typeof AccountManagementEulaRouteImport
parentRoute: typeof rootRoute
}
+ '/account-management/friends-management': {
+ preLoaderRoute: typeof AccountManagementFriendsManagementRouteImport
+ parentRoute: typeof rootRoute
+ }
'/account-management/redeem-codes': {
preLoaderRoute: typeof AccountManagementRedeemCodesRouteImport
parentRoute: typeof rootRoute
@@ -244,6 +255,7 @@ export const routeTree = rootRoute.addChildren([
AccountManagementDevicesAuthRouteRoute,
AccountManagementEpicGamesSettingsRouteRoute,
AccountManagementEulaRouteRoute,
+ AccountManagementFriendsManagementRouteRoute,
AccountManagementRedeemCodesRouteRoute,
AccountManagementVbucksInformationRouteRoute,
AccountsRemoveRouteRoute,
diff --git a/src/routes/account-management/friends-management/-columns.tsx b/src/routes/account-management/friends-management/-columns.tsx
new file mode 100644
index 00000000..d1fdc1e1
--- /dev/null
+++ b/src/routes/account-management/friends-management/-columns.tsx
@@ -0,0 +1,504 @@
+import type { ColumnDef } from '@tanstack/react-table'
+import type { FriendsSummary } from '../../../types/services/friends'
+
+import {
+ BanIcon,
+ CheckIcon,
+ LockKeyholeIcon,
+ RotateCcwIcon,
+ Trash2Icon,
+} from 'lucide-react'
+
+import { Button } from '../../../components/ui/button'
+import { Checkbox } from '../../../components/ui/checkbox'
+import {
+ Dialog,
+ DialogClose,
+ DialogContent,
+ DialogFooter,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from '../../../components/ui/dialog'
+
+import { useFriendsTableActions } from './-hooks'
+
+export function useFriendsColumns() {
+ const { handleBlock, handleRemove } = useFriendsTableActions()
+
+ const columns: Array> = [
+ {
+ id: 'select',
+ header: ({ table }) => (
+
+
+ table.toggleAllPageRowsSelected(!!value)
+ }
+ aria-label="select all"
+ />
+
+ ),
+ cell: ({ row }) => (
+
+ row.toggleSelected(!!value)}
+ aria-label="select row"
+ />
+
+ ),
+ },
+ {
+ accessorKey: 'accountId',
+ accessorFn: (row) =>
+ `${row.accountId}${row.displayName ?? ''}${row.alias}`,
+ header: () => {
+ return Name
+ },
+ cell: ({ row }) => {
+ return (
+
+
+
+ )
+ },
+ },
+ {
+ id: 'actions',
+ header: 'Actions',
+ cell: ({ row }) => (
+
+ {/*
*/}
+
+
+
+
+
+
+
+
+
+ ),
+ },
+ ]
+
+ return {
+ columns,
+ }
+}
+
+export function useIncomingColumns() {
+ const { handleAdd, handleBlock, handleRemove } = useFriendsTableActions()
+
+ const columns: Array> = [
+ {
+ id: 'select',
+ header: ({ table }) => (
+
+
+ table.toggleAllPageRowsSelected(!!value)
+ }
+ aria-label="select all"
+ />
+
+ ),
+ cell: ({ row }) => (
+
+ row.toggleSelected(!!value)}
+ aria-label="select row"
+ />
+
+ ),
+ },
+ {
+ accessorKey: 'accountId',
+ accessorFn: (row) => `${row.accountId}${row.displayName ?? ''}`,
+ header: () => {
+ return Name
+ },
+ cell: ({ row }) => {
+ return (
+
+
+
+ )
+ },
+ },
+ {
+ id: 'actions',
+ header: 'Actions',
+ cell: ({ row }) => (
+
+ ),
+ },
+ ]
+
+ return {
+ columns,
+ }
+}
+
+export function useOutgoingColumns() {
+ const { handleBlock, handleRemove } = useFriendsTableActions()
+
+ const columns: Array> = [
+ {
+ id: 'select',
+ header: ({ table }) => (
+
+
+ table.toggleAllPageRowsSelected(!!value)
+ }
+ aria-label="select all"
+ />
+
+ ),
+ cell: ({ row }) => (
+
+ row.toggleSelected(!!value)}
+ aria-label="select row"
+ />
+
+ ),
+ },
+ {
+ accessorKey: 'accountId',
+ accessorFn: (row) => `${row.accountId}${row.displayName ?? ''}`,
+ header: () => {
+ return Name
+ },
+ cell: ({ row }) => {
+ return (
+
+
+
+ )
+ },
+ },
+ {
+ id: 'actions',
+ header: 'Actions',
+ cell: ({ row }) => (
+
+ ),
+ },
+ ]
+
+ return {
+ columns,
+ }
+}
+
+export function useBlocklistColumns() {
+ const { handleUnblock } = useFriendsTableActions()
+
+ const columns: Array> = [
+ {
+ id: 'select',
+ header: ({ table }) => (
+
+
+ table.toggleAllPageRowsSelected(!!value)
+ }
+ aria-label="select all"
+ />
+
+ ),
+ cell: ({ row }) => (
+
+ row.toggleSelected(!!value)}
+ aria-label="select row"
+ />
+
+ ),
+ },
+ {
+ accessorKey: 'accountId',
+ accessorFn: (row) => `${row.accountId}${row.displayName ?? ''}`,
+ header: () => {
+ return Name
+ },
+ cell: ({ row }) => {
+ return (
+
+
+
+ )
+ },
+ },
+ {
+ id: 'actions',
+ header: 'Actions',
+ cell: ({ row }) => (
+
+
+
+ ),
+ },
+ ]
+
+ return {
+ columns,
+ }
+}
+
+function ActionDelete({
+ callback,
+ displayName,
+}: {
+ callback: () => void
+ displayName?: string
+}) {
+ return (
+
+ )
+}
+
+function ActionBlock({
+ callback,
+ displayName,
+}: {
+ callback: () => void
+ displayName?: string
+}) {
+ return (
+
+ )
+}
diff --git a/src/routes/account-management/friends-management/-hooks.ts b/src/routes/account-management/friends-management/-hooks.ts
new file mode 100644
index 00000000..3aa072d7
--- /dev/null
+++ b/src/routes/account-management/friends-management/-hooks.ts
@@ -0,0 +1,182 @@
+import type {
+ ComboboxOption,
+ ComboboxProps,
+} from '../../../components/ui/extended/combobox/hooks'
+import type {
+ EpicAddFriend,
+ EpicFriend,
+ FriendBlock,
+ FriendIncoming,
+ FriendOutgoing,
+ FriendsSummary,
+} from '../../../types/services/friends'
+
+import {
+ useFriendsManagement,
+ useFriendsManagementActions,
+} from '../../../hooks/management/friends'
+import { useGetAccounts } from '../../../hooks/accounts'
+import { useGetGroups } from '../../../hooks/groups'
+
+import { checkIfCustomDisplayNameIsValid } from '../../../lib/validations/properties'
+import { parseCustomDisplayName } from '../../../lib/utils'
+
+export function useAccountSelector() {
+ const { accountsArray } = useGetAccounts()
+ const { getGroupTagsByAccountId } = useGetGroups()
+ const { accounts, defaultFriendsSummary, selected, changeSelection } =
+ useFriendsManagement()
+
+ const data: FriendsSummary = selected
+ ? accounts[selected] ?? defaultFriendsSummary
+ : defaultFriendsSummary
+
+ const options = accountsArray.map((account) => {
+ const _keys: Array = [account.displayName]
+ const tags = getGroupTagsByAccountId(account.accountId)
+
+ if (checkIfCustomDisplayNameIsValid(account.customDisplayName)) {
+ _keys.push(account.customDisplayName)
+ }
+
+ if (tags.length > 0) {
+ tags.forEach((tagName) => {
+ _keys.push(tagName)
+ })
+ }
+
+ return {
+ keywords: _keys,
+ label: parseCustomDisplayName(account),
+ value: account.accountId,
+ } as ComboboxOption
+ })
+ const currentSelection =
+ options.filter((item) => selected && item.value === selected) ?? []
+ const accountSelectorIsDisabled = options.length <= 0
+
+ const customFilter: ComboboxProps['customFilter'] = (
+ _value,
+ search,
+ keywords
+ ) => {
+ const _search = search.toLowerCase().trim()
+ const _keys =
+ keywords &&
+ keywords.some((keyword) =>
+ keyword.toLowerCase().trim().includes(_search)
+ )
+
+ return _keys ? 1 : 0
+ }
+ const onChangeSelections = (values: Array) => {
+ if (values.length <= 0) {
+ changeSelection(null)
+ }
+ }
+ const onSelectItem = (value: string) => {
+ changeSelection(value)
+ }
+
+ return {
+ accountSelectorIsDisabled,
+ currentSelection,
+ data,
+ options,
+ selected,
+
+ customFilter,
+ onChangeSelections,
+ onSelectItem,
+ }
+}
+
+export function useFriendsActions() {
+ const { account, isLoading, updateLoading } =
+ useFriendsManagementActions()
+
+ const getFriendsSummary = () => {
+ if (!account) {
+ return
+ }
+
+ updateLoading(true)
+
+ window.electronAPI.requestFriendsSummary(account)
+ }
+
+ return {
+ isLoading,
+
+ getFriendsSummary,
+ }
+}
+
+export function useFriendsTableActions() {
+ const { account, getSummary } = useFriendsManagementActions()
+
+ const handleBlock =
+ (
+ friends: Array,
+ context?: 'incoming' | 'outgoing'
+ ) =>
+ () => {
+ if (!account) {
+ return
+ }
+
+ window.electronAPI.requestBlockFriends(account, friends, context)
+ }
+ const handleUnblock =
+ (unblocklist: 'full' | Array) => () => {
+ if (!account) {
+ return
+ }
+
+ const currentTotal = getSummary()?.blocklist.length ?? 0
+
+ window.electronAPI.requestUnblockFriends(
+ account,
+ unblocklist !== 'full' && currentTotal === unblocklist.length
+ ? 'full'
+ : unblocklist
+ )
+ }
+
+ const handleAdd =
+ (friends: Array, context?: 'incoming') => () => {
+ if (!account) {
+ return
+ }
+
+ window.electronAPI.requestAddFriends(account, friends, context)
+ }
+
+ const handleRemove =
+ (
+ friends: 'full' | Array,
+ context?: 'incoming' | 'outgoing'
+ ) =>
+ () => {
+ if (!account) {
+ return
+ }
+
+ const currentTotal = getSummary()?.friends.length ?? 0
+
+ window.electronAPI.requestRemoveFriends(
+ account,
+ friends !== 'full' && currentTotal === friends.length
+ ? 'full'
+ : friends,
+ context
+ )
+ }
+
+ return {
+ handleBlock,
+ handleUnblock,
+ handleAdd,
+ handleRemove,
+ }
+}
diff --git a/src/routes/account-management/friends-management/-sections/-blocklist.tsx b/src/routes/account-management/friends-management/-sections/-blocklist.tsx
new file mode 100644
index 00000000..cae2032b
--- /dev/null
+++ b/src/routes/account-management/friends-management/-sections/-blocklist.tsx
@@ -0,0 +1,53 @@
+import type { FriendsSummary } from '../../../../types/services/friends'
+
+import {
+ CustomSearch,
+ CustomTable,
+ CustomTablePagination,
+ useTableConfig,
+} from '../../../../components/ui/extended/form/table'
+
+import { useBlocklistColumns } from '../-columns'
+
+import { numberWithCommaSeparator } from '../../../../lib/parsers/numbers'
+
+export function BlocklistSection({
+ data,
+}: {
+ data: FriendsSummary['blocklist']
+}) {
+ const { columns } = useBlocklistColumns()
+
+ const { perPageList, table } = useTableConfig({
+ columns,
+ data,
+ rowId: 'accountId',
+ })
+
+ return (
+
+
+ Total Blocklist
+
+ {numberWithCommaSeparator(data.length)}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/routes/account-management/friends-management/-sections/-friends.tsx b/src/routes/account-management/friends-management/-sections/-friends.tsx
new file mode 100644
index 00000000..d936bc39
--- /dev/null
+++ b/src/routes/account-management/friends-management/-sections/-friends.tsx
@@ -0,0 +1,61 @@
+import type { FriendsSummary } from '../../../../types/services/friends'
+
+import {
+ CustomSearch,
+ CustomTable,
+ CustomTablePagination,
+ useTableConfig,
+} from '../../../../components/ui/extended/form/table'
+
+import { useFriendsColumns } from '../-columns'
+
+import { useFnnyHandleActions } from '../../../information/credits/-hooks'
+import { useFriendsManagement } from '../../../../hooks/management/friends'
+
+import { numberWithCommaSeparator } from '../../../../lib/parsers/numbers'
+
+export function FriendsSection({
+ data,
+}: {
+ data: FriendsSummary['friends']
+}) {
+ const { columns } = useFriendsColumns()
+ const { selected } = useFriendsManagement()
+ const { handleMnomoeAttrs } = useFnnyHandleActions()
+
+ const { perPageList, table } = useTableConfig({
+ columns,
+ data,
+ rowId: 'accountId',
+ })
+
+ return (
+
+
+ Total Friends
+
+ {numberWithCommaSeparator(data.length)}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/routes/account-management/friends-management/-sections/-incoming.tsx b/src/routes/account-management/friends-management/-sections/-incoming.tsx
new file mode 100644
index 00000000..eaf088f7
--- /dev/null
+++ b/src/routes/account-management/friends-management/-sections/-incoming.tsx
@@ -0,0 +1,53 @@
+import type { FriendsSummary } from '../../../../types/services/friends'
+
+import {
+ CustomSearch,
+ CustomTable,
+ CustomTablePagination,
+ useTableConfig,
+} from '../../../../components/ui/extended/form/table'
+
+import { useIncomingColumns } from '../-columns'
+
+import { numberWithCommaSeparator } from '../../../../lib/parsers/numbers'
+
+export function IncomingSection({
+ data,
+}: {
+ data: FriendsSummary['incoming']
+}) {
+ const { columns } = useIncomingColumns()
+
+ const { perPageList, table } = useTableConfig({
+ columns,
+ data,
+ rowId: 'accountId',
+ })
+
+ return (
+
+
+ Total Incoming
+
+ {numberWithCommaSeparator(data.length)}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/routes/account-management/friends-management/-sections/-outgoing.tsx b/src/routes/account-management/friends-management/-sections/-outgoing.tsx
new file mode 100644
index 00000000..af5feef8
--- /dev/null
+++ b/src/routes/account-management/friends-management/-sections/-outgoing.tsx
@@ -0,0 +1,53 @@
+import type { FriendsSummary } from '../../../../types/services/friends'
+
+import {
+ CustomSearch,
+ CustomTable,
+ CustomTablePagination,
+ useTableConfig,
+} from '../../../../components/ui/extended/form/table'
+
+import { useOutgoingColumns } from '../-columns'
+
+import { numberWithCommaSeparator } from '../../../../lib/parsers/numbers'
+
+export function OutgoingSection({
+ data,
+}: {
+ data: FriendsSummary['outgoing']
+}) {
+ const { columns } = useOutgoingColumns()
+
+ const { perPageList, table } = useTableConfig({
+ columns,
+ data,
+ rowId: 'accountId',
+ })
+
+ return (
+
+
+ Total Outgoing
+
+ {numberWithCommaSeparator(data.length)}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/routes/account-management/friends-management/route.tsx b/src/routes/account-management/friends-management/route.tsx
new file mode 100644
index 00000000..15d5bff4
--- /dev/null
+++ b/src/routes/account-management/friends-management/route.tsx
@@ -0,0 +1,175 @@
+import { createRoute } from '@tanstack/react-router'
+import { useTranslation } from 'react-i18next'
+
+import { Route as RootRoute } from '../../__root'
+
+import { HomeBreadcrumb } from '../../../components/navigations/breadcrumb/home'
+import {
+ Breadcrumb,
+ BreadcrumbItem,
+ BreadcrumbList,
+ BreadcrumbPage,
+ BreadcrumbSeparator,
+} from '../../../components/ui/breadcrumb'
+import { Button } from '../../../components/ui/button'
+import { Combobox } from '../../../components/ui/extended/combobox'
+// import { Label } from '../../../components/ui/label'
+import {
+ Tabs,
+ TabsContent,
+ TabsList,
+ TabsTrigger,
+} from '../../../components/ui/tabs'
+import { GoToTop } from '../../../components/go-to-top'
+
+import { FriendsSection } from './-sections/-friends'
+import { IncomingSection } from './-sections/-incoming'
+import { OutgoingSection } from './-sections/-outgoing'
+import { BlocklistSection } from './-sections/-blocklist'
+
+import { useCustomizableMenuSettingsVisibility } from '../../../hooks/settings'
+import { useAccountSelector, useFriendsActions } from './-hooks'
+
+import { cn } from '../../../lib/utils'
+
+export const Route = createRoute({
+ getParentRoute: () => RootRoute,
+ path: '/account-management/friends-management',
+ component: () => {
+ const { t } = useTranslation(['sidebar'], {
+ keyPrefix: 'account-management',
+ })
+
+ return (
+ <>
+
+
+
+
+
+ {t('title')}
+
+
+
+
+ {t('options.friends-management')}
+
+
+
+
+
+ >
+ )
+ },
+})
+
+function Content() {
+ const { t } = useTranslation(['account-management', 'general'])
+
+ const {
+ accountSelectorIsDisabled,
+ currentSelection,
+ data,
+ options,
+ selected,
+ customFilter,
+ onChangeSelections,
+ onSelectItem,
+ } = useAccountSelector()
+ const { isLoading, getFriendsSummary } = useFriendsActions()
+ const { getMenuOptionVisibility } =
+ useCustomizableMenuSettingsVisibility()
+
+ return (
+ <>
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Friends
+ Incoming
+ Outgoing
+ Blocklist
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ >
+ )
+}
diff --git a/src/routes/information/credits/-hooks.tsx b/src/routes/information/credits/-hooks.tsx
index 893c9edd..c851b63b 100644
--- a/src/routes/information/credits/-hooks.tsx
+++ b/src/routes/information/credits/-hooks.tsx
@@ -4,7 +4,7 @@ import { repositoryAssetsURL } from '../../../config/about/links'
import { toast } from '../../../lib/notifications'
-export function useActions() {
+export function useFnnyHandleActions() {
const handleFreshAttrs = useLongPress(
() => {
toast(
@@ -28,6 +28,29 @@ export function useActions() {
}
)
+ const handleMnomoeAttrs = useLongPress(
+ () => {
+ toast(
+
+
+ ,
+ {
+ style: {
+ padding: 0,
+ },
+ unstyled: true,
+ }
+ )
+ },
+ {
+ threshold: 500,
+ }
+ )
+
const handleEricDejaDeJoder = () => {
toast(
@@ -48,6 +71,7 @@ export function useActions() {
}
)
}
+
const handleSick = () => {
toast(
@@ -69,6 +93,7 @@ export function useActions() {
return {
handleEricDejaDeJoder,
handleFreshAttrs,
+ handleMnomoeAttrs,
handleSick,
}
}
@@ -78,3 +103,6 @@ const qdasickimg =
const frshimg =
''
+
+const mnomoe =
+ ''
diff --git a/src/routes/information/credits/route.tsx b/src/routes/information/credits/route.tsx
index 67fe2b2c..e0d9a7af 100644
--- a/src/routes/information/credits/route.tsx
+++ b/src/routes/information/credits/route.tsx
@@ -17,7 +17,7 @@ import {
BreadcrumbSeparator,
} from '../../../components/ui/breadcrumb'
-import { useActions } from './-hooks'
+import { useFnnyHandleActions } from './-hooks'
import { whatIsThis } from '../../../lib/callbacks'
import { cn } from '../../../lib/utils'
@@ -40,7 +40,7 @@ export function ComponentRoute() {
const { t } = useTranslation(['general'])
const { handleEricDejaDeJoder, handleFreshAttrs, handleSick } =
- useActions()
+ useFnnyHandleActions()
const openURL = (url: string) => (event: MouseEvent) => {
event.preventDefault()
diff --git a/src/routes/stw-operations/xpboosts/-why.tsx b/src/routes/stw-operations/xpboosts/-why.tsx
index a5b645b7..aa7dee50 100644
--- a/src/routes/stw-operations/xpboosts/-why.tsx
+++ b/src/routes/stw-operations/xpboosts/-why.tsx
@@ -9,6 +9,30 @@ export function useWhy({
}) {
const [showLink, setShowLink] = useState(false)
+ const handleLve = () => {
+ const options = [['ku', 'da', 'ena', 'mor', 'ado'].join('')]
+ const value = inputSearchValue.trim().toLowerCase()
+
+ if (options.includes(value)) {
+ toast(
+
+
+ ,
+ {
+ style: {
+ backgroundColor: 'hsl(0 0% 3.9%)',
+ padding: 0,
+ },
+ unstyled: true,
+ }
+ )
+ }
+ }
+
const handleXD = () => {
const options = [
['cu', 'di', 'ta'].join(''),
@@ -74,9 +98,13 @@ export function useWhy({
showLink,
handleXD,
+ handleLve,
handleWhy,
}
}
const qdaimg =
''
+
+const qdalve =
+ ''
diff --git a/src/routes/stw-operations/xpboosts/route.tsx b/src/routes/stw-operations/xpboosts/route.tsx
index 3476e563..6ffd6857 100644
--- a/src/routes/stw-operations/xpboosts/route.tsx
+++ b/src/routes/stw-operations/xpboosts/route.tsx
@@ -142,7 +142,7 @@ function Content() {
} = useSearchUser()
const { getMenuOptionVisibility } =
useCustomizableMenuSettingsVisibility()
- const { showLink, handleXD, handleWhy } = useWhy({
+ const { showLink, handleLve, handleXD, handleWhy } = useWhy({
inputSearchValue: inputSearchDisplayName,
})
const { recalculateTotal, teammateXPBoostsFiltered } = useFilterXPBoosts(
@@ -193,6 +193,7 @@ function Content() {
if (!inputSearchButtonIsDisabled) {
handleSearchUser()
+ handleLve()
handleXD()
handleWhy()
}
diff --git a/src/services/endpoints/friends.ts b/src/services/endpoints/friends.ts
index 4428b396..14fb3f74 100644
--- a/src/services/endpoints/friends.ts
+++ b/src/services/endpoints/friends.ts
@@ -1,4 +1,7 @@
-import type { FetchFriendResponse } from '../../types/services/friends'
+import type {
+ EpicFriend,
+ FriendsSummary,
+} from '../../types/services/friends'
import { friendsService } from '../config/friends'
@@ -11,7 +14,7 @@ export function getFriend({
accountId: string
friendId: string
}) {
- return friendsService.get(
+ return friendsService.get(
`/${accountId}/friends/${friendId}`,
{
headers: {
@@ -40,3 +43,37 @@ export function addFriend({
}
)
}
+
+export function blockFriend({
+ accessToken,
+ accountId,
+ friendId,
+}: {
+ accessToken: string
+ accountId: string
+ friendId: string
+}) {
+ return friendsService.post(
+ `/${accountId}/blocklist/${friendId}`,
+ {},
+ {
+ headers: {
+ Authorization: `bearer ${accessToken}`,
+ },
+ }
+ )
+}
+
+export function getFriendsSummary({
+ accessToken,
+ accountId,
+}: {
+ accessToken: string
+ accountId: string
+}) {
+ return friendsService.get(`/${accountId}/summary`, {
+ headers: {
+ Authorization: `bearer ${accessToken}`,
+ },
+ })
+}
diff --git a/src/services/endpoints/lookup.ts b/src/services/endpoints/lookup.ts
index 2ae50822..84bcf130 100644
--- a/src/services/endpoints/lookup.ts
+++ b/src/services/endpoints/lookup.ts
@@ -5,6 +5,24 @@ import type {
import { publicAccountService } from '../config/public-account'
+export function queryAccountsByIds({
+ accessToken,
+ ids,
+}: {
+ accessToken: string
+ ids: Array
+}) {
+ const parsed = ids.map((id) => `accountId=${id}`).slice(0, 100)
+
+ return publicAccountService.get<
+ Array
+ >(`?${parsed.join('&')}`, {
+ headers: {
+ Authorization: `bearer ${accessToken}`,
+ },
+ })
+}
+
export function findUserByAccountId({
accessToken,
accountId,
diff --git a/src/state/advanced-mode/matchmaking-track/temp-players.ts b/src/state/advanced-mode/matchmaking-track/temp-players.ts
index f6f8141a..e86f6c3e 100644
--- a/src/state/advanced-mode/matchmaking-track/temp-players.ts
+++ b/src/state/advanced-mode/matchmaking-track/temp-players.ts
@@ -24,7 +24,16 @@ export const useMatchmakingRecentlyPlayersStore =
if (addNewPlayer) {
set({
players: [...currentPlayers, player].toSorted((itemA, itemB) =>
- localeCompareForSorting(itemA.displayName, itemB.displayName)
+ localeCompareForSorting(
+ itemA.displayName ??
+ itemA.externalAuths?.xbl?.externalDisplayName ??
+ itemA.externalAuths?.psn?.externalDisplayName ??
+ itemA.id,
+ itemB.displayName ??
+ itemB.externalAuths?.xbl?.externalDisplayName ??
+ itemB.externalAuths?.psn?.externalDisplayName ??
+ itemB.id
+ )
),
})
@@ -41,7 +50,16 @@ export const useMatchmakingRecentlyPlayersStore =
: current.displayName,
}))
.toSorted((itemA, itemB) =>
- localeCompareForSorting(itemA.displayName, itemB.displayName)
+ localeCompareForSorting(
+ itemA.displayName ??
+ itemA.externalAuths?.xbl?.externalDisplayName ??
+ itemA.externalAuths?.psn?.externalDisplayName ??
+ itemA.id,
+ itemB.displayName ??
+ itemB.externalAuths?.xbl?.externalDisplayName ??
+ itemB.externalAuths?.psn?.externalDisplayName ??
+ itemB.id
+ )
),
})
},
diff --git a/src/state/management/friends.ts b/src/state/management/friends.ts
new file mode 100644
index 00000000..613b63d8
--- /dev/null
+++ b/src/state/management/friends.ts
@@ -0,0 +1,232 @@
+import type {
+ EpicFriend,
+ FriendBlock,
+ FriendIncoming,
+ FriendOutgoing,
+ FriendsSummary,
+} from '../../types/services/friends'
+
+import { immer } from 'zustand/middleware/immer'
+import { create } from 'zustand'
+
+import { localeCompareForSorting } from '../../lib/utils'
+
+export type FriendsState = {
+ accounts: Record
+ isLoading: boolean
+ selected: string | null
+
+ changeSelection: (value: string | null) => void
+ getSummary: (accountId?: string | null) => FriendsSummary | null
+ removeFriends: (accountId: string, ids?: Array) => void
+ syncIncoming: (
+ type: 'add' | 'remove',
+ accountId: string,
+ incoming?: Array
+ ) => void
+ syncOutgoing: (
+ type: 'add' | 'remove',
+ accountId: string,
+ outgoing?: Array
+ ) => void
+ syncBlocklist: (
+ type: 'add' | 'remove',
+ accountId: string,
+ blocklist?: Array
+ ) => void
+ syncSummary: (accountId: string, summary: FriendsSummary | null) => void
+ updateLoading: (value: boolean) => void
+}
+
+export const defaultFriendsSummary: FriendsSummary = {
+ friends: [],
+ incoming: [],
+ outgoing: [],
+ blocklist: [],
+ suggested: [],
+ settings: null,
+ limitsReached: null,
+}
+
+export const useFriendsStore = create()(
+ immer((set, get) => ({
+ accounts: {},
+ isLoading: false,
+ selected: null,
+
+ changeSelection: (selected) =>
+ set({
+ selected,
+ }),
+ getSummary: (accountId) => {
+ if (accountId === undefined) {
+ const selected = get().selected
+
+ if (!selected) {
+ return null
+ }
+
+ return get().accounts[selected] ?? null
+ }
+
+ if (!accountId) {
+ return null
+ }
+
+ return get().accounts[accountId] ?? null
+ },
+ removeFriends: (accountId, ids) => {
+ const current = get().accounts[accountId]
+
+ if (!current || (ids && ids.length <= 0)) {
+ return
+ }
+
+ if (ids === undefined) {
+ set((state) => {
+ state.accounts[accountId].friends = []
+ })
+
+ return
+ }
+
+ set((state) => {
+ state.accounts[accountId].friends = state.accounts[
+ accountId
+ ].friends.filter((friend) => !ids.includes(friend.accountId))
+ })
+ },
+ syncIncoming: (type, accountId, incoming) => {
+ const current = get().accounts[accountId]
+
+ if (!current || (incoming && incoming.length <= 0)) {
+ return
+ }
+
+ if (incoming === undefined) {
+ set((state) => {
+ state.accounts[accountId].incoming = []
+ })
+
+ return
+ }
+
+ if (type === 'add') {
+ const list = [...current.incoming, ...incoming].toSorted(
+ (itemA, itemB) =>
+ localeCompareForSorting(itemA.displayName!, itemB.displayName!)
+ )
+
+ set((state) => {
+ state.accounts[accountId].incoming = list.map(
+ (item) =>
+ ({
+ accountId: item.accountId,
+ created: item.created,
+ favorite: false,
+ mutual: 0,
+ displayName: item.displayName,
+ }) as FriendIncoming
+ )
+ })
+
+ return
+ }
+
+ const ids = incoming.map((item) => item.accountId)
+
+ set((state) => {
+ state.accounts[accountId].incoming = current.incoming.filter(
+ (item) => !ids.includes(item.accountId)
+ )
+ })
+ },
+ syncOutgoing: (type, accountId, outgoing) => {
+ const current = get().accounts[accountId]
+
+ if (!current || (outgoing && outgoing.length <= 0)) {
+ return
+ }
+
+ if (outgoing === undefined) {
+ set((state) => {
+ state.accounts[accountId].outgoing = []
+ })
+
+ return
+ }
+
+ if (type === 'add') {
+ const list = [...current.outgoing, ...outgoing].toSorted(
+ (itemA, itemB) =>
+ localeCompareForSorting(itemA.displayName!, itemB.displayName!)
+ )
+
+ set((state) => {
+ state.accounts[accountId].outgoing = list.map(
+ (item) =>
+ ({
+ accountId: item.accountId,
+ created: item.created,
+ favorite: false,
+ mutual: 0,
+ displayName: item.displayName,
+ }) as FriendOutgoing
+ )
+ })
+
+ return
+ }
+
+ const ids = outgoing.map((item) => item.accountId)
+
+ set((state) => {
+ state.accounts[accountId].outgoing = current.outgoing.filter(
+ (item) => !ids.includes(item.accountId)
+ )
+ })
+ },
+ syncBlocklist: (type, accountId, blocklist) => {
+ const current = get().accounts[accountId]
+
+ if (!current || (blocklist && blocklist.length <= 0)) {
+ return
+ }
+
+ if (blocklist === undefined) {
+ set((state) => {
+ state.accounts[accountId].blocklist = []
+ })
+
+ return
+ }
+
+ if (type === 'add') {
+ const list = [...current.blocklist, ...blocklist].toSorted(
+ (itemA, itemB) =>
+ localeCompareForSorting(itemA.displayName!, itemB.displayName!)
+ )
+
+ set((state) => {
+ state.accounts[accountId].blocklist = list
+ })
+
+ return
+ }
+
+ const ids = blocklist.map((item) => item.accountId)
+
+ set((state) => {
+ state.accounts[accountId].blocklist = current.blocklist.filter(
+ (item) => !ids.includes(item.accountId)
+ )
+ })
+ },
+ syncSummary: (accountId, summary) => {
+ set((state) => {
+ state.accounts[accountId] = summary ?? defaultFriendsSummary
+ })
+ },
+ updateLoading: (isLoading) => set({ isLoading }),
+ }))
+)
diff --git a/src/state/settings/customizable-menu.ts b/src/state/settings/customizable-menu.ts
index 36162b51..4823a577 100644
--- a/src/state/settings/customizable-menu.ts
+++ b/src/state/settings/customizable-menu.ts
@@ -36,6 +36,7 @@ export const customizableMenuSettingsRelations: Record<
],
accountManagement: [
'vbucksInformation',
+ 'friendsManagement',
'redeemCodes',
'devicesAuth',
'epicGamesSettings',
diff --git a/src/types/services/friends.d.ts b/src/types/services/friends.d.ts
index a187cfb0..0b3c5b8d 100644
--- a/src/types/services/friends.d.ts
+++ b/src/types/services/friends.d.ts
@@ -1,8 +1,62 @@
-export type FetchFriendResponse = {
+import type { StringUnion } from '../utils'
+
+export type EpicAddFriend = {
accountId: string
+ displayName?: string
+}
+
+export type EpicFriend = {
+ accountId: string
+ displayName?: string
groups: Array
alias: string
+ mutual: number
note: string
favorite: boolean
created: string
}
+
+export type FriendIncoming = {
+ accountId: string
+ displayName?: string
+ mutual: number
+ favorite: boolean
+ created: string
+}
+
+export type FriendOutgoing = {
+ accountId: string
+ displayName?: string
+ mutual: number
+ favorite: boolean
+ created: string
+}
+
+export type FriendBlock = {
+ accountId: string
+ displayName?: string
+ created: string
+}
+
+export type FriendSuggested = Record
+
+export type AccountSettings = {
+ acceptInvites: StringUnion<'public'>
+ mutualPrivacy: StringUnion<'ALL', 'NONE'>
+}
+
+export type AccountLimitsReached = {
+ incoming: boolean
+ outgoing: boolean
+ accepted: boolean
+}
+
+export type FriendsSummary = {
+ friends: Array
+ incoming: Array
+ outgoing: Array
+ blocklist: Array
+ suggested: Array
+ settings: AccountSettings | null
+ limitsReached: AccountLimitsReached | null
+}
diff --git a/src/types/services/lookup.d.ts b/src/types/services/lookup.d.ts
index 212654de..e5d02b6c 100644
--- a/src/types/services/lookup.d.ts
+++ b/src/types/services/lookup.d.ts
@@ -2,8 +2,27 @@ import { StringUnion } from '../utils.d'
export type LookupFindOneByDisplayNameResponse = {
id: string
- displayName: string
- externalAuths?: Partial>
+ displayName?: string
+ externalAuths?: {
+ psn?: {
+ accountId: string
+ type: StringUnion<'psn'>
+ externalAuthId: string
+ externalAuthIdType: StringUnion<'psn_user_id'>
+ externalDisplayName: string
+ authIds: Array<{
+ id: string
+ type: StringUnion<'psn_user_id'>
+ }>
+ }
+ xbl?: {
+ accountId: string
+ type: StringUnion<'xbl'>
+ externalAuthIdType: StringUnion<'xuid'>
+ externalDisplayName: string
+ authIds: Array
+ }
+ }
}
export type LookupFindManyByDisplayNameResponse =
@@ -11,7 +30,7 @@ export type LookupFindManyByDisplayNameResponse =
export type LookupFindManyByDisplayName = {
id: string
- displayName: string
+ displayName?: string
externalAuths: {
psn?: {
accountId: string
diff --git a/src/types/services/mcp/query-profile-main.d.ts b/src/types/services/mcp/query-profile-main.d.ts
index a946d6c5..30e5f52e 100644
--- a/src/types/services/mcp/query-profile-main.d.ts
+++ b/src/types/services/mcp/query-profile-main.d.ts
@@ -23,14 +23,14 @@ export type MCPQueryProfileMainProfile = {
lastInterval: string
purchaseList: Record
}
- daily_purchases: {
+ daily_purchases?: {
lastInterval: string
- purchaseList: Record
+ purchaseList?: Record
}
- in_app_purchases: {
+ in_app_purchases?: {
receipts: Array
ignoredReceipts: Array
- fulfillmentCounts: Record
+ fulfillmentCounts?: Record
refreshTimers: {
EpicPurchasingService: {
nextEntitlementRefresh: string