format: prettify entire project
This commit is contained in:
parent
86f0782a98
commit
c4e5762224
8
.gitattributes
vendored
Normal file
8
.gitattributes
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
* text=auto
|
||||
*.sh text eol=lf
|
||||
*.js text eol=lf
|
||||
*.ts text eol=lf
|
||||
*.json text eol=lf
|
||||
*.md text eol=lf
|
||||
*.html text eol=lf
|
||||
*.css text eol=lf
|
8
.prettierrc
Normal file
8
.prettierrc
Normal file
@ -0,0 +1,8 @@
|
||||
{
|
||||
"semi": true,
|
||||
"singleQuote": true,
|
||||
"trailingComma": "es5",
|
||||
"endOfLine": "lf",
|
||||
"experimentalTernaries": true,
|
||||
"bracketSameLine": true
|
||||
}
|
15
README.md
15
README.md
@ -24,9 +24,11 @@ COD Tracker provides a clean interface to fetch, display, and export Call of Dut
|
||||
- Call of Duty account with API security settings set to "ALL"
|
||||
|
||||
## Authentication Setup
|
||||
|
||||
<!-- <div align="center">
|
||||
<img src="https://img.shields.io/badge/IMPORTANT-Authentication_Required-red" alt="Authentication Required"/>
|
||||
</div> -->
|
||||
|
||||
### Changing Account API Privacy Settings
|
||||
|
||||
To use this application, you need to update your Call of Duty profile settings:
|
||||
@ -41,14 +43,14 @@ To use this application, you need to update your Call of Duty profile settings:
|
||||
### Obtaining Your SSO Token
|
||||
|
||||
<table>
|
||||
<tr>
|
||||
<tr></tr>
|
||||
<td width="60%">
|
||||
|
||||
The application requires an authentication token to access the Call of Duty API:
|
||||
|
||||
1. Log in to [Call of Duty](https://profile.callofduty.com/cod/profile)
|
||||
2. Open developer tools (F12 or right-click → Inspect)
|
||||
3. Navigate to: **Application** → **Storage** → **Cookies** → **https://profile.callofduty.com**
|
||||
3. Navigate to: **Application** → **Storage** → **Cookies** → **<https://profile.callofduty.com>**
|
||||
4. Find and copy the value of `ACT_SSO_COOKIE`
|
||||
5. Paste this token into the "SSO Token" field
|
||||
|
||||
@ -57,10 +59,10 @@ The application requires an authentication token to access the Call of Duty API:
|
||||
</td>
|
||||
<td width="40%">
|
||||
|
||||
```
|
||||
```plaintext
|
||||
Application
|
||||
└─ Storage
|
||||
└─ Cookies
|
||||
└─ Cookies
|
||||
└─ ACT_SSO_COOKIE
|
||||
```
|
||||
|
||||
@ -75,17 +77,20 @@ The SSO token typically expires after 24 hours. If you encounter authentication
|
||||
## Installation
|
||||
|
||||
1. Clone the repository:
|
||||
|
||||
```bash
|
||||
git clone https://git.rimmyscorner.com/Rim/codtracker-js.git && cd codtracker-js
|
||||
```
|
||||
|
||||
2. Start the application:
|
||||
|
||||
```bash
|
||||
npm run start
|
||||
```
|
||||
|
||||
3. Open your browser and navigate to:
|
||||
```
|
||||
|
||||
```bash
|
||||
http://127.0.0.1:3513
|
||||
```
|
||||
|
||||
|
@ -187,7 +187,7 @@ button:hover {
|
||||
top: 5px;
|
||||
}
|
||||
|
||||
.checkbox-group input[type="checkbox"] {
|
||||
.checkbox-group input[type='checkbox'] {
|
||||
width: auto;
|
||||
margin-right: 5px;
|
||||
}
|
||||
|
@ -1,153 +1,153 @@
|
||||
{
|
||||
"br_25": "BR Trios",
|
||||
"br_dmz_38": "Plunder Quads",
|
||||
"br_71": "BR Solos",
|
||||
"br_74": "BR Trios",
|
||||
"br_dmz_76": "Plunder Quads",
|
||||
"br_77": "BR Scopes & Scatterguns",
|
||||
"br_dmz_85": "Plunder Duos",
|
||||
"br_86": "Realism Battle Royale",
|
||||
"br_87": "BR Solos",
|
||||
"br_88": "BR Duos",
|
||||
"br_89": "BR Quads",
|
||||
"br_dmz_104": "Blood Money",
|
||||
"brtdm_113": "Warzone Rumble",
|
||||
"brtdm_rmbl": "Warzone Rumble",
|
||||
"br_brsolo": "BR Solos",
|
||||
"br_brduos": "BR Duos",
|
||||
"br_brtrios": "BR Trios",
|
||||
"br_brquads": "BR Quads",
|
||||
"br_br_real": "Realism Battle Royale",
|
||||
"br_dmz_plnbld": "Plunder Blood Money",
|
||||
"br_brthquad": "BR 200 Quads",
|
||||
"br_brduostim_name2": "BR Stimulus Duos",
|
||||
"br_brtriostim_name2": "BR Stimulus Trios",
|
||||
"br_dmz_pluntrios": "Plunder Trios",
|
||||
"br_dmz_plunquad": "Plunder Quads",
|
||||
"br_jugg_brtriojugr": "Juggernaut Royal Trios",
|
||||
"br_jugg_brquadjugr": "Juggernaut Royal Quads",
|
||||
"br_mini_miniroyale": "Mini Royale",
|
||||
"br_brbbsolo": "BR Buyback Solos",
|
||||
"br_brbbduo": "BR Buyback Duos",
|
||||
"br_brbbtrio": "BR Buyback Trios",
|
||||
"br_brbbquad": "BR Buyback Quads",
|
||||
"br_miniroyale": "Mini Royale",
|
||||
"br_kingslayer_kingsltrios": "King Slayer Trios",
|
||||
"br_truckwar_trwarsquads": "Armored Royale",
|
||||
"br_dmz_plndtrios": "Plunder Trios",
|
||||
"br_dmz_plndquad": "Plunder Quads",
|
||||
"br_zxp_zmbroy": "Zombie Royale",
|
||||
"br_jugg_jugpmpkn": "Juggourdnaut Royale",
|
||||
"br_brsolohwn": "BR Solo Survivor",
|
||||
"br_brduohwn": "BR Duo Die",
|
||||
"br_brhwntrios": "BR Trick-Or-Trios",
|
||||
"br_brhwnquad": "BR Monster Quads",
|
||||
"br_dmz_plndcndy": "Plunder: Candy Collector",
|
||||
"br_dmz_bldmnytrio": "Blood Money Trios",
|
||||
"br_dmz_bldmnyquad": "Blood Money Quads",
|
||||
"br_rebirth_rbrthduos": "Rebirth Duos",
|
||||
"br_rebirth_rbrthtrios": "Rebirth Trios",
|
||||
"br_rebirth_rbrthquad": "Rebirth Quads",
|
||||
"br_mini_rebirth_mini_royale_solo": "Rebirth Mini Royale Solos",
|
||||
"br_mini_rebirth_mini_royale_duos": "Rebirth Mini Royale Duos",
|
||||
"br_mini_rebirth_mini_royale_trios": "Rebirth Mini Royale Trios",
|
||||
"br_mini_rebirth_mini_royale_quads": "Rebirth Mini Royale Quads",
|
||||
"brtdm_wzrumval2": "Warzone Rumble in the Sheets",
|
||||
"br_dmz_plndval1": "Love And Plunder",
|
||||
"br_rebirth_rbrthex": "Resurgence Extreme",
|
||||
"br_exfiltrios": "Exfiltration Trios",
|
||||
"br_rbrthduos": "Rebirth Resurgence Duos",
|
||||
"br_rbrthquad": "Rebirth Resurgence Quads",
|
||||
"br_brz_brduos": "BR Duos (Containment Protocol Event)",
|
||||
"br_brz_brtrios": "BR Trios (Containment Protocol Event)",
|
||||
"br_brz_brquads": "BR Quads (Containment Protocol Event)",
|
||||
"br_kingslayer_rebirth_king_slayer": "Rebirth King Slayer",
|
||||
"br_reveal_dov": "Destruction Of Verdansk Part 1",
|
||||
"br_reveal_2_dov2": "Destruction Of Verdansk Part 2",
|
||||
"br_brdov_dov2": "Destruction Of Verdansk Part 2 Verdansk '84",
|
||||
"br_rebirth_resurgence_trios": "Verdansk Resurgence Trios",
|
||||
"br_bodycount_pwergrb": "Power Grab",
|
||||
"br_rebirth_resurgence_mini": "Verdansk Resurgence Mini",
|
||||
"br_plnbld": "Blood Money",
|
||||
"br_plndtrios": "Plunder Trios",
|
||||
"br_payload_payload": "Payload",
|
||||
"br_x2_br_reveal_x2_event/event_title_x2": "Battle of Verdansk",
|
||||
"br_rumble_lua_menu_mp/clash": "Rebirth Payload",
|
||||
"br_rumble_clash": "Clash",
|
||||
"br_dbd_dbd": "Iron Trials '84",
|
||||
"br_rebirth_rebirth_rex": "Rebirth Extreme",
|
||||
"br_rebirth_shsnp_name3": "Rebirth Scopes & Scatterguns",
|
||||
"br_payload_pay_prom": "Payload - Promenade",
|
||||
"br_gxp_gov": "Ghosts Of Verdansk",
|
||||
"br_dbd_iron_trials_solos": "Iron Trials Solos",
|
||||
"br_dbd_iron_trials_duos": "Iron Trials Duos",
|
||||
"br_dbd_iron_trials_trios": "Iron Trials Trios",
|
||||
"br_dbd_iron_trials_quads": "Iron Trials Quads",
|
||||
"br_buy_back_solo": "BR Buy Back Solos",
|
||||
"br_buy_back_duos": "BR Buy Back Duos",
|
||||
"br_buy_back_trios": "BR Buy Back Trios",
|
||||
"br_buy_back_quads": "BR Buy Back Quads",
|
||||
"br_rebirth_rust_1v1": "Rebirth Resurgence Duos",
|
||||
"br_vov_op_flash": "Operation: Flashback",
|
||||
"br_lep_br_lep_event/ltm_gamemode": "Final Hours of Verdansk",
|
||||
"br_vg_royale_solo": "Vanguard Royale Solos",
|
||||
"br_vg_royale_duos": "Vanguard Royale Duos",
|
||||
"br_vg_royale_trios": "Vanguard Royale Trios",
|
||||
"br_vg_royale_quads": "Vanguard Royale Quads",
|
||||
"br_br_solo": "BR Solos",
|
||||
"br_br_duos": "BR Duos",
|
||||
"br_br_trios": "BR Trios",
|
||||
"br_br_quads": "BR Quads",
|
||||
"br_rebirth_vg_res_44": "Vanguard Resurgence",
|
||||
"br_rebirth_cal_res_royale": "Caldera Resurgence Quads",
|
||||
"br_dmz_plnduo": "Plunder Duos",
|
||||
"br_dmz_vg_pln_trios": "Vanguard Plunder Trios",
|
||||
"br_dmz_vg_pln_quads": "Vanguard Plunder Quads",
|
||||
"br_rumble_clash_caldera": "Caldera Clash",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_solos": "Iron Trials Rebirth Solos",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_duos": "Iron Trials Rebirth Duos",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_trios": "Iron Trials Rebirth Trios",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_quads": "Iron Trials Rebirth Quads",
|
||||
"br_rebirth_calderaresurgence": "Caldera Resurgence",
|
||||
"br_rebirth_reverse_playlist_wz325/rbrthsolos": "Rebirth Resurgence Solos",
|
||||
"br_payload_playlist_wz325/rbrthpayload": "Rebirth Payload",
|
||||
"br_rebirth_cdl_resurgence_rebirth_quads": "CDL Resurgence Rebirth Quads",
|
||||
"br_dmz_playlist_wz325/rbrthbmo_quads": "Rebirth Blood Money Quads",
|
||||
"br_playlist_wz325/br_aprl_fool_name4": "Totally Normal BR",
|
||||
"br_rebirth_playlist_wz325/afd_resurgence": "Totally Normal Rebirth",
|
||||
"br_dbd_playlist_wz330/cal_iron_solos": "Iron Trials Solos",
|
||||
"br_dbd_playlist_wz330/cal_iron_duos": "Iron Trials Duos",
|
||||
"br_dbd_playlist_wz330/cal_iron_trios": "Iron Trials Trios",
|
||||
"br_dbd_playlist_wz330/cal_iron_quads": "Iron Trials Quads",
|
||||
"br_mendota_playlist_wz330/op_mon": "Operation Monarch",
|
||||
"br_respect_playlist_wz335/respect": "Champions of Caldera",
|
||||
"br_playlist_wz335/rebirthexfilttrios": "Rebirth Exfiltration Trios",
|
||||
"br_rebirth_cal_res_trios": "Caldera Resurgence Trios",
|
||||
"br_rebirth_cal_res_quads": "Caldera Resurgence Quads",
|
||||
"br_rebirth_reverse_playlist_wz340/fortkeep_res_solo": "Fortunes Keep Resurgence Solos",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_duos": "Fortunes Keep Resurgence Duos",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_trios": "Fortunes Keep Resurgence Trios",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_quad": "Fortunes Keep Resurgence Quads",
|
||||
"br_gold_war_playlist_wz340/gld_pldr": "Golden Plunder",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_solo": "Titanium Trials Solos",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_duos": "Titanium Trials Duos",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_trios": "Titanium Trials Trios",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_quads": "Titanium Trials Quads",
|
||||
"br_rebirth_playlist_wz340/fortkeep_extreme": "Fortunes Keep Extreme",
|
||||
"br_rumble_playlist_wz340/storage_town_clash_title": "Storage Town Clash",
|
||||
"br_zxp_playlist_wz345/rxp": "Rebirth Of The Dead",
|
||||
"br_respect_playlist_wz345/respect_solo": "Champion of Caldera Solos",
|
||||
"br_respect_playlist_wz345/respect_trios": "Champion Of Caldera Trios",
|
||||
"br_wsow_br_trios": "World Series Of Warzone Battle Royale Trios",
|
||||
"br_olaride_playlist_wz350/olaride": "Operation: Last Call",
|
||||
"br_mmp_playlist_wz350/mmp": "Sticks & Stones",
|
||||
"br_rebirth_cdlr:_fortune's_keep_trios": "CDLR: Fortune's Keep Trios",
|
||||
"br_mini_minibrsolo": "Mini Royale Solos",
|
||||
"br_mini_minibrduos": "Mini Royale Duos",
|
||||
"br_mini_minibrtrios": "Mini Royale Trios",
|
||||
"br_mini_minibrquads": "Mini Royale Quads",
|
||||
"br_rebirth_dbd_reverse_playlist_wz355/res_trials_solos": "Rebirth Supreme Solos",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_duos": "Rebirth Supreme Duos",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_trios": "Rebirth Supreme Trios",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_quads": "Rebirth Supreme Quads"
|
||||
"br_25": "BR Trios",
|
||||
"br_dmz_38": "Plunder Quads",
|
||||
"br_71": "BR Solos",
|
||||
"br_74": "BR Trios",
|
||||
"br_dmz_76": "Plunder Quads",
|
||||
"br_77": "BR Scopes & Scatterguns",
|
||||
"br_dmz_85": "Plunder Duos",
|
||||
"br_86": "Realism Battle Royale",
|
||||
"br_87": "BR Solos",
|
||||
"br_88": "BR Duos",
|
||||
"br_89": "BR Quads",
|
||||
"br_dmz_104": "Blood Money",
|
||||
"brtdm_113": "Warzone Rumble",
|
||||
"brtdm_rmbl": "Warzone Rumble",
|
||||
"br_brsolo": "BR Solos",
|
||||
"br_brduos": "BR Duos",
|
||||
"br_brtrios": "BR Trios",
|
||||
"br_brquads": "BR Quads",
|
||||
"br_br_real": "Realism Battle Royale",
|
||||
"br_dmz_plnbld": "Plunder Blood Money",
|
||||
"br_brthquad": "BR 200 Quads",
|
||||
"br_brduostim_name2": "BR Stimulus Duos",
|
||||
"br_brtriostim_name2": "BR Stimulus Trios",
|
||||
"br_dmz_pluntrios": "Plunder Trios",
|
||||
"br_dmz_plunquad": "Plunder Quads",
|
||||
"br_jugg_brtriojugr": "Juggernaut Royal Trios",
|
||||
"br_jugg_brquadjugr": "Juggernaut Royal Quads",
|
||||
"br_mini_miniroyale": "Mini Royale",
|
||||
"br_brbbsolo": "BR Buyback Solos",
|
||||
"br_brbbduo": "BR Buyback Duos",
|
||||
"br_brbbtrio": "BR Buyback Trios",
|
||||
"br_brbbquad": "BR Buyback Quads",
|
||||
"br_miniroyale": "Mini Royale",
|
||||
"br_kingslayer_kingsltrios": "King Slayer Trios",
|
||||
"br_truckwar_trwarsquads": "Armored Royale",
|
||||
"br_dmz_plndtrios": "Plunder Trios",
|
||||
"br_dmz_plndquad": "Plunder Quads",
|
||||
"br_zxp_zmbroy": "Zombie Royale",
|
||||
"br_jugg_jugpmpkn": "Juggourdnaut Royale",
|
||||
"br_brsolohwn": "BR Solo Survivor",
|
||||
"br_brduohwn": "BR Duo Die",
|
||||
"br_brhwntrios": "BR Trick-Or-Trios",
|
||||
"br_brhwnquad": "BR Monster Quads",
|
||||
"br_dmz_plndcndy": "Plunder: Candy Collector",
|
||||
"br_dmz_bldmnytrio": "Blood Money Trios",
|
||||
"br_dmz_bldmnyquad": "Blood Money Quads",
|
||||
"br_rebirth_rbrthduos": "Rebirth Duos",
|
||||
"br_rebirth_rbrthtrios": "Rebirth Trios",
|
||||
"br_rebirth_rbrthquad": "Rebirth Quads",
|
||||
"br_mini_rebirth_mini_royale_solo": "Rebirth Mini Royale Solos",
|
||||
"br_mini_rebirth_mini_royale_duos": "Rebirth Mini Royale Duos",
|
||||
"br_mini_rebirth_mini_royale_trios": "Rebirth Mini Royale Trios",
|
||||
"br_mini_rebirth_mini_royale_quads": "Rebirth Mini Royale Quads",
|
||||
"brtdm_wzrumval2": "Warzone Rumble in the Sheets",
|
||||
"br_dmz_plndval1": "Love And Plunder",
|
||||
"br_rebirth_rbrthex": "Resurgence Extreme",
|
||||
"br_exfiltrios": "Exfiltration Trios",
|
||||
"br_rbrthduos": "Rebirth Resurgence Duos",
|
||||
"br_rbrthquad": "Rebirth Resurgence Quads",
|
||||
"br_brz_brduos": "BR Duos (Containment Protocol Event)",
|
||||
"br_brz_brtrios": "BR Trios (Containment Protocol Event)",
|
||||
"br_brz_brquads": "BR Quads (Containment Protocol Event)",
|
||||
"br_kingslayer_rebirth_king_slayer": "Rebirth King Slayer",
|
||||
"br_reveal_dov": "Destruction Of Verdansk Part 1",
|
||||
"br_reveal_2_dov2": "Destruction Of Verdansk Part 2",
|
||||
"br_brdov_dov2": "Destruction Of Verdansk Part 2 Verdansk '84",
|
||||
"br_rebirth_resurgence_trios": "Verdansk Resurgence Trios",
|
||||
"br_bodycount_pwergrb": "Power Grab",
|
||||
"br_rebirth_resurgence_mini": "Verdansk Resurgence Mini",
|
||||
"br_plnbld": "Blood Money",
|
||||
"br_plndtrios": "Plunder Trios",
|
||||
"br_payload_payload": "Payload",
|
||||
"br_x2_br_reveal_x2_event/event_title_x2": "Battle of Verdansk",
|
||||
"br_rumble_lua_menu_mp/clash": "Rebirth Payload",
|
||||
"br_rumble_clash": "Clash",
|
||||
"br_dbd_dbd": "Iron Trials '84",
|
||||
"br_rebirth_rebirth_rex": "Rebirth Extreme",
|
||||
"br_rebirth_shsnp_name3": "Rebirth Scopes & Scatterguns",
|
||||
"br_payload_pay_prom": "Payload - Promenade",
|
||||
"br_gxp_gov": "Ghosts Of Verdansk",
|
||||
"br_dbd_iron_trials_solos": "Iron Trials Solos",
|
||||
"br_dbd_iron_trials_duos": "Iron Trials Duos",
|
||||
"br_dbd_iron_trials_trios": "Iron Trials Trios",
|
||||
"br_dbd_iron_trials_quads": "Iron Trials Quads",
|
||||
"br_buy_back_solo": "BR Buy Back Solos",
|
||||
"br_buy_back_duos": "BR Buy Back Duos",
|
||||
"br_buy_back_trios": "BR Buy Back Trios",
|
||||
"br_buy_back_quads": "BR Buy Back Quads",
|
||||
"br_rebirth_rust_1v1": "Rebirth Resurgence Duos",
|
||||
"br_vov_op_flash": "Operation: Flashback",
|
||||
"br_lep_br_lep_event/ltm_gamemode": "Final Hours of Verdansk",
|
||||
"br_vg_royale_solo": "Vanguard Royale Solos",
|
||||
"br_vg_royale_duos": "Vanguard Royale Duos",
|
||||
"br_vg_royale_trios": "Vanguard Royale Trios",
|
||||
"br_vg_royale_quads": "Vanguard Royale Quads",
|
||||
"br_br_solo": "BR Solos",
|
||||
"br_br_duos": "BR Duos",
|
||||
"br_br_trios": "BR Trios",
|
||||
"br_br_quads": "BR Quads",
|
||||
"br_rebirth_vg_res_44": "Vanguard Resurgence",
|
||||
"br_rebirth_cal_res_royale": "Caldera Resurgence Quads",
|
||||
"br_dmz_plnduo": "Plunder Duos",
|
||||
"br_dmz_vg_pln_trios": "Vanguard Plunder Trios",
|
||||
"br_dmz_vg_pln_quads": "Vanguard Plunder Quads",
|
||||
"br_rumble_clash_caldera": "Caldera Clash",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_solos": "Iron Trials Rebirth Solos",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_duos": "Iron Trials Rebirth Duos",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_trios": "Iron Trials Rebirth Trios",
|
||||
"br_dbd_playlist_wz320/rbrthdbd_quads": "Iron Trials Rebirth Quads",
|
||||
"br_rebirth_calderaresurgence": "Caldera Resurgence",
|
||||
"br_rebirth_reverse_playlist_wz325/rbrthsolos": "Rebirth Resurgence Solos",
|
||||
"br_payload_playlist_wz325/rbrthpayload": "Rebirth Payload",
|
||||
"br_rebirth_cdl_resurgence_rebirth_quads": "CDL Resurgence Rebirth Quads",
|
||||
"br_dmz_playlist_wz325/rbrthbmo_quads": "Rebirth Blood Money Quads",
|
||||
"br_playlist_wz325/br_aprl_fool_name4": "Totally Normal BR",
|
||||
"br_rebirth_playlist_wz325/afd_resurgence": "Totally Normal Rebirth",
|
||||
"br_dbd_playlist_wz330/cal_iron_solos": "Iron Trials Solos",
|
||||
"br_dbd_playlist_wz330/cal_iron_duos": "Iron Trials Duos",
|
||||
"br_dbd_playlist_wz330/cal_iron_trios": "Iron Trials Trios",
|
||||
"br_dbd_playlist_wz330/cal_iron_quads": "Iron Trials Quads",
|
||||
"br_mendota_playlist_wz330/op_mon": "Operation Monarch",
|
||||
"br_respect_playlist_wz335/respect": "Champions of Caldera",
|
||||
"br_playlist_wz335/rebirthexfilttrios": "Rebirth Exfiltration Trios",
|
||||
"br_rebirth_cal_res_trios": "Caldera Resurgence Trios",
|
||||
"br_rebirth_cal_res_quads": "Caldera Resurgence Quads",
|
||||
"br_rebirth_reverse_playlist_wz340/fortkeep_res_solo": "Fortunes Keep Resurgence Solos",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_duos": "Fortunes Keep Resurgence Duos",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_trios": "Fortunes Keep Resurgence Trios",
|
||||
"br_rebirth_playlist_wz340/fortkeep_res_quad": "Fortunes Keep Resurgence Quads",
|
||||
"br_gold_war_playlist_wz340/gld_pldr": "Golden Plunder",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_solo": "Titanium Trials Solos",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_duos": "Titanium Trials Duos",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_trios": "Titanium Trials Trios",
|
||||
"br_tdbd_playlist_wz345/cal_titanium_quads": "Titanium Trials Quads",
|
||||
"br_rebirth_playlist_wz340/fortkeep_extreme": "Fortunes Keep Extreme",
|
||||
"br_rumble_playlist_wz340/storage_town_clash_title": "Storage Town Clash",
|
||||
"br_zxp_playlist_wz345/rxp": "Rebirth Of The Dead",
|
||||
"br_respect_playlist_wz345/respect_solo": "Champion of Caldera Solos",
|
||||
"br_respect_playlist_wz345/respect_trios": "Champion Of Caldera Trios",
|
||||
"br_wsow_br_trios": "World Series Of Warzone Battle Royale Trios",
|
||||
"br_olaride_playlist_wz350/olaride": "Operation: Last Call",
|
||||
"br_mmp_playlist_wz350/mmp": "Sticks & Stones",
|
||||
"br_rebirth_cdlr:_fortune's_keep_trios": "CDLR: Fortune's Keep Trios",
|
||||
"br_mini_minibrsolo": "Mini Royale Solos",
|
||||
"br_mini_minibrduos": "Mini Royale Duos",
|
||||
"br_mini_minibrtrios": "Mini Royale Trios",
|
||||
"br_mini_minibrquads": "Mini Royale Quads",
|
||||
"br_rebirth_dbd_reverse_playlist_wz355/res_trials_solos": "Rebirth Supreme Solos",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_duos": "Rebirth Supreme Duos",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_trios": "Rebirth Supreme Trios",
|
||||
"br_rebirth_dbd_playlist_wz355/res_trials_quads": "Rebirth Supreme Quads"
|
||||
}
|
||||
|
@ -1,337 +1,337 @@
|
||||
{
|
||||
"career": "Career",
|
||||
"mp_hackney_yard": "Hackney Yard (Night)",
|
||||
"mp_aniyah": "Aniyah Palace",
|
||||
"mp_euphrates": "Euphrates Bridge",
|
||||
"mp_raid": "Grazna Raid",
|
||||
"mp_m_pine": "Pine",
|
||||
"mp_m_stack": "Stack",
|
||||
"mp_deadzone": "Arklov Peak",
|
||||
"mp_quarry": "Karst River Quarry",
|
||||
"mp_m_overunder": "Docks",
|
||||
"mp_cave_am": "Azhir Cave",
|
||||
"mp_cave": "Azhir Cave (Night)",
|
||||
"mp_runner": "Gun Runner",
|
||||
"mp_runner_pm": "Gun Runner (Night)",
|
||||
"mp_hackney_am": "Hackney Yard",
|
||||
"mp_piccadilly": "Piccadilly",
|
||||
"mp_spear": "Rammaza",
|
||||
"mp_spear_pm": "Rammaza (Night)",
|
||||
"mp_petrograd": "St. Petrograd",
|
||||
"mp_m_hill": "Hill",
|
||||
"mp_m_king": "King",
|
||||
"mp_m_speedball": "Speedball",
|
||||
"mp_m_showers": "Gulag Showers",
|
||||
"mp_downtown_gw": "Tarvosk District",
|
||||
"mp_m_speed": "Shoot House",
|
||||
"mp_farms2_gw": "Krovnik Farmland",
|
||||
"mp_port2_gw": "Port",
|
||||
"mp_crash2": "Crash",
|
||||
"mp_vacant": "Vacant",
|
||||
"mp_shipment": "Shipment",
|
||||
"mp_m_cargo": "Cargo",
|
||||
"mp_m_cage": "Atrium",
|
||||
"mp_m_overwinter": "Docks",
|
||||
"mp_emporium": "Atlas Superstore",
|
||||
"mp_rust": "Rust",
|
||||
"mp_boneyard_gw": "Zhokov Boneyard",
|
||||
"mp_m_fork": "Bazaar",
|
||||
"mp_donetsk": "Verdansk",
|
||||
"mp_hideout": "Khandor Hideout",
|
||||
"loading_mp_hideout": "Khandor Hideout",
|
||||
"mp_aniyah_tac": "Aniyah Incursion",
|
||||
"mp_backlot2": "Talsik Backlot",
|
||||
"mp_village2": "Hovec Sawmill",
|
||||
"mp_hardhat": "Hardhat",
|
||||
"mp_m_wallco2": "Aisle 9",
|
||||
"mp_scrapyard": "Zhokov Scrapyard",
|
||||
"mp_m_trench": "Trench",
|
||||
"mp_promenade_gw": "Barakett Promenade",
|
||||
"mp_don": "Verdansk",
|
||||
"mp_garden": "Cheshire Park",
|
||||
"mp_oilrig": "Petrov Oil Rig",
|
||||
"mp_harbor": "Suldal Harbor",
|
||||
"mp_layover_gw": "Verdansk International Airport",
|
||||
"mp_m_cornfield": "Livestock",
|
||||
"mp_m_stadium": "Verdansk Stadium",
|
||||
"mp_malyshev": "Mialstor Tank Factory",
|
||||
"mp_malyshev_10v10": "Mialstor Tank Factory",
|
||||
"mp_broadcast2": "Broadcast",
|
||||
"mp_riverside_gw": "Verdansk Riverside",
|
||||
"mp_m_train": "Station",
|
||||
"mp_kstenod": "Verdansk (Night)",
|
||||
"mp_escape": "Rebirth",
|
||||
"mp_herat": "Al-Raab Airbase",
|
||||
"mp_killhouse": "Killhouse",
|
||||
"mp_m_drainage": "Drainage",
|
||||
"war": "Team Deathmatch",
|
||||
"sd": "Search and Destroy",
|
||||
"dom": "Domination",
|
||||
"tdef": "Team Defender",
|
||||
"dm": "Free-for-all",
|
||||
"koth": "Hardpoint",
|
||||
"hq": "Headquarters",
|
||||
"arena": "Gunfight",
|
||||
"arm": "Ground War",
|
||||
"conf": "Kill Confirmed",
|
||||
"cyber": "Cyber Attack",
|
||||
"hc_war": "Team Deathmatch Hardcore",
|
||||
"hc_arena": "Gunfight Hardcore",
|
||||
"hc_arm": "Ground War Hardcore",
|
||||
"hc_conf": "Kill Confirmed Hardcore",
|
||||
"hc_cyber": "Cyber Attack Hardcore",
|
||||
"hc_dm": "Free-for-all Hardcore",
|
||||
"hc_hq": "Headquarters Hardcore",
|
||||
"hc_dom": "Domination Hardcore",
|
||||
"hc_sd": "Search and Destroy Hardcore",
|
||||
"cyber_hc": "Cyber Attack Hardcore",
|
||||
"war_hc": "Team Deathmatch Hardcore",
|
||||
"dom_hc": "Domination Hardcore",
|
||||
"sd_hc": "Search and Destroy Hardcore",
|
||||
"conf_hc": "Kill Confirmed Hardcore",
|
||||
"gun": "Gun Game",
|
||||
"gun_hc": "Gun Game Hardcore",
|
||||
"siege": "Reinforce",
|
||||
"infect": "Infected",
|
||||
"arena_osp": "Gunfight O.S.P.",
|
||||
"hq_hc": "Headquarters Hardcore",
|
||||
"grnd": "Grind",
|
||||
"grind": "Grind",
|
||||
"ctf": "Capture the Flag",
|
||||
"br_all": "All",
|
||||
"br": "Battle Royale",
|
||||
"br_dmz": "Plunder",
|
||||
"br_dmz_38": "Plunder Quads",
|
||||
"br_87": "BR Solos",
|
||||
"br_dmz_104": "Blood Money",
|
||||
"koth_hc": "Hardpoint Hardcore",
|
||||
"br_25": "BR Trios",
|
||||
"br_89": "BR Quads",
|
||||
"br_dmz_76": "Plunder Quads",
|
||||
"br_77": "BR Scopes & Scatterguns",
|
||||
"br_dmz_85": "Plunder Duos",
|
||||
"dd_hc": "Demolition Hardcore",
|
||||
"dd": "Demolition",
|
||||
"br_71": "BR Solos",
|
||||
"br_74": "BR Trios",
|
||||
"br_88": "BR Duos",
|
||||
"brtdm_113": "Warzone Rumble",
|
||||
"brtdm_rmbl": "Warzone Rumble",
|
||||
"br_brsolo": "BR Solos",
|
||||
"br_brduos": "BR Duos",
|
||||
"br_brtrios": "BR Trios",
|
||||
"br_brquads": "BR Quads",
|
||||
"br_dmz_plnbld": "Blood Money",
|
||||
"br_br_real": "Realism Battle Royale",
|
||||
"br_86": "Realism Battle Royale",
|
||||
"br_brthquad": "BR 200 Quads",
|
||||
"br_jugg_brtriojugr": "Juggernaut Royal Trios",
|
||||
"br_dmz_plunquad": "Plunder Quads",
|
||||
"br_dmz_bldmnytrio": "Blood Money Trios",
|
||||
"br_mini_miniroyale": "Mini Royale",
|
||||
"br_brbbsolo": "BR Buyback Solos",
|
||||
"br_jugg_brquadjugr": "Juggernaut Royal Quads",
|
||||
"br_kingslayer_kingsltrios": "King Slayer Trios",
|
||||
"br_truckwar_trwarsquads": "Armored Royale Quads",
|
||||
"br_zxp_zmbroy": "Zombie Royale",
|
||||
"br_brhwntrios": "BR Trick-Or-Trios",
|
||||
"rugby": "Onslaughter",
|
||||
"br_brsolohwn": "BR Solo Survivor",
|
||||
"br_dmz_plndcndy": "Plunder: Candy Collector",
|
||||
"br_jugg_jugpmpkn": "Juggourdnaut Royale",
|
||||
"br_rebirth_rbrthtrios": "Resurgence Trio",
|
||||
"br_rebirth_rbrthduos": "Resurgence Duos",
|
||||
"br_rebirth_rbrthquad": "Rebirth Resurgance Quads",
|
||||
"br_dmz_plndtrios": "Plunder Trios",
|
||||
"br_rebirth_resurgence_trios": "Verdansk Resurgence Trios",
|
||||
"br_mini_rebirth_mini_royale_quads": "Rebirth Mini Royale Quads",
|
||||
"br_bodycount_pwergrb": "Power Grab",
|
||||
"br_rebirth_resurgence_mini": "Verdansk Resurgence Mini",
|
||||
"br_payload_payload": "Payload",
|
||||
"br_mini_rebirth_mini_royale_trios": "Rebirth Mini Royale Trios",
|
||||
"br_x2_br_reveal_x2_event/event_title_x2": "Battle of Verdansk",
|
||||
"br_rumble_clash": "Clash",
|
||||
"br_dbd_dbd": "Iron Trials '84",
|
||||
"br_gxp_gov": "Ghosts of Verdansk",
|
||||
"scorestreak": "Scorestreak",
|
||||
"equipment": "Equipment",
|
||||
"gear": "Gear",
|
||||
"weapon_bare_hands": "Bare Hands",
|
||||
"weapon_tactical_rifle": "Tactical Rifle",
|
||||
"weapon_shotgun": "Shotgun",
|
||||
"weapon_sniper": "Sniper",
|
||||
"weapon_lmg": "Light Machine Guns",
|
||||
"weapon_launcher": "Launcher",
|
||||
"weapon_pistol": "Pistol",
|
||||
"weapon_smg": "Submachine Guns",
|
||||
"weapon_melee": "Melee",
|
||||
"weapon_assault_rifle": "Assault Rifle",
|
||||
"attachments": "Attachments",
|
||||
"weapons": "Weapons",
|
||||
"specialist": "Specialist",
|
||||
"weapon": "Weapon",
|
||||
"weapon_special": "Special",
|
||||
"iw8_ar_akilo47": "AK-47",
|
||||
"iw8_ar_kilo433": "Kilo-141",
|
||||
"iw8_ar_mcharlie": "M13",
|
||||
"iw8_ar_falima": "FAL",
|
||||
"iw8_ar_asierra12": "Oden",
|
||||
"iw8_sm_mpapa7": "MP7",
|
||||
"iw8_sm_augolf": "AUG",
|
||||
"iw8_sm_uzulu": "Uzi",
|
||||
"iw8_sh_romeo870": "Model 680",
|
||||
"iw8_sh_charlie725": "725",
|
||||
"iw8_sh_aalpha12": "JAK-12",
|
||||
"iw8_sh_oscar12": "Origin 12",
|
||||
"iw8_lm_pkilo": "PKM",
|
||||
"iw8_lm_mgolf34": "MG34",
|
||||
"iw8_lm_lima86": "SA87",
|
||||
"iw8_lm_dblmg": "MP Juggernaut",
|
||||
"iw8_sn_mike14": "EBR-14",
|
||||
"iw8_sn_delta": "Dragunov",
|
||||
"iw8_sn_alpha50": "AX-50",
|
||||
"iw8_sn_hdromeo": "HDR",
|
||||
"iw8_sn_sbeta": "Mk2 Carbine",
|
||||
"iw8_pi_papa320": "M19",
|
||||
"iw8_pi_cpapa": ".357",
|
||||
"iw8_la_rpapa7": "RPG-7",
|
||||
"iw8_la_juliet": "JOKR",
|
||||
"iw8_la_gromeo": "PILA",
|
||||
"iw8_la_kgolf": "Strela-P",
|
||||
"iw8_me_riotshield": "Riot Shield",
|
||||
"equip_gas_grenade": "Gas Grenade",
|
||||
"equip_snapshot_grenade": "Snapshot Grenade",
|
||||
"equip_decoy": "Decoy Grenade",
|
||||
"equip_smoke": "Smoke Grenade",
|
||||
"equip_concussion": "Stun Grenade",
|
||||
"equip_hb_sensor": "Heartbeat Sensor",
|
||||
"equip_flash": "Flash Grenade",
|
||||
"equip_adrenaline": "Stim",
|
||||
"equip_frag": "Frag Grenade",
|
||||
"equip_thermite": "Thermite",
|
||||
"equip_semtex": "Semtex",
|
||||
"equip_claymore": "Claymore",
|
||||
"equip_c4": "C4",
|
||||
"equip_at_mine": "Proximity Mine",
|
||||
"equip_throwing_knife": "Throwing Knife",
|
||||
"equip_molotov": "Molotov Cocktail",
|
||||
"iw8_knife": "Combat Knife",
|
||||
"weapon_other": "Primary Melee",
|
||||
"iw8_ar_tango21": "RAM-7",
|
||||
"iw8_ar_falpha": "FR 5.56",
|
||||
"iw8_ar_mike4": "M4A1",
|
||||
"iw8_sm_papa90": "P90",
|
||||
"iw8_sm_mpapa5": "MP5",
|
||||
"iw8_sm_beta": "PP19 Bizon",
|
||||
"iw8_sh_dpapa12": "R9-0",
|
||||
"iw8_lm_mgolf36": "Holger-26",
|
||||
"iw8_sn_kilo98": "Kar98k",
|
||||
"iw8_pi_mike1911": "1911",
|
||||
"iw8_pi_golf21": "X16",
|
||||
"iw8_pi_decho": ".50 GS",
|
||||
"weapon_marksman": "Marksman Rifles",
|
||||
"iw8_lm_kilo121": "M91",
|
||||
"iw8_ar_scharlie": "FN Scar 17",
|
||||
"iw8_ar_sierra552": "Grau 5.56",
|
||||
"iw8_sm_smgolf45": "Striker 45",
|
||||
"iw8_pi_mike9a3": "Renetti",
|
||||
"iw8_lm_mkilo3": "Bruen MK9",
|
||||
"iw8_sh_mike26": "VLK Rogue",
|
||||
"iw8_sn_crossbow": "Crossbow",
|
||||
"iw8_sn_sksierra": "SKS",
|
||||
"iw8_ar_galima": "CR-56 AMAX",
|
||||
"iw8_me_kalistick": "Kali Sticks",
|
||||
"iw8_sm_victor": "Fennec Mk9",
|
||||
"iw8_sn_xmike109": "Rytec AMR",
|
||||
"iw8_pi_mike9": "Renetti",
|
||||
"iw8_me_akimboblunt": "Kali Sticks",
|
||||
"iw8_ar_anovember94": "AN-94",
|
||||
"iw8_sm_charlie9": "ISO",
|
||||
"iw8_me_akimboblades": "Dual Kodachis",
|
||||
"iw8_lm_sierrax": "FiNN",
|
||||
"iw8_ar_valpha": "AS VAL",
|
||||
"iw8_sn_romeo700": "SP-R 208",
|
||||
"cruise_predator": "Cruise Missile",
|
||||
"manual_turret": "Shield Turret",
|
||||
"toma_strike": "Cluster Strike",
|
||||
"sentry_gun": "Sentry Gun",
|
||||
"hover_jet": "VTOL Jet",
|
||||
"precision_airstrike": "Precision Airstrike",
|
||||
"juggernaut": "Juggernaut",
|
||||
"pac_sentry": "",
|
||||
"chopper_gunner": "Chopper Gunner",
|
||||
"gunship": "Gunship",
|
||||
"white_phosphorus": "White Phosphorus",
|
||||
"nuke": "Nuke",
|
||||
"chopper_support": "Support Helo",
|
||||
"bradley": "Infantry Assault Vehicle",
|
||||
"uav": "UAV",
|
||||
"directional_uav": "Advanced UAV",
|
||||
"airdrop": "Care Package",
|
||||
"airdrop_multiple": "Emergency Airdrop",
|
||||
"radar_drone_overwatch": "Personal Radar",
|
||||
"scrambler_drone_guard": "Counter UAV",
|
||||
"super_emp_drone": "EMP Drone",
|
||||
"super_trophy": "Trophy System",
|
||||
"super_ammo_drop": "Munitions Box",
|
||||
"super_weapon_drop": "Weapon Drop",
|
||||
"super_fulton": "Cash Deposit Balloon",
|
||||
"super_armor_drop": "Armor Box",
|
||||
"super_select": "Field Upgrade Pro (Any)",
|
||||
"super_tac_insert": "Tactical Insertion",
|
||||
"super_recon_drone": "Recon Drone",
|
||||
"super_deadsilence": "Dead Silence",
|
||||
"super_supply_drop": "Loadout Drop",
|
||||
"super_tac_cover": "Deployable Cover",
|
||||
"super_support_box": "Stopping Power Rounds",
|
||||
"mp_stat": "Statistic",
|
||||
"session_start": "Session Start",
|
||||
"uno": "PC",
|
||||
"psn": "Playstation Network",
|
||||
"xbl": "Xbox Live",
|
||||
"steam": "Steam",
|
||||
"battle": "BattleNET",
|
||||
"mw": "Modern Warfare",
|
||||
"cw": "Cold War",
|
||||
"mp_cartel": "Cartel",
|
||||
"mp_tank": "Garrison",
|
||||
"mp_miami": "Miami",
|
||||
"mp_moscow": "Moscow",
|
||||
"mp_satellite": "Satellite",
|
||||
"mp_kgb": "Checkmate",
|
||||
"wz_forest": "Ruka",
|
||||
"wz_ski_slopes": "Alpine",
|
||||
"mp_nuketown6": "Nuketown '84",
|
||||
"mp_tundra": "Crossroads",
|
||||
"mp_black_sea": "Armada",
|
||||
"mp_mall": "The Pines",
|
||||
"mp_raid_rm": "Raid",
|
||||
"mp_sm_berlin_tunnel": "U-Bahn",
|
||||
"mp_sm_finance": "KGB",
|
||||
"mp_sm_game_show": "Game Show",
|
||||
"mp_sm_central": "ICBM",
|
||||
"wz_sanatorium": "Sanatorium",
|
||||
"nuketown6_holiday": "Nuketown '84 Holiday",
|
||||
"mp_express_rm": "Express",
|
||||
"mp_apocalypse": "Apocalypse",
|
||||
"mp_sm_market": "Mansion",
|
||||
"mp_miami_strike": "Miami Strike",
|
||||
"wz_golova": "Golova",
|
||||
"mp_cliffhanger": "Yamantau",
|
||||
"mp_sm_gas_station": "Diesel",
|
||||
"wz_duga": "Duga",
|
||||
"mp_village_rm": "Standoff",
|
||||
"mp_sm_amsterdam": "Amsterdam",
|
||||
"mp_dune": "Collateral",
|
||||
"mp_hijacked_rm": "Hijacked",
|
||||
"mp_paintball_rm": "Rush",
|
||||
"mp_sm_deptstore": "Showroom",
|
||||
"mp_slums_rm": "Slums",
|
||||
"mp_echelon": "Echelon",
|
||||
"mp_drivein_rm": "Drive In",
|
||||
"mp_zoo_rm": "Zoo",
|
||||
"mp_firebase": "Deprogram",
|
||||
"mp_amerika": "Amerika",
|
||||
"mp_sm_vault": "Gluboko",
|
||||
"mp_don4_pm": "Nuketown '84 Halloween"
|
||||
"career": "Career",
|
||||
"mp_hackney_yard": "Hackney Yard (Night)",
|
||||
"mp_aniyah": "Aniyah Palace",
|
||||
"mp_euphrates": "Euphrates Bridge",
|
||||
"mp_raid": "Grazna Raid",
|
||||
"mp_m_pine": "Pine",
|
||||
"mp_m_stack": "Stack",
|
||||
"mp_deadzone": "Arklov Peak",
|
||||
"mp_quarry": "Karst River Quarry",
|
||||
"mp_m_overunder": "Docks",
|
||||
"mp_cave_am": "Azhir Cave",
|
||||
"mp_cave": "Azhir Cave (Night)",
|
||||
"mp_runner": "Gun Runner",
|
||||
"mp_runner_pm": "Gun Runner (Night)",
|
||||
"mp_hackney_am": "Hackney Yard",
|
||||
"mp_piccadilly": "Piccadilly",
|
||||
"mp_spear": "Rammaza",
|
||||
"mp_spear_pm": "Rammaza (Night)",
|
||||
"mp_petrograd": "St. Petrograd",
|
||||
"mp_m_hill": "Hill",
|
||||
"mp_m_king": "King",
|
||||
"mp_m_speedball": "Speedball",
|
||||
"mp_m_showers": "Gulag Showers",
|
||||
"mp_downtown_gw": "Tarvosk District",
|
||||
"mp_m_speed": "Shoot House",
|
||||
"mp_farms2_gw": "Krovnik Farmland",
|
||||
"mp_port2_gw": "Port",
|
||||
"mp_crash2": "Crash",
|
||||
"mp_vacant": "Vacant",
|
||||
"mp_shipment": "Shipment",
|
||||
"mp_m_cargo": "Cargo",
|
||||
"mp_m_cage": "Atrium",
|
||||
"mp_m_overwinter": "Docks",
|
||||
"mp_emporium": "Atlas Superstore",
|
||||
"mp_rust": "Rust",
|
||||
"mp_boneyard_gw": "Zhokov Boneyard",
|
||||
"mp_m_fork": "Bazaar",
|
||||
"mp_donetsk": "Verdansk",
|
||||
"mp_hideout": "Khandor Hideout",
|
||||
"loading_mp_hideout": "Khandor Hideout",
|
||||
"mp_aniyah_tac": "Aniyah Incursion",
|
||||
"mp_backlot2": "Talsik Backlot",
|
||||
"mp_village2": "Hovec Sawmill",
|
||||
"mp_hardhat": "Hardhat",
|
||||
"mp_m_wallco2": "Aisle 9",
|
||||
"mp_scrapyard": "Zhokov Scrapyard",
|
||||
"mp_m_trench": "Trench",
|
||||
"mp_promenade_gw": "Barakett Promenade",
|
||||
"mp_don": "Verdansk",
|
||||
"mp_garden": "Cheshire Park",
|
||||
"mp_oilrig": "Petrov Oil Rig",
|
||||
"mp_harbor": "Suldal Harbor",
|
||||
"mp_layover_gw": "Verdansk International Airport",
|
||||
"mp_m_cornfield": "Livestock",
|
||||
"mp_m_stadium": "Verdansk Stadium",
|
||||
"mp_malyshev": "Mialstor Tank Factory",
|
||||
"mp_malyshev_10v10": "Mialstor Tank Factory",
|
||||
"mp_broadcast2": "Broadcast",
|
||||
"mp_riverside_gw": "Verdansk Riverside",
|
||||
"mp_m_train": "Station",
|
||||
"mp_kstenod": "Verdansk (Night)",
|
||||
"mp_escape": "Rebirth",
|
||||
"mp_herat": "Al-Raab Airbase",
|
||||
"mp_killhouse": "Killhouse",
|
||||
"mp_m_drainage": "Drainage",
|
||||
"war": "Team Deathmatch",
|
||||
"sd": "Search and Destroy",
|
||||
"dom": "Domination",
|
||||
"tdef": "Team Defender",
|
||||
"dm": "Free-for-all",
|
||||
"koth": "Hardpoint",
|
||||
"hq": "Headquarters",
|
||||
"arena": "Gunfight",
|
||||
"arm": "Ground War",
|
||||
"conf": "Kill Confirmed",
|
||||
"cyber": "Cyber Attack",
|
||||
"hc_war": "Team Deathmatch Hardcore",
|
||||
"hc_arena": "Gunfight Hardcore",
|
||||
"hc_arm": "Ground War Hardcore",
|
||||
"hc_conf": "Kill Confirmed Hardcore",
|
||||
"hc_cyber": "Cyber Attack Hardcore",
|
||||
"hc_dm": "Free-for-all Hardcore",
|
||||
"hc_hq": "Headquarters Hardcore",
|
||||
"hc_dom": "Domination Hardcore",
|
||||
"hc_sd": "Search and Destroy Hardcore",
|
||||
"cyber_hc": "Cyber Attack Hardcore",
|
||||
"war_hc": "Team Deathmatch Hardcore",
|
||||
"dom_hc": "Domination Hardcore",
|
||||
"sd_hc": "Search and Destroy Hardcore",
|
||||
"conf_hc": "Kill Confirmed Hardcore",
|
||||
"gun": "Gun Game",
|
||||
"gun_hc": "Gun Game Hardcore",
|
||||
"siege": "Reinforce",
|
||||
"infect": "Infected",
|
||||
"arena_osp": "Gunfight O.S.P.",
|
||||
"hq_hc": "Headquarters Hardcore",
|
||||
"grnd": "Grind",
|
||||
"grind": "Grind",
|
||||
"ctf": "Capture the Flag",
|
||||
"br_all": "All",
|
||||
"br": "Battle Royale",
|
||||
"br_dmz": "Plunder",
|
||||
"br_dmz_38": "Plunder Quads",
|
||||
"br_87": "BR Solos",
|
||||
"br_dmz_104": "Blood Money",
|
||||
"koth_hc": "Hardpoint Hardcore",
|
||||
"br_25": "BR Trios",
|
||||
"br_89": "BR Quads",
|
||||
"br_dmz_76": "Plunder Quads",
|
||||
"br_77": "BR Scopes & Scatterguns",
|
||||
"br_dmz_85": "Plunder Duos",
|
||||
"dd_hc": "Demolition Hardcore",
|
||||
"dd": "Demolition",
|
||||
"br_71": "BR Solos",
|
||||
"br_74": "BR Trios",
|
||||
"br_88": "BR Duos",
|
||||
"brtdm_113": "Warzone Rumble",
|
||||
"brtdm_rmbl": "Warzone Rumble",
|
||||
"br_brsolo": "BR Solos",
|
||||
"br_brduos": "BR Duos",
|
||||
"br_brtrios": "BR Trios",
|
||||
"br_brquads": "BR Quads",
|
||||
"br_dmz_plnbld": "Blood Money",
|
||||
"br_br_real": "Realism Battle Royale",
|
||||
"br_86": "Realism Battle Royale",
|
||||
"br_brthquad": "BR 200 Quads",
|
||||
"br_jugg_brtriojugr": "Juggernaut Royal Trios",
|
||||
"br_dmz_plunquad": "Plunder Quads",
|
||||
"br_dmz_bldmnytrio": "Blood Money Trios",
|
||||
"br_mini_miniroyale": "Mini Royale",
|
||||
"br_brbbsolo": "BR Buyback Solos",
|
||||
"br_jugg_brquadjugr": "Juggernaut Royal Quads",
|
||||
"br_kingslayer_kingsltrios": "King Slayer Trios",
|
||||
"br_truckwar_trwarsquads": "Armored Royale Quads",
|
||||
"br_zxp_zmbroy": "Zombie Royale",
|
||||
"br_brhwntrios": "BR Trick-Or-Trios",
|
||||
"rugby": "Onslaughter",
|
||||
"br_brsolohwn": "BR Solo Survivor",
|
||||
"br_dmz_plndcndy": "Plunder: Candy Collector",
|
||||
"br_jugg_jugpmpkn": "Juggourdnaut Royale",
|
||||
"br_rebirth_rbrthtrios": "Resurgence Trio",
|
||||
"br_rebirth_rbrthduos": "Resurgence Duos",
|
||||
"br_rebirth_rbrthquad": "Rebirth Resurgance Quads",
|
||||
"br_dmz_plndtrios": "Plunder Trios",
|
||||
"br_rebirth_resurgence_trios": "Verdansk Resurgence Trios",
|
||||
"br_mini_rebirth_mini_royale_quads": "Rebirth Mini Royale Quads",
|
||||
"br_bodycount_pwergrb": "Power Grab",
|
||||
"br_rebirth_resurgence_mini": "Verdansk Resurgence Mini",
|
||||
"br_payload_payload": "Payload",
|
||||
"br_mini_rebirth_mini_royale_trios": "Rebirth Mini Royale Trios",
|
||||
"br_x2_br_reveal_x2_event/event_title_x2": "Battle of Verdansk",
|
||||
"br_rumble_clash": "Clash",
|
||||
"br_dbd_dbd": "Iron Trials '84",
|
||||
"br_gxp_gov": "Ghosts of Verdansk",
|
||||
"scorestreak": "Scorestreak",
|
||||
"equipment": "Equipment",
|
||||
"gear": "Gear",
|
||||
"weapon_bare_hands": "Bare Hands",
|
||||
"weapon_tactical_rifle": "Tactical Rifle",
|
||||
"weapon_shotgun": "Shotgun",
|
||||
"weapon_sniper": "Sniper",
|
||||
"weapon_lmg": "Light Machine Guns",
|
||||
"weapon_launcher": "Launcher",
|
||||
"weapon_pistol": "Pistol",
|
||||
"weapon_smg": "Submachine Guns",
|
||||
"weapon_melee": "Melee",
|
||||
"weapon_assault_rifle": "Assault Rifle",
|
||||
"attachments": "Attachments",
|
||||
"weapons": "Weapons",
|
||||
"specialist": "Specialist",
|
||||
"weapon": "Weapon",
|
||||
"weapon_special": "Special",
|
||||
"iw8_ar_akilo47": "AK-47",
|
||||
"iw8_ar_kilo433": "Kilo-141",
|
||||
"iw8_ar_mcharlie": "M13",
|
||||
"iw8_ar_falima": "FAL",
|
||||
"iw8_ar_asierra12": "Oden",
|
||||
"iw8_sm_mpapa7": "MP7",
|
||||
"iw8_sm_augolf": "AUG",
|
||||
"iw8_sm_uzulu": "Uzi",
|
||||
"iw8_sh_romeo870": "Model 680",
|
||||
"iw8_sh_charlie725": "725",
|
||||
"iw8_sh_aalpha12": "JAK-12",
|
||||
"iw8_sh_oscar12": "Origin 12",
|
||||
"iw8_lm_pkilo": "PKM",
|
||||
"iw8_lm_mgolf34": "MG34",
|
||||
"iw8_lm_lima86": "SA87",
|
||||
"iw8_lm_dblmg": "MP Juggernaut",
|
||||
"iw8_sn_mike14": "EBR-14",
|
||||
"iw8_sn_delta": "Dragunov",
|
||||
"iw8_sn_alpha50": "AX-50",
|
||||
"iw8_sn_hdromeo": "HDR",
|
||||
"iw8_sn_sbeta": "Mk2 Carbine",
|
||||
"iw8_pi_papa320": "M19",
|
||||
"iw8_pi_cpapa": ".357",
|
||||
"iw8_la_rpapa7": "RPG-7",
|
||||
"iw8_la_juliet": "JOKR",
|
||||
"iw8_la_gromeo": "PILA",
|
||||
"iw8_la_kgolf": "Strela-P",
|
||||
"iw8_me_riotshield": "Riot Shield",
|
||||
"equip_gas_grenade": "Gas Grenade",
|
||||
"equip_snapshot_grenade": "Snapshot Grenade",
|
||||
"equip_decoy": "Decoy Grenade",
|
||||
"equip_smoke": "Smoke Grenade",
|
||||
"equip_concussion": "Stun Grenade",
|
||||
"equip_hb_sensor": "Heartbeat Sensor",
|
||||
"equip_flash": "Flash Grenade",
|
||||
"equip_adrenaline": "Stim",
|
||||
"equip_frag": "Frag Grenade",
|
||||
"equip_thermite": "Thermite",
|
||||
"equip_semtex": "Semtex",
|
||||
"equip_claymore": "Claymore",
|
||||
"equip_c4": "C4",
|
||||
"equip_at_mine": "Proximity Mine",
|
||||
"equip_throwing_knife": "Throwing Knife",
|
||||
"equip_molotov": "Molotov Cocktail",
|
||||
"iw8_knife": "Combat Knife",
|
||||
"weapon_other": "Primary Melee",
|
||||
"iw8_ar_tango21": "RAM-7",
|
||||
"iw8_ar_falpha": "FR 5.56",
|
||||
"iw8_ar_mike4": "M4A1",
|
||||
"iw8_sm_papa90": "P90",
|
||||
"iw8_sm_mpapa5": "MP5",
|
||||
"iw8_sm_beta": "PP19 Bizon",
|
||||
"iw8_sh_dpapa12": "R9-0",
|
||||
"iw8_lm_mgolf36": "Holger-26",
|
||||
"iw8_sn_kilo98": "Kar98k",
|
||||
"iw8_pi_mike1911": "1911",
|
||||
"iw8_pi_golf21": "X16",
|
||||
"iw8_pi_decho": ".50 GS",
|
||||
"weapon_marksman": "Marksman Rifles",
|
||||
"iw8_lm_kilo121": "M91",
|
||||
"iw8_ar_scharlie": "FN Scar 17",
|
||||
"iw8_ar_sierra552": "Grau 5.56",
|
||||
"iw8_sm_smgolf45": "Striker 45",
|
||||
"iw8_pi_mike9a3": "Renetti",
|
||||
"iw8_lm_mkilo3": "Bruen MK9",
|
||||
"iw8_sh_mike26": "VLK Rogue",
|
||||
"iw8_sn_crossbow": "Crossbow",
|
||||
"iw8_sn_sksierra": "SKS",
|
||||
"iw8_ar_galima": "CR-56 AMAX",
|
||||
"iw8_me_kalistick": "Kali Sticks",
|
||||
"iw8_sm_victor": "Fennec Mk9",
|
||||
"iw8_sn_xmike109": "Rytec AMR",
|
||||
"iw8_pi_mike9": "Renetti",
|
||||
"iw8_me_akimboblunt": "Kali Sticks",
|
||||
"iw8_ar_anovember94": "AN-94",
|
||||
"iw8_sm_charlie9": "ISO",
|
||||
"iw8_me_akimboblades": "Dual Kodachis",
|
||||
"iw8_lm_sierrax": "FiNN",
|
||||
"iw8_ar_valpha": "AS VAL",
|
||||
"iw8_sn_romeo700": "SP-R 208",
|
||||
"cruise_predator": "Cruise Missile",
|
||||
"manual_turret": "Shield Turret",
|
||||
"toma_strike": "Cluster Strike",
|
||||
"sentry_gun": "Sentry Gun",
|
||||
"hover_jet": "VTOL Jet",
|
||||
"precision_airstrike": "Precision Airstrike",
|
||||
"juggernaut": "Juggernaut",
|
||||
"pac_sentry": "",
|
||||
"chopper_gunner": "Chopper Gunner",
|
||||
"gunship": "Gunship",
|
||||
"white_phosphorus": "White Phosphorus",
|
||||
"nuke": "Nuke",
|
||||
"chopper_support": "Support Helo",
|
||||
"bradley": "Infantry Assault Vehicle",
|
||||
"uav": "UAV",
|
||||
"directional_uav": "Advanced UAV",
|
||||
"airdrop": "Care Package",
|
||||
"airdrop_multiple": "Emergency Airdrop",
|
||||
"radar_drone_overwatch": "Personal Radar",
|
||||
"scrambler_drone_guard": "Counter UAV",
|
||||
"super_emp_drone": "EMP Drone",
|
||||
"super_trophy": "Trophy System",
|
||||
"super_ammo_drop": "Munitions Box",
|
||||
"super_weapon_drop": "Weapon Drop",
|
||||
"super_fulton": "Cash Deposit Balloon",
|
||||
"super_armor_drop": "Armor Box",
|
||||
"super_select": "Field Upgrade Pro (Any)",
|
||||
"super_tac_insert": "Tactical Insertion",
|
||||
"super_recon_drone": "Recon Drone",
|
||||
"super_deadsilence": "Dead Silence",
|
||||
"super_supply_drop": "Loadout Drop",
|
||||
"super_tac_cover": "Deployable Cover",
|
||||
"super_support_box": "Stopping Power Rounds",
|
||||
"mp_stat": "Statistic",
|
||||
"session_start": "Session Start",
|
||||
"uno": "PC",
|
||||
"psn": "Playstation Network",
|
||||
"xbl": "Xbox Live",
|
||||
"steam": "Steam",
|
||||
"battle": "BattleNET",
|
||||
"mw": "Modern Warfare",
|
||||
"cw": "Cold War",
|
||||
"mp_cartel": "Cartel",
|
||||
"mp_tank": "Garrison",
|
||||
"mp_miami": "Miami",
|
||||
"mp_moscow": "Moscow",
|
||||
"mp_satellite": "Satellite",
|
||||
"mp_kgb": "Checkmate",
|
||||
"wz_forest": "Ruka",
|
||||
"wz_ski_slopes": "Alpine",
|
||||
"mp_nuketown6": "Nuketown '84",
|
||||
"mp_tundra": "Crossroads",
|
||||
"mp_black_sea": "Armada",
|
||||
"mp_mall": "The Pines",
|
||||
"mp_raid_rm": "Raid",
|
||||
"mp_sm_berlin_tunnel": "U-Bahn",
|
||||
"mp_sm_finance": "KGB",
|
||||
"mp_sm_game_show": "Game Show",
|
||||
"mp_sm_central": "ICBM",
|
||||
"wz_sanatorium": "Sanatorium",
|
||||
"nuketown6_holiday": "Nuketown '84 Holiday",
|
||||
"mp_express_rm": "Express",
|
||||
"mp_apocalypse": "Apocalypse",
|
||||
"mp_sm_market": "Mansion",
|
||||
"mp_miami_strike": "Miami Strike",
|
||||
"wz_golova": "Golova",
|
||||
"mp_cliffhanger": "Yamantau",
|
||||
"mp_sm_gas_station": "Diesel",
|
||||
"wz_duga": "Duga",
|
||||
"mp_village_rm": "Standoff",
|
||||
"mp_sm_amsterdam": "Amsterdam",
|
||||
"mp_dune": "Collateral",
|
||||
"mp_hijacked_rm": "Hijacked",
|
||||
"mp_paintball_rm": "Rush",
|
||||
"mp_sm_deptstore": "Showroom",
|
||||
"mp_slums_rm": "Slums",
|
||||
"mp_echelon": "Echelon",
|
||||
"mp_drivein_rm": "Drive In",
|
||||
"mp_zoo_rm": "Zoo",
|
||||
"mp_firebase": "Deprogram",
|
||||
"mp_amerika": "Amerika",
|
||||
"mp_sm_vault": "Gluboko",
|
||||
"mp_don4_pm": "Nuketown '84 Halloween"
|
||||
}
|
||||
|
@ -1,236 +1,235 @@
|
||||
{
|
||||
"_MODERN WARFARE WEAPONS_": "iw8_",
|
||||
"iw8_pi_golf21": "Pistol_X16",
|
||||
"iw8_pi_papa320": "Pistol_M19",
|
||||
"iw8_pi_decho": "Pistol_.50 GS",
|
||||
"iw8_pi_mike1911": "Pistol_1911",
|
||||
"iw8_pi_cpapa": "Pistol_.357 Magnum",
|
||||
"iw8_pi_mike9": "Pistol_Renetti",
|
||||
"iw8_pi_mike": "Pistol_Sykov",
|
||||
"iw8_sm_mpapa5": "SMG_MP5",
|
||||
"iw8_sm_beta": "SMG_PP19 Bizon",
|
||||
"iw8_sm_augolf": "SMG_AUG",
|
||||
"iw8_sm_papa90": "SMG_P90",
|
||||
"iw8_sm_mpapa7": "SMG_MP7",
|
||||
"iw8_sm_uzulu": "SMG_UZI",
|
||||
"iw8_sm_charlie9": "SMG_CX-9",
|
||||
"iw8_sm_smgolf45": "SMG_Striker 45",
|
||||
"iw8_sm_victor": "SMG_Fennec",
|
||||
"iw8_sm_secho": "SMG_ISO",
|
||||
"iw8_me_riotshield": "Special_Riot Shield",
|
||||
"iw8_knife": "Special_Knife",
|
||||
"iw8_me_akimboblades": "Special_Dual Kodachis",
|
||||
"iw8_me_akimboblunt": "Special_Kali Sticks",
|
||||
"iw8_fists": "Special_Fists",
|
||||
"iw8_knifestab": "Special_Combat Knife",
|
||||
"iw8_fists_mp_zmb": "Special_Fists",
|
||||
"iw8_ar_mike4": "AR_M4A1",
|
||||
"iw8_ar_akilo47": "AR_AK-47",
|
||||
"iw8_ar_asierra12": "AR_Oden",
|
||||
"iw8_ar_falpha": "AR_FR 5.56",
|
||||
"iw8_ar_mcharlie": "AR_M13",
|
||||
"iw8_ar_kilo433": "AR_Kilo 141",
|
||||
"iw8_ar_falima": "AR_FAL",
|
||||
"iw8_ar_scharlie": "AR_FN Scar 17",
|
||||
"iw8_ar_tango21": "AR_RAM-7",
|
||||
"iw8_ar_sierra552": "AR_Grau 5.56",
|
||||
"iw8_ar_galima": "AR_CR-56 AMAX",
|
||||
"iw8_ar_anovember94": "AR_AN-94",
|
||||
"iw8_ar_valpha": "AR_AS VAL",
|
||||
"iw8_la_rpapa7": "Launcher_RPG-7",
|
||||
"iw8_la_gromeo": "Launcher_PILA",
|
||||
"iw8_la_juliet": "Launcher_JOKR",
|
||||
"iw8_la_kgolf": "Launcher_Strela-P",
|
||||
"iw8_la_mike32": "Launcher_MGL-32 Grenade Launcher",
|
||||
"iw8_sn_mike14": "Marksman_EBR-14",
|
||||
"iw8_sn_kilo98": "Marksman_Kar98k",
|
||||
"iw8_sn_sbeta": "Marksman_MK2 Carbine",
|
||||
"iw8_sn_golf28": "Marksman_SP-R 208",
|
||||
"iw8_sn_crossbow": "Marksman_Crossbow",
|
||||
"iw8_sn_sksierra": "Marksman_SKS",
|
||||
"iw8_sn_romeo700": "Sniper_SP-R 208",
|
||||
"iw8_sn_alpha50": "Sniper_AX-50",
|
||||
"iw8_sn_delta": "Sniper_Dragunov",
|
||||
"iw8_sn_hdromeo": "Sniper_HDR",
|
||||
"iw8_sn_xmike109": "Sniper_Rytec AMR",
|
||||
"iw8_sh_dpapa12": "Shotgun_R9-0",
|
||||
"iw8_sh_oscar12": "Shotgun_Origin 12",
|
||||
"iw8_sh_charlie725": "Shotgun_725",
|
||||
"iw8_sh_romeo870": "Shotgun_Model 680",
|
||||
"iw8_sh_mike26": "Shotgun_VLK Rogue",
|
||||
"iw8_sh_aalpha12": "Shotgun_JAK-12",
|
||||
"iw8_lm_kilo121": "LMG_M91",
|
||||
"iw8_lm_pkilo": "LMG_PKM",
|
||||
"iw8_lm_lima86": "LMG_SA87",
|
||||
"iw8_lm_mgolf34": "LMG_MG34",
|
||||
"iw8_lm_mgolf36": "LMG_Holger-26",
|
||||
"iw8_lm_mkilo3": "LMG_Bruen Mk9",
|
||||
"iw8_lm_sierrax": "LMG_FiNN LMG",
|
||||
"iw8_lm_dblmg": "LMG_Minigun",
|
||||
"iw8_lm_slima": "LMG_RAAL MG",
|
||||
"_COLD WAR WEAPONS_": "iw8_t9",
|
||||
"iw8_sn_t9quickscope": "Sniper_Pelington 703",
|
||||
"iw8_sn_t9standard": "Sniper_LW3 - Tundra",
|
||||
"iw8_sn_t9powersemi": "Sniper_M82",
|
||||
"iw8_sn_t9damagesemi": "Sniper_Type 63",
|
||||
"iw8_sn_t9precisionsemi": "Sniper_DMR 14",
|
||||
"iw8_sn_t9cannon": "Sniper_ZRG 20mm",
|
||||
"iw8_sn_t9crossbow": "Sniper_Crossbow",
|
||||
"iw8_sn_t9accurate": "Sniper_Swiss K31",
|
||||
"iw8_sn_t9explosivebow": "Sniper_Explosive Tip Crossbow",
|
||||
"iw8_pi_t9burst": "Pistol_Diamatti",
|
||||
"iw8_pi_t9revolver": "Pistol_Magnum",
|
||||
"iw8_pi_t9semiauto": "Pistol_1911",
|
||||
"iw8_pi_t9fullauto": "Pistol_AMP63",
|
||||
"iw8_sm_t9standard": "SMG_MP5",
|
||||
"iw8_sm_t9handling": "SMG_Milano 821",
|
||||
"iw8_sm_t9heavy": "SMG_AK-74u",
|
||||
"iw8_sm_t9fastfire": "SMG_MAC-10",
|
||||
"iw8_sm_t9burst": "SMG_KSP 45",
|
||||
"iw8_sm_t9capacity": "SMG_Bullfrog",
|
||||
"iw8_sm_t9powerburst": "SMG_AUG",
|
||||
"iw8_sm_t9accurate": "SMG_LC10",
|
||||
"iw8_sm_t9spray": "SMG_PPsh-41",
|
||||
"iw8_sm_t9nailgun": "SMG_Nailgun",
|
||||
"iw8_ar_t9standard": "AR_XM4",
|
||||
"iw8_ar_t9damage": "AR_AK-47",
|
||||
"iw8_ar_t9accurate": "AR_Krig 6",
|
||||
"iw8_ar_t9mobility": "AR_QBZ-83",
|
||||
"iw8_ar_t9longburst": "AR_M16",
|
||||
"iw8_ar_t9fasthandling": "AR_Groza",
|
||||
"iw8_ar_t9fastfire": "AR_FFAR",
|
||||
"iw8_ar_t9slowhandling": "AR_Fara 83",
|
||||
"iw8_ar_t9slowfire": "AR_C58",
|
||||
"iw8_ar_t9soviet": "AR_Vargo-S",
|
||||
"iw8_sh_t9pump": "Shotgun_Hauer 77",
|
||||
"iw8_sh_t9semiauto": "Shotgun_Gallo SA12",
|
||||
"iw8_sh_t9fullauto": "Shotgun_Streetsweeper",
|
||||
"iw8_lm_t9accurate": "LMG_Stoner 63",
|
||||
"iw8_lm_t9slowfire": "LMG_M60",
|
||||
"iw8_lm_t9light": "LMG_RPD",
|
||||
"iw8_lm_t9fastfire": "LMG_Ameli",
|
||||
"iw8_la_t9standard": "Launcher_Cigma 2",
|
||||
"iw8_la_t9freefire": "Launcher_RPG-7",
|
||||
"iw8_la_t9launcher": "Launcher_M79",
|
||||
"iw8_me_t9sledgehammer": "Special_Sledgehammer",
|
||||
"iw8_me_t9wakizashi": "Special_Wakizashi",
|
||||
"iw8_me_t9loadout": "Special_Knife",
|
||||
"iw8_me_t9machete": "Special_Machete",
|
||||
"iw8_me_t9etool": "Special_Shovel",
|
||||
"iw8_me_t9ballisticknife": "Special_Ballistic Knife",
|
||||
"iw8_me_t9bat": "Special_Baseball Bat",
|
||||
"_VANGUARD WEAPONS_": "s4_",
|
||||
"s4_mr_moscar": "Sniper_3-Line Rifle",
|
||||
"s4_mr_kalpha98": "Sniper_Kar98k",
|
||||
"s4_mr_aromeo99": "Sniper_Type 99",
|
||||
"s4_mr_ptango41": "Sniper_Gorenko Anti-Tank Rifle",
|
||||
"s4_pi_malpha96": "Pistol_Machine Pistol",
|
||||
"s4_pi_ttango33": "Pistol_RATT",
|
||||
"s4_pi_mike1911": "Pistol_1911",
|
||||
"s4_pi_wecho": "Pistol_Top Break",
|
||||
"s4_pi_luniform08": "Pistol_Klauser",
|
||||
"s4_sm_thotel": "SMG_M1928",
|
||||
"s4_sm_stango5": "SMG_Sten",
|
||||
"s4_sm_mpapa40": "SMG_Mp-40",
|
||||
"s4_sm_ppapa41": "SMG_PPsh-41",
|
||||
"s4_sm_owhiskey": "SMG_Owen Gun",
|
||||
"s4_sm_tyankee100": "SMG_Type 100",
|
||||
"s4_sm_wecho43": "SMG_Welgun",
|
||||
"s4_sm_fromeo57": "SMG_Marco 5",
|
||||
"s4_sm_guniform45": "SMG_H4 Blixen",
|
||||
"s4_sm_aromeo43": "SMG_Armaguerra 43",
|
||||
"s4_sm_salpha26": "SMG_RA 225",
|
||||
"s4_ar_stango44": "AR_STG-44",
|
||||
"s4_ar_bromeopg": "AR_ITRA Burst",
|
||||
"s4_ar_balpha": "AR_BAR",
|
||||
"s4_ar_chotel41": "AR_NZ-41",
|
||||
"s4_ar_voscar": "AR_Volkssturmgewehr",
|
||||
"s4_ar_asierra44": "AR_AS-44",
|
||||
"s4_ar_fecho": "AR_Automaton",
|
||||
"s4_ar_hyankee44": "AR_Cooper Carbine",
|
||||
"s4_ar_kgolf40": "AR_KG M40",
|
||||
"s4_ar_promeo45": "AR_Nikita AVT",
|
||||
"s4_ar_emike1": "AR_EX1",
|
||||
"s4_sh_becho": "Shotgun_Einhorn Revolving",
|
||||
"s4_sh_bromeo5": "Shotgun_Gracey Auto",
|
||||
"s4_sh_mike97": "Shotgun_Combat Shotgun",
|
||||
"s4_sh_lindia98": "Shotgun_Double Barrel",
|
||||
"s4_mg_mgolf42": "LMG_MG42",
|
||||
"s4_mg_dpapa27": "LMG_DP27",
|
||||
"s4_mg_bromeo37": "LMG_Bren",
|
||||
"s4_mg_tyankee11": "LMG_Type11",
|
||||
"s4_mg_malpha7": "LMG_Whitley",
|
||||
"s4_mg_aalpha52": "LMG_UGM-8",
|
||||
"s4_la_m1bravo": "Launcher_M1 Bazooka",
|
||||
"s4_la_palpha": "Launcher_Panzerschreck",
|
||||
"s4_la_palpha42": "Launcher_Panzerfaust",
|
||||
"s4_la_mkilo1": "Launcher_Mk11 Launcher",
|
||||
"s4_me_rindigo": "Melee_Combat Shield",
|
||||
"s4_me_knife": "Melee_FS Fighting Knife",
|
||||
"s4_me_leiomano": "Melee_Sawtooth",
|
||||
"s4_me_katana": "Melee_Katanah",
|
||||
"s4_me_axe": "Melee_Skål Crusher",
|
||||
"s4_me_sledgehammer": "Melee_Sledgehammer",
|
||||
"s4_mr_m1golf": "Marksman_M1 Grand",
|
||||
"s4_mr_svictor40": "Marksman_SVT-40",
|
||||
"s4_mr_gecho43": "Marksman_G-43",
|
||||
"s4_mr_kalpha98": "Marksman_M1916",
|
||||
"_Modern Warfare 2_": "iw9_",
|
||||
"iw9_ar_mike4_mp": "AR_M4",
|
||||
"iw9_ar_golf3_mp": "AR_Lachman-545",
|
||||
"iw9_ar_mike4": "AR_M4 Variant",
|
||||
"iw9_ar_kilo53_mp": "AR_Lachman-556",
|
||||
"iw9_ar_schotel_mp": "AR_TAQ-V",
|
||||
"iw9_ar_akilo74": "AR_Kastov-74U",
|
||||
"iw9_ar_augolf": "AR_STB 556",
|
||||
"iw9_ar_akilo": "AR_Kastov 762",
|
||||
"iw9_ar_mike16": "AR_M16",
|
||||
"iw9_ar_scharlie": "AR_TAQ-56",
|
||||
"iw9_ar_scharlie_mp": "AR_TAQ-56",
|
||||
"iw9_ar_akilo105": "AR_Kastov 545",
|
||||
"iw9_sm_mpapa7_mp": "SMG_Vel 46",
|
||||
"iw9_sm_victor_mp": "SMG_Fennec 45",
|
||||
"iw9_sm_aviktor_mp": "SMG_Vaznev-9K",
|
||||
"iw9_sm_alpha57_mp": "SMG_FSS Hurricane",
|
||||
"iw9_sm_papa90_mp": "SMG_SMG_PDSW 528",
|
||||
"iw9_sm_beta_mp": "SMG_Minibak",
|
||||
"iw9_sm_apapa_mp": "SMG_MX9",
|
||||
"iw9_sm_mpapa5_mp": "SMG_Lachmann 764",
|
||||
"iw9_sh_mike1014": "Shotgun_Expedite 12",
|
||||
"iw9_sh_charlie725_mp": "Shotgun_Lockwood 300",
|
||||
"iw9_sh_mbravo_mp": "Shotgun_Bryson 800",
|
||||
"iw9_sh_mviktor_mp": "Shotgun_Bryson 890",
|
||||
"iw9_sn_mromeo": "Sniper_MCPR-300",
|
||||
"iw9_sn_limax": "Sniper_Signal 50",
|
||||
"iw9_sn_xmike2010": "Sniper_SP-X 80",
|
||||
"iw9_dm_sbeta_mp": "Sniper_Lockwood MK2",
|
||||
"iw9_dm_mike24_mp": "Sniper_SP-R 208",
|
||||
"iw9_pi_papa220_mp": "Pistol_P890",
|
||||
"iw9_pi_golf18": "Pistol_X13",
|
||||
"iw9_pi_decho_mp": "Pistol_.50 GS",
|
||||
"iw9_pi_golf17_mp": "Pistol_X12",
|
||||
"iw9_pi_swhiskey_mp": "Pistol_Basilisk",
|
||||
"iw9_me_climbfists": "Melee_Fists",
|
||||
"iw9_me_riotshield_mp": "Melee_Riot Shield",
|
||||
"throwingknife_mp": "Melee_Throwing Knife",
|
||||
"iw9_la_juliet_mp": "Launcher_JOKR",
|
||||
"iw9_la_gromeo_mp": "Launcher_PILA",
|
||||
"iw9_la_rpapa7_mp": "Launcher_RPG-7",
|
||||
"iw9_la_kgolf_mp": "Launcher_Strella-P",
|
||||
"iw9_lm_rkilo_mp": "LMG_RPK",
|
||||
"iw9_lm_rkilo": "LMG_RPK",
|
||||
"iw9_lm_ahotel_mp": "LMG_HCR 56",
|
||||
"iw9_lm_slima_mp": "LMG_RAAL MG",
|
||||
"iw9_lm_slima": "LMG_RAAL MG",
|
||||
"iw9_lm_ngolf7_mp": "LMG_SAKIN MG38",
|
||||
"iw9_lm_rkilo21_mp": "LMG_Rapp H",
|
||||
"iw9_lm_foxtrot": "LMG_556 Icarus",
|
||||
"iw9_br_msecho_mp": "BR_FTac Recon",
|
||||
"iw9_br_msecho": "BR_FTac Recon",
|
||||
"iw9_br_soscar14": "BR_SO-14",
|
||||
"iw9_dm_pgolf1_mp": "DMR_LM-S",
|
||||
"iw9_dm_la700": "DMR_LA-B 330",
|
||||
"iw9_dm_sa700": "DMR_SA-B 50",
|
||||
"iw9_dm_scromeo": "DMR_TAQ-M",
|
||||
"iw9_dm_mike14": "DMR_ERB-14"
|
||||
"_MODERN WARFARE WEAPONS_": "iw8_",
|
||||
"iw8_pi_golf21": "Pistol_X16",
|
||||
"iw8_pi_papa320": "Pistol_M19",
|
||||
"iw8_pi_decho": "Pistol_.50 GS",
|
||||
"iw8_pi_mike1911": "Pistol_1911",
|
||||
"iw8_pi_cpapa": "Pistol_.357 Magnum",
|
||||
"iw8_pi_mike9": "Pistol_Renetti",
|
||||
"iw8_pi_mike": "Pistol_Sykov",
|
||||
"iw8_sm_mpapa5": "SMG_MP5",
|
||||
"iw8_sm_beta": "SMG_PP19 Bizon",
|
||||
"iw8_sm_augolf": "SMG_AUG",
|
||||
"iw8_sm_papa90": "SMG_P90",
|
||||
"iw8_sm_mpapa7": "SMG_MP7",
|
||||
"iw8_sm_uzulu": "SMG_UZI",
|
||||
"iw8_sm_charlie9": "SMG_CX-9",
|
||||
"iw8_sm_smgolf45": "SMG_Striker 45",
|
||||
"iw8_sm_victor": "SMG_Fennec",
|
||||
"iw8_sm_secho": "SMG_ISO",
|
||||
"iw8_me_riotshield": "Special_Riot Shield",
|
||||
"iw8_knife": "Special_Knife",
|
||||
"iw8_me_akimboblades": "Special_Dual Kodachis",
|
||||
"iw8_me_akimboblunt": "Special_Kali Sticks",
|
||||
"iw8_fists": "Special_Fists",
|
||||
"iw8_knifestab": "Special_Combat Knife",
|
||||
"iw8_fists_mp_zmb": "Special_Fists",
|
||||
"iw8_ar_mike4": "AR_M4A1",
|
||||
"iw8_ar_akilo47": "AR_AK-47",
|
||||
"iw8_ar_asierra12": "AR_Oden",
|
||||
"iw8_ar_falpha": "AR_FR 5.56",
|
||||
"iw8_ar_mcharlie": "AR_M13",
|
||||
"iw8_ar_kilo433": "AR_Kilo 141",
|
||||
"iw8_ar_falima": "AR_FAL",
|
||||
"iw8_ar_scharlie": "AR_FN Scar 17",
|
||||
"iw8_ar_tango21": "AR_RAM-7",
|
||||
"iw8_ar_sierra552": "AR_Grau 5.56",
|
||||
"iw8_ar_galima": "AR_CR-56 AMAX",
|
||||
"iw8_ar_anovember94": "AR_AN-94",
|
||||
"iw8_ar_valpha": "AR_AS VAL",
|
||||
"iw8_la_rpapa7": "Launcher_RPG-7",
|
||||
"iw8_la_gromeo": "Launcher_PILA",
|
||||
"iw8_la_juliet": "Launcher_JOKR",
|
||||
"iw8_la_kgolf": "Launcher_Strela-P",
|
||||
"iw8_la_mike32": "Launcher_MGL-32 Grenade Launcher",
|
||||
"iw8_sn_mike14": "Marksman_EBR-14",
|
||||
"iw8_sn_kilo98": "Marksman_Kar98k",
|
||||
"iw8_sn_sbeta": "Marksman_MK2 Carbine",
|
||||
"iw8_sn_golf28": "Marksman_SP-R 208",
|
||||
"iw8_sn_crossbow": "Marksman_Crossbow",
|
||||
"iw8_sn_sksierra": "Marksman_SKS",
|
||||
"iw8_sn_romeo700": "Sniper_SP-R 208",
|
||||
"iw8_sn_alpha50": "Sniper_AX-50",
|
||||
"iw8_sn_delta": "Sniper_Dragunov",
|
||||
"iw8_sn_hdromeo": "Sniper_HDR",
|
||||
"iw8_sn_xmike109": "Sniper_Rytec AMR",
|
||||
"iw8_sh_dpapa12": "Shotgun_R9-0",
|
||||
"iw8_sh_oscar12": "Shotgun_Origin 12",
|
||||
"iw8_sh_charlie725": "Shotgun_725",
|
||||
"iw8_sh_romeo870": "Shotgun_Model 680",
|
||||
"iw8_sh_mike26": "Shotgun_VLK Rogue",
|
||||
"iw8_sh_aalpha12": "Shotgun_JAK-12",
|
||||
"iw8_lm_kilo121": "LMG_M91",
|
||||
"iw8_lm_pkilo": "LMG_PKM",
|
||||
"iw8_lm_lima86": "LMG_SA87",
|
||||
"iw8_lm_mgolf34": "LMG_MG34",
|
||||
"iw8_lm_mgolf36": "LMG_Holger-26",
|
||||
"iw8_lm_mkilo3": "LMG_Bruen Mk9",
|
||||
"iw8_lm_sierrax": "LMG_FiNN LMG",
|
||||
"iw8_lm_dblmg": "LMG_Minigun",
|
||||
"iw8_lm_slima": "LMG_RAAL MG",
|
||||
"_COLD WAR WEAPONS_": "iw8_t9",
|
||||
"iw8_sn_t9quickscope": "Sniper_Pelington 703",
|
||||
"iw8_sn_t9standard": "Sniper_LW3 - Tundra",
|
||||
"iw8_sn_t9powersemi": "Sniper_M82",
|
||||
"iw8_sn_t9damagesemi": "Sniper_Type 63",
|
||||
"iw8_sn_t9precisionsemi": "Sniper_DMR 14",
|
||||
"iw8_sn_t9cannon": "Sniper_ZRG 20mm",
|
||||
"iw8_sn_t9crossbow": "Sniper_Crossbow",
|
||||
"iw8_sn_t9accurate": "Sniper_Swiss K31",
|
||||
"iw8_sn_t9explosivebow": "Sniper_Explosive Tip Crossbow",
|
||||
"iw8_pi_t9burst": "Pistol_Diamatti",
|
||||
"iw8_pi_t9revolver": "Pistol_Magnum",
|
||||
"iw8_pi_t9semiauto": "Pistol_1911",
|
||||
"iw8_pi_t9fullauto": "Pistol_AMP63",
|
||||
"iw8_sm_t9standard": "SMG_MP5",
|
||||
"iw8_sm_t9handling": "SMG_Milano 821",
|
||||
"iw8_sm_t9heavy": "SMG_AK-74u",
|
||||
"iw8_sm_t9fastfire": "SMG_MAC-10",
|
||||
"iw8_sm_t9burst": "SMG_KSP 45",
|
||||
"iw8_sm_t9capacity": "SMG_Bullfrog",
|
||||
"iw8_sm_t9powerburst": "SMG_AUG",
|
||||
"iw8_sm_t9accurate": "SMG_LC10",
|
||||
"iw8_sm_t9spray": "SMG_PPsh-41",
|
||||
"iw8_sm_t9nailgun": "SMG_Nailgun",
|
||||
"iw8_ar_t9standard": "AR_XM4",
|
||||
"iw8_ar_t9damage": "AR_AK-47",
|
||||
"iw8_ar_t9accurate": "AR_Krig 6",
|
||||
"iw8_ar_t9mobility": "AR_QBZ-83",
|
||||
"iw8_ar_t9longburst": "AR_M16",
|
||||
"iw8_ar_t9fasthandling": "AR_Groza",
|
||||
"iw8_ar_t9fastfire": "AR_FFAR",
|
||||
"iw8_ar_t9slowhandling": "AR_Fara 83",
|
||||
"iw8_ar_t9slowfire": "AR_C58",
|
||||
"iw8_ar_t9soviet": "AR_Vargo-S",
|
||||
"iw8_sh_t9pump": "Shotgun_Hauer 77",
|
||||
"iw8_sh_t9semiauto": "Shotgun_Gallo SA12",
|
||||
"iw8_sh_t9fullauto": "Shotgun_Streetsweeper",
|
||||
"iw8_lm_t9accurate": "LMG_Stoner 63",
|
||||
"iw8_lm_t9slowfire": "LMG_M60",
|
||||
"iw8_lm_t9light": "LMG_RPD",
|
||||
"iw8_lm_t9fastfire": "LMG_Ameli",
|
||||
"iw8_la_t9standard": "Launcher_Cigma 2",
|
||||
"iw8_la_t9freefire": "Launcher_RPG-7",
|
||||
"iw8_la_t9launcher": "Launcher_M79",
|
||||
"iw8_me_t9sledgehammer": "Special_Sledgehammer",
|
||||
"iw8_me_t9wakizashi": "Special_Wakizashi",
|
||||
"iw8_me_t9loadout": "Special_Knife",
|
||||
"iw8_me_t9machete": "Special_Machete",
|
||||
"iw8_me_t9etool": "Special_Shovel",
|
||||
"iw8_me_t9ballisticknife": "Special_Ballistic Knife",
|
||||
"iw8_me_t9bat": "Special_Baseball Bat",
|
||||
"_VANGUARD WEAPONS_": "s4_",
|
||||
"s4_mr_moscar": "Sniper_3-Line Rifle",
|
||||
"s4_mr_kalpha98": "Sniper_Kar98k",
|
||||
"s4_mr_aromeo99": "Sniper_Type 99",
|
||||
"s4_mr_ptango41": "Sniper_Gorenko Anti-Tank Rifle",
|
||||
"s4_pi_malpha96": "Pistol_Machine Pistol",
|
||||
"s4_pi_ttango33": "Pistol_RATT",
|
||||
"s4_pi_mike1911": "Pistol_1911",
|
||||
"s4_pi_wecho": "Pistol_Top Break",
|
||||
"s4_pi_luniform08": "Pistol_Klauser",
|
||||
"s4_sm_thotel": "SMG_M1928",
|
||||
"s4_sm_stango5": "SMG_Sten",
|
||||
"s4_sm_mpapa40": "SMG_Mp-40",
|
||||
"s4_sm_ppapa41": "SMG_PPsh-41",
|
||||
"s4_sm_owhiskey": "SMG_Owen Gun",
|
||||
"s4_sm_tyankee100": "SMG_Type 100",
|
||||
"s4_sm_wecho43": "SMG_Welgun",
|
||||
"s4_sm_fromeo57": "SMG_Marco 5",
|
||||
"s4_sm_guniform45": "SMG_H4 Blixen",
|
||||
"s4_sm_aromeo43": "SMG_Armaguerra 43",
|
||||
"s4_sm_salpha26": "SMG_RA 225",
|
||||
"s4_ar_stango44": "AR_STG-44",
|
||||
"s4_ar_bromeopg": "AR_ITRA Burst",
|
||||
"s4_ar_balpha": "AR_BAR",
|
||||
"s4_ar_chotel41": "AR_NZ-41",
|
||||
"s4_ar_voscar": "AR_Volkssturmgewehr",
|
||||
"s4_ar_asierra44": "AR_AS-44",
|
||||
"s4_ar_fecho": "AR_Automaton",
|
||||
"s4_ar_hyankee44": "AR_Cooper Carbine",
|
||||
"s4_ar_kgolf40": "AR_KG M40",
|
||||
"s4_ar_promeo45": "AR_Nikita AVT",
|
||||
"s4_ar_emike1": "AR_EX1",
|
||||
"s4_sh_becho": "Shotgun_Einhorn Revolving",
|
||||
"s4_sh_bromeo5": "Shotgun_Gracey Auto",
|
||||
"s4_sh_mike97": "Shotgun_Combat Shotgun",
|
||||
"s4_sh_lindia98": "Shotgun_Double Barrel",
|
||||
"s4_mg_mgolf42": "LMG_MG42",
|
||||
"s4_mg_dpapa27": "LMG_DP27",
|
||||
"s4_mg_bromeo37": "LMG_Bren",
|
||||
"s4_mg_tyankee11": "LMG_Type11",
|
||||
"s4_mg_malpha7": "LMG_Whitley",
|
||||
"s4_mg_aalpha52": "LMG_UGM-8",
|
||||
"s4_la_m1bravo": "Launcher_M1 Bazooka",
|
||||
"s4_la_palpha": "Launcher_Panzerschreck",
|
||||
"s4_la_palpha42": "Launcher_Panzerfaust",
|
||||
"s4_la_mkilo1": "Launcher_Mk11 Launcher",
|
||||
"s4_me_rindigo": "Melee_Combat Shield",
|
||||
"s4_me_knife": "Melee_FS Fighting Knife",
|
||||
"s4_me_leiomano": "Melee_Sawtooth",
|
||||
"s4_me_katana": "Melee_Katanah",
|
||||
"s4_me_axe": "Melee_Skål Crusher",
|
||||
"s4_me_sledgehammer": "Melee_Sledgehammer",
|
||||
"s4_mr_m1golf": "Marksman_M1 Grand",
|
||||
"s4_mr_svictor40": "Marksman_SVT-40",
|
||||
"s4_mr_gecho43": "Marksman_G-43",
|
||||
"_Modern Warfare 2_": "iw9_",
|
||||
"iw9_ar_mike4_mp": "AR_M4",
|
||||
"iw9_ar_golf3_mp": "AR_Lachman-545",
|
||||
"iw9_ar_mike4": "AR_M4 Variant",
|
||||
"iw9_ar_kilo53_mp": "AR_Lachman-556",
|
||||
"iw9_ar_schotel_mp": "AR_TAQ-V",
|
||||
"iw9_ar_akilo74": "AR_Kastov-74U",
|
||||
"iw9_ar_augolf": "AR_STB 556",
|
||||
"iw9_ar_akilo": "AR_Kastov 762",
|
||||
"iw9_ar_mike16": "AR_M16",
|
||||
"iw9_ar_scharlie": "AR_TAQ-56",
|
||||
"iw9_ar_scharlie_mp": "AR_TAQ-56",
|
||||
"iw9_ar_akilo105": "AR_Kastov 545",
|
||||
"iw9_sm_mpapa7_mp": "SMG_Vel 46",
|
||||
"iw9_sm_victor_mp": "SMG_Fennec 45",
|
||||
"iw9_sm_aviktor_mp": "SMG_Vaznev-9K",
|
||||
"iw9_sm_alpha57_mp": "SMG_FSS Hurricane",
|
||||
"iw9_sm_papa90_mp": "SMG_SMG_PDSW 528",
|
||||
"iw9_sm_beta_mp": "SMG_Minibak",
|
||||
"iw9_sm_apapa_mp": "SMG_MX9",
|
||||
"iw9_sm_mpapa5_mp": "SMG_Lachmann 764",
|
||||
"iw9_sh_mike1014": "Shotgun_Expedite 12",
|
||||
"iw9_sh_charlie725_mp": "Shotgun_Lockwood 300",
|
||||
"iw9_sh_mbravo_mp": "Shotgun_Bryson 800",
|
||||
"iw9_sh_mviktor_mp": "Shotgun_Bryson 890",
|
||||
"iw9_sn_mromeo": "Sniper_MCPR-300",
|
||||
"iw9_sn_limax": "Sniper_Signal 50",
|
||||
"iw9_sn_xmike2010": "Sniper_SP-X 80",
|
||||
"iw9_dm_sbeta_mp": "Sniper_Lockwood MK2",
|
||||
"iw9_dm_mike24_mp": "Sniper_SP-R 208",
|
||||
"iw9_pi_papa220_mp": "Pistol_P890",
|
||||
"iw9_pi_golf18": "Pistol_X13",
|
||||
"iw9_pi_decho_mp": "Pistol_.50 GS",
|
||||
"iw9_pi_golf17_mp": "Pistol_X12",
|
||||
"iw9_pi_swhiskey_mp": "Pistol_Basilisk",
|
||||
"iw9_me_climbfists": "Melee_Fists",
|
||||
"iw9_me_riotshield_mp": "Melee_Riot Shield",
|
||||
"throwingknife_mp": "Melee_Throwing Knife",
|
||||
"iw9_la_juliet_mp": "Launcher_JOKR",
|
||||
"iw9_la_gromeo_mp": "Launcher_PILA",
|
||||
"iw9_la_rpapa7_mp": "Launcher_RPG-7",
|
||||
"iw9_la_kgolf_mp": "Launcher_Strella-P",
|
||||
"iw9_lm_rkilo_mp": "LMG_RPK",
|
||||
"iw9_lm_rkilo": "LMG_RPK",
|
||||
"iw9_lm_ahotel_mp": "LMG_HCR 56",
|
||||
"iw9_lm_slima_mp": "LMG_RAAL MG",
|
||||
"iw9_lm_slima": "LMG_RAAL MG",
|
||||
"iw9_lm_ngolf7_mp": "LMG_SAKIN MG38",
|
||||
"iw9_lm_rkilo21_mp": "LMG_Rapp H",
|
||||
"iw9_lm_foxtrot": "LMG_556 Icarus",
|
||||
"iw9_br_msecho_mp": "BR_FTac Recon",
|
||||
"iw9_br_msecho": "BR_FTac Recon",
|
||||
"iw9_br_soscar14": "BR_SO-14",
|
||||
"iw9_dm_pgolf1_mp": "DMR_LM-S",
|
||||
"iw9_dm_la700": "DMR_LA-B 330",
|
||||
"iw9_dm_sa700": "DMR_SA-B 50",
|
||||
"iw9_dm_scromeo": "DMR_TAQ-M",
|
||||
"iw9_dm_mike14": "DMR_ERB-14"
|
||||
}
|
||||
|
933
src/index.html
933
src/index.html
@ -1,429 +1,536 @@
|
||||
<!DOCTYPE html>
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="language" content="EN" />
|
||||
<meta name="robots" content="index, follow" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Call of Duty Stats Tracker</title>
|
||||
<meta
|
||||
property="og:title"
|
||||
content="#1 Open Source Call of Duty Stat Tracker" />
|
||||
<meta property="og:url" content="https://codtracker.rimmyscorner.com" />
|
||||
<meta name="application-name" content="codtracker-js" />
|
||||
<meta name="generator" content="1.0" />
|
||||
<meta name="rating" content="General" />
|
||||
<meta name="author" content="thahrimdon" />
|
||||
<meta name="designer" content="thahrimdon" />
|
||||
<meta name="copyright" content="thahrimdon" />
|
||||
<meta name="geo.region" content="US-VA" />
|
||||
<meta name="geo.placename" content="Ashburn" />
|
||||
<meta name="geo.position" content="39.0437;-77.4874" />
|
||||
<!-- <link rel="apple-touch-icon" sizes="180x180" href="/images/apple-touch-icon.png"> -->
|
||||
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="language" content="EN">
|
||||
<meta name="robots" content="index, follow">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Call of Duty Stats Tracker</title>
|
||||
<meta property="og:title" content="#1 Open Source Call of Duty Stat Tracker">
|
||||
<meta property="og:url" content="https://codtracker.rimmyscorner.com">
|
||||
<meta name="application-name" content="codtracker-js">
|
||||
<meta name="generator" content="1.0">
|
||||
<meta name="rating" content="General">
|
||||
<meta name="author" content="thahrimdon">
|
||||
<meta name="designer" content="thahrimdon">
|
||||
<meta name="copyright" content="thahrimdon">
|
||||
<meta name="geo.region" content="US-VA">
|
||||
<meta name="geo.placename" content="Ashburn">
|
||||
<meta name="geo.position" content="39.0437;-77.4874">
|
||||
<!-- <link rel="apple-touch-icon" sizes="180x180" href="/images/apple-touch-icon.png"> -->
|
||||
<meta
|
||||
name="description"
|
||||
content="Extremely detailed Web GUI to fetch comprehensive user, player and recent game statistics from the API. Delivers insights beyond the in-game Barracks and cod.tracker.gg" />
|
||||
<meta
|
||||
property="og:description"
|
||||
content="Open Source Call of Duty Statistic Tracker" />
|
||||
<meta
|
||||
name="keywords"
|
||||
content="HTML, CSS, JavaScript, call of duty, cod, modern warfare, modern warfare 2019, call of duty modern warfare 2019, cod mw2019, cod modern warfare 2019, cold war, call of duty cold war, cod cw, cod cold war, vanguard, call of duty vanguard, cod vg, cod vanguard, mw2, mwii, call of duty mwii, call of duty modern warfare ii, cod mwii, cod modern warfare 2, mw3, call of duty mwiii, call of duty modern warfare iii, cod mwiii, cod modern warfare 3, mw2019, git, rimmyscorner, rimmy, ahrimdon, thahrimdon, gitea, Rimmys Corner, Rimmy’s Corner" />
|
||||
<meta
|
||||
name="twitter:title"
|
||||
content="#1 Open Source Call of Duty Stat Tracker" />
|
||||
<meta
|
||||
name="twitter:description"
|
||||
content="Extremely detailed Web GUI to fetch comprehensive user, player and recent game statistics from the API. Delivers insights beyond the in-game Barracks and cod.tracker.gg" />
|
||||
|
||||
<meta name="description" content="Extremely detailed Web GUI to fetch comprehensive user, player and recent game statistics from the API. Delivers insights beyond the in-game Barracks and cod.tracker.gg">
|
||||
<meta property="og:description"
|
||||
content="Open Source Call of Duty Statistic Tracker">
|
||||
<meta name="keywords"
|
||||
content="HTML, CSS, JavaScript, call of duty, cod, modern warfare, modern warfare 2019, call of duty modern warfare 2019, cod mw2019, cod modern warfare 2019, cold war, call of duty cold war, cod cw, cod cold war, vanguard, call of duty vanguard, cod vg, cod vanguard, mw2, mwii, call of duty mwii, call of duty modern warfare ii, cod mwii, cod modern warfare 2, mw3, call of duty mwiii, call of duty modern warfare iii, cod mwiii, cod modern warfare 3, mw2019, git, rimmyscorner, rimmy, ahrimdon, thahrimdon, gitea, Rimmys Corner, Rimmy’s Corner">
|
||||
<meta name="twitter:title" content="#1 Open Source Call of Duty Stat Tracker">
|
||||
<meta name="twitter:description"
|
||||
content="Extremely detailed Web GUI to fetch comprehensive user, player and recent game statistics from the API. Delivers insights beyond the in-game Barracks and cod.tracker.gg">
|
||||
<link rel="stylesheet" type="text/css" href="./src/css/styles.css" />
|
||||
<link rel="icon" type="image/x-icon" href="./src/images/favicon.ico" />
|
||||
<script src="./src/js/backend.js" defer></script>
|
||||
<script src="./src/js/frontend.js" defer></script>
|
||||
<script src="./src/js/localStorage.js" defer></script>
|
||||
</head>
|
||||
|
||||
<link rel="stylesheet" type="text/css" href="./src/css/styles.css">
|
||||
<link rel="icon" type="image/x-icon" href="./src/images/favicon.ico">
|
||||
<script src="./src/js/backend.js" defer></script>
|
||||
<script src="./src/js/frontend.js" defer></script>
|
||||
<script src="./src/js/localStorage.js" defer></script>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="container">
|
||||
<h1>Call of Duty Stats Tracker</h1>
|
||||
<div align="center" class="small-header2-text">
|
||||
<body>
|
||||
<div class="container">
|
||||
<h1>Call of Duty Stats Tracker</h1>
|
||||
<div align="center" class="small-header2-text">
|
||||
<p class="small-text">
|
||||
The #1 <a href="https://git.rimmyscorner.com/Rim/codtracker-js" target="_blank">Open Source</a> Call of Duty Statistic Tracker since cod.tracker.gg shutdown!
|
||||
<br>
|
||||
New? View some <a href="https://rimmyscorner.com/codtracker-examples" target="_blank">examples</a>
|
||||
The #1
|
||||
<a
|
||||
href="https://git.rimmyscorner.com/Rim/codtracker-js"
|
||||
target="_blank"
|
||||
>Open Source</a
|
||||
>
|
||||
Call of Duty Statistic Tracker since cod.tracker.gg shutdown!
|
||||
<br />
|
||||
New? View some
|
||||
<a href="https://rimmyscorner.com/codtracker-examples" target="_blank"
|
||||
>examples</a
|
||||
>
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div class="tabs">
|
||||
<div class="tab active" data-tab="stats">Player Stats</div>
|
||||
<div class="tab" data-tab="matches">Matches</div>
|
||||
<div class="tab" data-tab="user">User Info</div>
|
||||
<div class="tab" data-tab="other">Other</div>
|
||||
</div>
|
||||
|
||||
<!-- Output format selection -->
|
||||
<div class="format-options">
|
||||
<div class="form-group">
|
||||
<label for="outputFormat">Output Format:</label>
|
||||
<select id="outputFormat">
|
||||
<option value="json">JSON</option>
|
||||
<option value="yaml">YAML</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Processing options -->
|
||||
<div class="processing-options">
|
||||
<div class="form-group">
|
||||
<label>Processing Options:</label>
|
||||
<div class="checkbox-group">
|
||||
<div>
|
||||
<input type="checkbox" id="sanitizeOption" checked>
|
||||
<label for="sanitizeOption">Sanitize Output</label>
|
||||
</div>
|
||||
<div>
|
||||
<input type="checkbox" id="replaceKeysOption" checked>
|
||||
<label for="replaceKeysOption">Replace Keys</label>
|
||||
</div>
|
||||
</div>
|
||||
<div class="tabs">
|
||||
<div class="tab active" data-tab="stats">Player Stats</div>
|
||||
<div class="tab" data-tab="matches">Matches</div>
|
||||
<div class="tab" data-tab="user">User Info</div>
|
||||
<div class="tab" data-tab="other">Other</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Time options -->
|
||||
<div class="time-options">
|
||||
<div class="form-group">
|
||||
<label>Time Display Options:</label>
|
||||
<div class="checkbox-group">
|
||||
<div>
|
||||
<input type="checkbox" id="convertTimeOption">
|
||||
<label for="convertTimeOption">Convert Epoch Times</label>
|
||||
</div>
|
||||
</div>
|
||||
<div class="timezone-select" style="margin-top: 10px;">
|
||||
<label for="timezoneSelect">Timezone:</label>
|
||||
<select id="timezoneSelect" disabled>
|
||||
<option value="UTC">UTC</option>
|
||||
<option value="GMT-12">GMT-12</option>
|
||||
<option value="GMT-11">GMT-11</option>
|
||||
<option value="GMT-10">GMT-10 (Hawaii)</option>
|
||||
<option value="GMT-9">GMT-9 (Alaska)</option>
|
||||
<option value="GMT-8">GMT-8 (Pacific)</option>
|
||||
<option value="GMT-7">GMT-7 (Mountain)</option>
|
||||
<option value="GMT-6">GMT-6 (Central)</option>
|
||||
<option value="GMT-5">GMT-5 (Eastern)</option>
|
||||
<option value="GMT-4">GMT-4 (Atlantic)</option>
|
||||
<option value="GMT-3">GMT-3 (Brazil)</option>
|
||||
<option value="GMT-2">GMT-2</option>
|
||||
<option value="GMT-1">GMT-1</option>
|
||||
<option value="GMT+0">GMT+0 (London)</option>
|
||||
<option value="GMT+1">GMT+1 (Paris)</option>
|
||||
<option value="GMT+2">GMT+2 (Athens)</option>
|
||||
<option value="GMT+3">GMT+3 (Moscow)</option>
|
||||
<option value="GMT+4">GMT+4 (Dubai)</option>
|
||||
<option value="GMT+5">GMT+5 (Karachi)</option>
|
||||
<option value="GMT+5:30">GMT+5:30 (India)</option>
|
||||
<option value="GMT+6">GMT+6 (Dhaka)</option>
|
||||
<option value="GMT+7">GMT+7 (Bangkok)</option>
|
||||
<option value="GMT+8">GMT+8 (Singapore)</option>
|
||||
<option value="GMT+9">GMT+9 (Tokyo)</option>
|
||||
<option value="GMT+10">GMT+10 (Sydney)</option>
|
||||
<option value="GMT+11">GMT+11</option>
|
||||
<option value="GMT+12">GMT+12 (Auckland)</option>
|
||||
<!-- Output format selection -->
|
||||
<div class="format-options">
|
||||
<div class="form-group">
|
||||
<label for="outputFormat">Output Format:</label>
|
||||
<select id="outputFormat">
|
||||
<option value="json">JSON</option>
|
||||
<option value="yaml">YAML</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Processing options -->
|
||||
<div class="processing-options">
|
||||
<div class="form-group">
|
||||
<label>Processing Options:</label>
|
||||
<div class="checkbox-group">
|
||||
<div>
|
||||
<input type="checkbox" id="sanitizeOption" checked />
|
||||
<label for="sanitizeOption">Sanitize Output</label>
|
||||
</div>
|
||||
<div>
|
||||
<input type="checkbox" id="replaceKeysOption" checked />
|
||||
<label for="replaceKeysOption">Replace Keys</label>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Time options -->
|
||||
<div class="time-options">
|
||||
<div class="form-group">
|
||||
<label>Time Display Options:</label>
|
||||
<div class="checkbox-group">
|
||||
<div>
|
||||
<input type="checkbox" id="convertTimeOption" />
|
||||
<label for="convertTimeOption">Convert Epoch Times</label>
|
||||
</div>
|
||||
</div>
|
||||
<div class="timezone-select" style="margin-top: 10px">
|
||||
<label for="timezoneSelect">Timezone:</label>
|
||||
<select id="timezoneSelect" disabled>
|
||||
<option value="UTC">UTC</option>
|
||||
<option value="GMT-12">GMT-12</option>
|
||||
<option value="GMT-11">GMT-11</option>
|
||||
<option value="GMT-10">GMT-10 (Hawaii)</option>
|
||||
<option value="GMT-9">GMT-9 (Alaska)</option>
|
||||
<option value="GMT-8">GMT-8 (Pacific)</option>
|
||||
<option value="GMT-7">GMT-7 (Mountain)</option>
|
||||
<option value="GMT-6">GMT-6 (Central)</option>
|
||||
<option value="GMT-5">GMT-5 (Eastern)</option>
|
||||
<option value="GMT-4">GMT-4 (Atlantic)</option>
|
||||
<option value="GMT-3">GMT-3 (Brazil)</option>
|
||||
<option value="GMT-2">GMT-2</option>
|
||||
<option value="GMT-1">GMT-1</option>
|
||||
<option value="GMT+0">GMT+0 (London)</option>
|
||||
<option value="GMT+1">GMT+1 (Paris)</option>
|
||||
<option value="GMT+2">GMT+2 (Athens)</option>
|
||||
<option value="GMT+3">GMT+3 (Moscow)</option>
|
||||
<option value="GMT+4">GMT+4 (Dubai)</option>
|
||||
<option value="GMT+5">GMT+5 (Karachi)</option>
|
||||
<option value="GMT+5:30">GMT+5:30 (India)</option>
|
||||
<option value="GMT+6">GMT+6 (Dhaka)</option>
|
||||
<option value="GMT+7">GMT+7 (Bangkok)</option>
|
||||
<option value="GMT+8">GMT+8 (Singapore)</option>
|
||||
<option value="GMT+9">GMT+9 (Tokyo)</option>
|
||||
<option value="GMT+10">GMT+10 (Sydney)</option>
|
||||
<option value="GMT+11">GMT+11</option>
|
||||
<option value="GMT+12">GMT+12 (Auckland)</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Common fields for all tabs -->
|
||||
<div class="form-group">
|
||||
<label for="ssoToken">SSO Token:</label>
|
||||
<input
|
||||
type="password"
|
||||
id="ssoToken"
|
||||
placeholder="Enter your SSO Token" />
|
||||
</div>
|
||||
|
||||
<!-- Stats tab -->
|
||||
<div class="tab-content active" id="stats-tab">
|
||||
<div class="form-group">
|
||||
<label for="username">Username (e.g., User or User#1234567):</label>
|
||||
<input
|
||||
type="text"
|
||||
id="username"
|
||||
placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="platform">Platform:</label>
|
||||
<select id="platform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="game">Game:</label>
|
||||
<select id="game">
|
||||
<option value="mw">Modern Warfare / Warzone</option>
|
||||
<option value="mw2">Modern Warfare 2</option>
|
||||
<option value="wz2">Warzone 2</option>
|
||||
<option value="mw3">Modern Warfare 3</option>
|
||||
<option value="cw">Cold War</option>
|
||||
<option value="vg">Vanguard</option>
|
||||
<option value="wzm">Warzone Mobile</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="apiCall">API Call:</label>
|
||||
<select id="apiCall">
|
||||
<option value="fullData">Lifetime Statistics</option>
|
||||
<option value="combatHistory">Recent Match History</option>
|
||||
<option value="mapList">Map List</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fetchStats">Fetch Stats</button>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>
|
||||
Scroll down to the <b>Game Data and Profile Privacy</b> section
|
||||
</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Matches tab -->
|
||||
<div class="tab-content" id="matches-tab">
|
||||
<div class="form-group">
|
||||
<label for="matchUsername"
|
||||
>Username (e.g., User or User#1234567):</label
|
||||
>
|
||||
<input
|
||||
type="text"
|
||||
id="matchUsername"
|
||||
placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchPlatform">Platform:</label>
|
||||
<select id="matchPlatform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchGame">Game:</label>
|
||||
<select id="matchGame">
|
||||
<option value="mw">Modern Warfare / Warzone</option>
|
||||
<option value="mw2">Modern Warfare 2</option>
|
||||
<option value="wz2">Warzone 2</option>
|
||||
<option value="mw3">Modern Warfare 3</option>
|
||||
<option value="cw">Cold War</option>
|
||||
<option value="vg">Vanguard</option>
|
||||
<option value="wzm">Warzone Mobile</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchId">Match ID:</label>
|
||||
<input
|
||||
type="text"
|
||||
id="matchId"
|
||||
placeholder="Enter Match ID (Required for Match Info)" />
|
||||
</div>
|
||||
|
||||
<div class="button-group">
|
||||
<button id="fetchMatches">Fetch Recent Matches</button>
|
||||
<button id="fetchMatchInfo">Fetch Match Details</button>
|
||||
</div>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>
|
||||
Scroll down to the <b>Game Data and Profile Privacy</b> section
|
||||
</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- User tab -->
|
||||
<div class="tab-content" id="user-tab">
|
||||
<div class="form-group">
|
||||
<label for="userUsername"
|
||||
>Username (e.g., User or User#1234567):</label
|
||||
>
|
||||
<input
|
||||
type="text"
|
||||
id="userUsername"
|
||||
placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="userPlatform">Platform:</label>
|
||||
<select id="userPlatform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="userCall">User Info:</label>
|
||||
<select id="userCall">
|
||||
<option value="codPoints">COD Points</option>
|
||||
<option value="connectedAccounts">Connected Accounts</option>
|
||||
<option value="eventFeed">Event Feed (Logged In User Only)</option>
|
||||
<option value="friendFeed">
|
||||
Friend Feed (Logged In User Only)
|
||||
</option>
|
||||
<option value="identities">Identities (Logged In User Only)</option>
|
||||
<option value="friendsList">
|
||||
Friends List (Logged In User Only)
|
||||
</option>
|
||||
<!-- <option value="settings">Settings</option> -->
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fetchUserInfo">Fetch User Info</button>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>
|
||||
Scroll down to the <b>Game Data and Profile Privacy</b> section
|
||||
</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Search tab -->
|
||||
<div class="tab-content" id="other-tab">
|
||||
<div class="form-group">
|
||||
<label for="searchUsername"
|
||||
>Username to Search (e.g., User or User#1234567):</label
|
||||
>
|
||||
<input
|
||||
type="text"
|
||||
id="searchUsername"
|
||||
placeholder="Enter username to search" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="searchPlatform">Platform:</label>
|
||||
<select id="searchPlatform">
|
||||
<option value="all">All Platforms</option>
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fuzzySearch">Fuzzy Search</button>
|
||||
|
||||
<p class="small-text">
|
||||
Note: Fuzzy search looks up a gamertag and retrieves fuzzy matches
|
||||
along with their respective platforms.
|
||||
</p>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>
|
||||
Log in to
|
||||
<a href="https://profile.callofduty.com" target="_blank"
|
||||
>Call of Duty</a
|
||||
>
|
||||
</li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>
|
||||
Scroll down to the <b>Game Data and Profile Privacy</b> section
|
||||
</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="error" class="error"></div>
|
||||
<div id="loading" class="loading">Loading data...</div>
|
||||
<div id="download-container" style="display: none; margin-top: 10px">
|
||||
<button id="downloadJson" class="download-btn">
|
||||
Download JSON Data
|
||||
</button>
|
||||
</div>
|
||||
<pre id="results"></pre>
|
||||
</div>
|
||||
|
||||
<!-- Common fields for all tabs -->
|
||||
<div class="form-group">
|
||||
<label for="ssoToken">SSO Token:</label>
|
||||
<input type="password" id="ssoToken" placeholder="Enter your SSO Token" />
|
||||
</div>
|
||||
|
||||
<!-- Stats tab -->
|
||||
<div class="tab-content active" id="stats-tab">
|
||||
<div class="form-group">
|
||||
<label for="username">Username (e.g., Ahrimdon or Ahrimdon#1234567):</label>
|
||||
<input type="text" id="username" placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="platform">Platform:</label>
|
||||
<select id="platform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="game">Game:</label>
|
||||
<select id="game">
|
||||
<option value="mw">Modern Warfare / Warzone</option>
|
||||
<option value="mw2">Modern Warfare 2</option>
|
||||
<option value="wz2">Warzone 2</option>
|
||||
<option value="mw3">Modern Warfare 3</option>
|
||||
<option value="cw">Cold War</option>
|
||||
<option value="vg">Vanguard</option>
|
||||
<option value="wzm">Warzone Mobile</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="apiCall">API Call:</label>
|
||||
<select id="apiCall">
|
||||
<option value="fullData">Lifetime Statistics</option>
|
||||
<option value="combatHistory">Recent Match History</option>
|
||||
<option value="mapList">Map List</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fetchStats">Fetch Stats</button>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>Scroll down to the <b>Game Data and Profile Privacy</b> section</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<!-- Matches tab -->
|
||||
<div class="tab-content" id="matches-tab">
|
||||
<div class="form-group">
|
||||
<label for="matchUsername">Username (e.g., Ahrimdon or Ahrimdon#1234567):</label>
|
||||
<input type="text" id="matchUsername" placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchPlatform">Platform:</label>
|
||||
<select id="matchPlatform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchGame">Game:</label>
|
||||
<select id="matchGame">
|
||||
<option value="mw">Modern Warfare / Warzone</option>
|
||||
<option value="mw2">Modern Warfare 2</option>
|
||||
<option value="wz2">Warzone 2</option>
|
||||
<option value="mw3">Modern Warfare 3</option>
|
||||
<option value="cw">Cold War</option>
|
||||
<option value="vg">Vanguard</option>
|
||||
<option value="wzm">Warzone Mobile</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="matchId">Match ID:</label>
|
||||
<input type="text" id="matchId" placeholder="Enter Match ID (Required for Match Info)" />
|
||||
</div>
|
||||
|
||||
<div class="button-group">
|
||||
<button id="fetchMatches">Fetch Recent Matches</button>
|
||||
<button id="fetchMatchInfo">Fetch Match Details</button>
|
||||
</div>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>Scroll down to the <b>Game Data and Profile Privacy</b> section</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<!-- User tab -->
|
||||
<div class="tab-content" id="user-tab">
|
||||
<div class="form-group">
|
||||
<label for="userUsername">Username (e.g., Ahrimdon or Ahrimdon#1234567):</label>
|
||||
<input type="text" id="userUsername" placeholder="Enter your Call of Duty username" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="userPlatform">Platform:</label>
|
||||
<select id="userPlatform">
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="userCall">User Info:</label>
|
||||
<select id="userCall">
|
||||
<option value="codPoints">COD Points</option>
|
||||
<option value="connectedAccounts">Connected Accounts</option>
|
||||
<option value="eventFeed">Event Feed (Logged In User Only)</option>
|
||||
<option value="friendFeed">Friend Feed (Logged In User Only)</option>
|
||||
<option value="identities">Identities (Logged In User Only)</option>
|
||||
<option value="friendsList">Friends List (Logged In User Only)</option>
|
||||
<!-- <option value="settings">Settings</option> -->
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fetchUserInfo">Fetch User Info</button>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>Scroll down to the <b>Game Data and Profile Privacy</b> section</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<!-- Search tab -->
|
||||
<div class="tab-content" id="other-tab">
|
||||
<div class="form-group">
|
||||
<label for="searchUsername">Username to Search (e.g., Ahrimdon or Ahrimdon#1234567):</label>
|
||||
<input type="text" id="searchUsername" placeholder="Enter username to search" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="searchPlatform">Platform:</label>
|
||||
<select id="searchPlatform">
|
||||
<option value="all">All Platforms</option>
|
||||
<option value="acti">Activision</option>
|
||||
<option value="battle">Battle.net</option>
|
||||
<option value="psn">PlayStation</option>
|
||||
<option value="xbl">Xbox Live</option>
|
||||
<option value="steam">Steam</option>
|
||||
<option value="uno">Uno (numerical ID)</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<button id="fuzzySearch">Fuzzy Search</button>
|
||||
|
||||
<p class="small-text">
|
||||
Note: Fuzzy search looks up a gamertag and retrieves fuzzy matches
|
||||
along with their respective platforms.
|
||||
</p>
|
||||
|
||||
<div id="tutorial" class="tutorial">
|
||||
<h2>Authentication Setup</h2>
|
||||
<h3>Obtaining Your SSO Token</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Open developer tools (F12 or right-click → Inspect)</li>
|
||||
<li>
|
||||
Navigate to: <b>Application</b> → <b>Storage</b> →
|
||||
<b>Cookies</b> → <b>https://profile.callofduty.com</b>
|
||||
</li>
|
||||
<li>Copy the value of <code>ACT_SSO_COOKIE</code></li>
|
||||
<li>Paste this value into SSO Token</li>
|
||||
</ol>
|
||||
|
||||
<h3>Changing Account API Privacy Settings</h3>
|
||||
<ol>
|
||||
<li>Log in to <a href="https://profile.callofduty.com" target="_blank">Call of Duty</a></li>
|
||||
<li>Navigate to the <b>Privacy & Security</b> tab</li>
|
||||
<li>Scroll down to the <b>Game Data and Profile Privacy</b> section</li>
|
||||
<li>
|
||||
Set the following options to "<b>ALL</b>":
|
||||
<ul>
|
||||
<li><b>Game Tag Searchable</b></li>
|
||||
<li><b>Game Play Data</b></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ol>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<div id="error" class="error"></div>
|
||||
<div id="loading" class="loading">Loading data...</div>
|
||||
<div id="download-container" style="display: none; margin-top: 10px;">
|
||||
<button id="downloadJson" class="download-btn">Download JSON Data</button>
|
||||
</div>
|
||||
<pre id="results"></pre>
|
||||
</div>
|
||||
|
||||
<a href="https://git.rimmyscorner.com/Rim/codtracker-js" target="_blank" class="github-corner"
|
||||
aria-label="View source on Gitea" title="View Source Code on Gitea">
|
||||
<svg width="120" height="120" viewBox="0 0 250 250"
|
||||
style="fill:#151513; color:#fff; position: absolute; top: 0; right: 0; border: 0;">
|
||||
<path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path>
|
||||
<path
|
||||
d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2"
|
||||
fill="currentColor" style="transform-origin: 130px 106px;" class="octo-arm"></path>
|
||||
<path
|
||||
d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z"
|
||||
fill="currentColor" class="octo-body"></path>
|
||||
</svg>
|
||||
</a>
|
||||
</body>
|
||||
|
||||
<a
|
||||
href="https://git.rimmyscorner.com/Rim/codtracker-js"
|
||||
target="_blank"
|
||||
class="github-corner"
|
||||
aria-label="View source on Gitea"
|
||||
title="View Source Code on Gitea">
|
||||
<svg
|
||||
width="120"
|
||||
height="120"
|
||||
viewBox="0 0 250 250"
|
||||
style="
|
||||
fill: #151513;
|
||||
color: #fff;
|
||||
position: absolute;
|
||||
top: 0;
|
||||
right: 0;
|
||||
border: 0;
|
||||
">
|
||||
<path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path>
|
||||
<path
|
||||
d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2"
|
||||
fill="currentColor"
|
||||
style="transform-origin: 130px 106px"
|
||||
class="octo-arm"></path>
|
||||
<path
|
||||
d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z"
|
||||
fill="currentColor"
|
||||
class="octo-body"></path>
|
||||
</svg>
|
||||
</a>
|
||||
</body>
|
||||
</html>
|
||||
|
@ -4,16 +4,16 @@ window.backendAPI = {
|
||||
jsonToYAML,
|
||||
formatDuration,
|
||||
formatEpochTime,
|
||||
processTimestamps
|
||||
processTimestamps,
|
||||
};
|
||||
|
||||
window.appState = {
|
||||
currentData: null,
|
||||
outputFormat: "json",
|
||||
tutorialDismissed: false
|
||||
outputFormat: 'json',
|
||||
tutorialDismissed: false,
|
||||
};
|
||||
|
||||
document.addEventListener("DOMContentLoaded", function() {
|
||||
document.addEventListener('DOMContentLoaded', function () {
|
||||
// Backend-specific initialization
|
||||
});
|
||||
|
||||
@ -29,7 +29,11 @@ function jsonToYAML(json) {
|
||||
|
||||
if (typeof value === 'string') {
|
||||
// Check if string needs quotes (contains special chars)
|
||||
if (/[:{}[\],&*#?|\-<>=!%@`]/.test(value) || value === '' || !isNaN(value)) {
|
||||
if (
|
||||
/[:{}[\],&*#?|\-<>=!%@`]/.test(value) ||
|
||||
value === '' ||
|
||||
!isNaN(value)
|
||||
) {
|
||||
return `"${value.replace(/"/g, '\\"')}"`;
|
||||
}
|
||||
return value;
|
||||
@ -43,7 +47,10 @@ function jsonToYAML(json) {
|
||||
if (value.length === 0) return '[]';
|
||||
let result = '';
|
||||
for (const item of value) {
|
||||
result += `\n${indent}- ${formatValue(item, indentLevel + INDENT_SIZE).trimStart()}`;
|
||||
result += `\n${indent}- ${formatValue(
|
||||
item,
|
||||
indentLevel + INDENT_SIZE
|
||||
).trimStart()}`;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
@ -72,32 +79,37 @@ function jsonToYAML(json) {
|
||||
// Common fetch function
|
||||
async function fetchData(endpoint, requestData) {
|
||||
console.log(`[CLIENT] Request to ${endpoint} at ${new Date().toISOString()}`);
|
||||
console.log(`[CLIENT] Request data: ${JSON.stringify({
|
||||
...requestData,
|
||||
ssoToken: requestData.ssoToken ? requestData.ssoToken.substring(0, 5) + '...' : 'none'
|
||||
})}`);
|
||||
console.log(
|
||||
`[CLIENT] Request data: ${JSON.stringify({
|
||||
...requestData,
|
||||
ssoToken:
|
||||
requestData.ssoToken ?
|
||||
requestData.ssoToken.substring(0, 5) + '...'
|
||||
: 'none',
|
||||
})}`
|
||||
);
|
||||
|
||||
const errorElement = document.getElementById("error");
|
||||
const loadingElement = document.getElementById("loading");
|
||||
const resultsElement = document.getElementById("results");
|
||||
const errorElement = document.getElementById('error');
|
||||
const loadingElement = document.getElementById('loading');
|
||||
const resultsElement = document.getElementById('results');
|
||||
|
||||
// Reset display
|
||||
errorElement.textContent = "";
|
||||
resultsElement.style.display = "none";
|
||||
loadingElement.style.display = "block";
|
||||
|
||||
errorElement.textContent = '';
|
||||
resultsElement.style.display = 'none';
|
||||
loadingElement.style.display = 'block';
|
||||
|
||||
// Hide tutorial if not already dismissed
|
||||
if (!window.appState.tutorialDismissed) {
|
||||
window.appState.tutorialDismissed = true;
|
||||
document.querySelectorAll(".tutorial").forEach(element => {
|
||||
element.style.display = "none";
|
||||
document.querySelectorAll('.tutorial').forEach((element) => {
|
||||
element.style.display = 'none';
|
||||
});
|
||||
}
|
||||
|
||||
// Validate request data
|
||||
if (!requestData.ssoToken) {
|
||||
window.uiAPI.displayError("SSO Token is required");
|
||||
loadingElement.style.display = "none";
|
||||
window.uiAPI.displayError('SSO Token is required');
|
||||
loadingElement.style.display = 'none';
|
||||
return;
|
||||
}
|
||||
|
||||
@ -105,28 +117,30 @@ async function fetchData(endpoint, requestData) {
|
||||
// Set up the request with a timeout
|
||||
const controller = new AbortController();
|
||||
const timeoutId = setTimeout(() => controller.abort(), 30000); // 30 second timeout
|
||||
|
||||
|
||||
const response = await fetch(endpoint, {
|
||||
method: "POST",
|
||||
method: 'POST',
|
||||
headers: {
|
||||
"Content-Type": "application/json",
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
body: JSON.stringify(requestData),
|
||||
signal: controller.signal
|
||||
signal: controller.signal,
|
||||
});
|
||||
|
||||
clearTimeout(timeoutId);
|
||||
|
||||
// Handle non-JSON responses
|
||||
const contentType = response.headers.get("content-type");
|
||||
if (!contentType || !contentType.includes("application/json")) {
|
||||
throw new Error("Server returned non-JSON response");
|
||||
const contentType = response.headers.get('content-type');
|
||||
if (!contentType || !contentType.includes('application/json')) {
|
||||
throw new Error('Server returned non-JSON response');
|
||||
}
|
||||
|
||||
const data = await response.json();
|
||||
console.log(`[CLIENT] Response received at ${new Date().toISOString()}`);
|
||||
console.log(`[CLIENT] Response status: ${response.status}`);
|
||||
console.log(`[CLIENT] Response size: ~${JSON.stringify(data).length / 1024} KB`);
|
||||
console.log(
|
||||
`[CLIENT] Response size: ~${JSON.stringify(data).length / 1024} KB`
|
||||
);
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(data.message || `Error: ${response.status}`);
|
||||
@ -134,39 +148,39 @@ async function fetchData(endpoint, requestData) {
|
||||
|
||||
if (data.error) {
|
||||
window.uiAPI.displayError(data.error);
|
||||
} else if (data.status === "error") {
|
||||
window.uiAPI.displayError(data.message || "An error occurred");
|
||||
} else if (data.status === 'error') {
|
||||
window.uiAPI.displayError(data.message || 'An error occurred');
|
||||
} else {
|
||||
window.appState.currentData = data;
|
||||
window.uiAPI.displayResults(data);
|
||||
}
|
||||
} catch (error) {
|
||||
if (error.name === 'AbortError') {
|
||||
window.uiAPI.displayError("Request timed out. Please try again.");
|
||||
window.uiAPI.displayError('Request timed out. Please try again.');
|
||||
} else {
|
||||
window.uiAPI.displayError(
|
||||
`Error: ${error.message || "An error occurred while fetching data."}`
|
||||
`Error: ${error.message || 'An error occurred while fetching data.'}`
|
||||
);
|
||||
console.error("Fetch error:", error);
|
||||
console.error('Fetch error:', error);
|
||||
}
|
||||
} finally {
|
||||
loadingElement.style.display = "none";
|
||||
loadingElement.style.display = 'none';
|
||||
}
|
||||
}
|
||||
|
||||
// Function to convert seconds to human readable duration
|
||||
function formatDuration(seconds) {
|
||||
if (!seconds || isNaN(seconds)) return seconds;
|
||||
|
||||
|
||||
// Convert to number in case it's a string
|
||||
const totalSeconds = parseFloat(seconds);
|
||||
|
||||
|
||||
// Calculate days, hours, minutes, seconds
|
||||
const days = Math.floor(totalSeconds / 86400);
|
||||
const hours = Math.floor((totalSeconds % 86400) / 3600);
|
||||
const minutes = Math.floor((totalSeconds % 3600) / 60);
|
||||
const remainingSeconds = Math.floor(totalSeconds % 60);
|
||||
|
||||
|
||||
return `${days} Days ${hours} Hours ${minutes} Minutes ${remainingSeconds} Seconds`;
|
||||
}
|
||||
|
||||
@ -179,7 +193,8 @@ function formatEpochTime(epoch, timezone) {
|
||||
if (isNaN(epochNumber)) return epoch;
|
||||
|
||||
// Convert to milliseconds if needed
|
||||
const epochMs = epochNumber.toString().length <= 10 ? epochNumber * 1000 : epochNumber;
|
||||
const epochMs =
|
||||
epochNumber.toString().length <= 10 ? epochNumber * 1000 : epochNumber;
|
||||
|
||||
// Parse the timezone offset
|
||||
let offset = 0;
|
||||
@ -201,21 +216,52 @@ function formatEpochTime(epoch, timezone) {
|
||||
}
|
||||
|
||||
// Function to recursively process timestamps and durations in the data
|
||||
function processTimestamps(data, timezone, keysToConvert = ['date', 'dateAdded', 'utcStartSeconds', 'utcEndSeconds', 'timestamp', 'startTime', 'endTime'], durationKeys = ['time', 'timePlayedTotal', 'timePlayed', 'avgLifeTime', 'duration', 'objTime']) {
|
||||
function processTimestamps(
|
||||
data,
|
||||
timezone,
|
||||
keysToConvert = [
|
||||
'date',
|
||||
'dateAdded',
|
||||
'utcStartSeconds',
|
||||
'utcEndSeconds',
|
||||
'timestamp',
|
||||
'startTime',
|
||||
'endTime',
|
||||
],
|
||||
durationKeys = [
|
||||
'time',
|
||||
'timePlayedTotal',
|
||||
'timePlayed',
|
||||
'avgLifeTime',
|
||||
'duration',
|
||||
'objTime',
|
||||
]
|
||||
) {
|
||||
if (!data || typeof data !== 'object') return data;
|
||||
|
||||
if (Array.isArray(data)) {
|
||||
return data.map(item => processTimestamps(item, timezone, keysToConvert, durationKeys));
|
||||
return data.map((item) =>
|
||||
processTimestamps(item, timezone, keysToConvert, durationKeys)
|
||||
);
|
||||
}
|
||||
|
||||
const result = {};
|
||||
for (const [key, value] of Object.entries(data)) {
|
||||
if (keysToConvert.includes(key) && typeof value === 'number') {
|
||||
result[key] = formatEpochTime(value, timezone);
|
||||
} else if (durationKeys.includes(key) && typeof value === 'number' && document.getElementById("replaceKeysOption").checked) {
|
||||
} else if (
|
||||
durationKeys.includes(key) &&
|
||||
typeof value === 'number' &&
|
||||
document.getElementById('replaceKeysOption').checked
|
||||
) {
|
||||
result[key] = formatDuration(value);
|
||||
} else if (typeof value === 'object' && value !== null) {
|
||||
result[key] = processTimestamps(value, timezone, keysToConvert, durationKeys);
|
||||
result[key] = processTimestamps(
|
||||
value,
|
||||
timezone,
|
||||
keysToConvert,
|
||||
durationKeys
|
||||
);
|
||||
} else {
|
||||
result[key] = value;
|
||||
}
|
||||
|
236
src/js/deploy.js
236
src/js/deploy.js
@ -10,30 +10,35 @@ const config = {
|
||||
rootJs: {
|
||||
src: '*.js',
|
||||
exclude: ['*.min.js', 'build*.js'],
|
||||
outputExt: '.js'
|
||||
outputExt: '.js',
|
||||
},
|
||||
js: {
|
||||
src: ['src/**/*.js', 'node_modules/**/*.js'],
|
||||
exclude: ['src/**/*.min.js', 'src/**/build*.js'],
|
||||
outputExt: '.js'
|
||||
outputExt: '.js',
|
||||
},
|
||||
css: {
|
||||
src: ['src/**/*.css', 'node_modules/**/*.css'],
|
||||
exclude: 'src/**/*.min.css',
|
||||
outputExt: '.css'
|
||||
outputExt: '.css',
|
||||
},
|
||||
json: {
|
||||
src: ['src/**/*.json', './package.json', './package-lock.json', 'node_modules/**/*.json']
|
||||
src: [
|
||||
'src/**/*.json',
|
||||
'./package.json',
|
||||
'./package-lock.json',
|
||||
'node_modules/**/*.json',
|
||||
],
|
||||
},
|
||||
html: {
|
||||
src: ['src/**/*.html', 'node_modules/**/*.html']
|
||||
src: ['src/**/*.html', 'node_modules/**/*.html'],
|
||||
},
|
||||
images: {
|
||||
src: 'src/**/*.+(png|jpg|jpeg|gif|svg|ico)'
|
||||
src: 'src/**/*.+(png|jpg|jpeg|gif|svg|ico)',
|
||||
},
|
||||
typescript: {
|
||||
src: ['src/**/*.+(ts|ts.map|d.ts)', 'node_modules/**/*.+(ts|ts.map|d.ts)']
|
||||
}
|
||||
src: ['src/**/*.+(ts|ts.map|d.ts)', 'node_modules/**/*.+(ts|ts.map|d.ts)'],
|
||||
},
|
||||
// Add all files that might contain references
|
||||
// allFiles: {
|
||||
// src: ['./*.js', 'src/**/*.js', 'src/**/*.css', 'src/**/*.json']
|
||||
@ -62,63 +67,73 @@ function cleanPublicDir() {
|
||||
function createOutputPath(file, baseDir, outputExt = null) {
|
||||
// Get relative path from the source root
|
||||
const relativePath = path.relative(baseDir, file);
|
||||
|
||||
|
||||
// Create public path with the same relative structure
|
||||
let outputPath = path.join('public', relativePath);
|
||||
|
||||
|
||||
// Apply output extension if provided
|
||||
if (outputExt) {
|
||||
outputPath = outputPath.replace(path.extname(outputPath), outputExt);
|
||||
}
|
||||
|
||||
|
||||
return outputPath;
|
||||
}
|
||||
|
||||
// Debug minification for app.js
|
||||
async function debugAppJsMinification() {
|
||||
console.log('Debugging app.js minification specifically...');
|
||||
|
||||
|
||||
if (!fs.existsSync('app.js')) {
|
||||
console.error('❌ app.js not found in root directory');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
const appJsContent = fs.readFileSync('app.js', 'utf8');
|
||||
console.log(`✓ app.js loaded successfully - file size: ${appJsContent.length} bytes`);
|
||||
console.log(
|
||||
`✓ app.js loaded successfully - file size: ${appJsContent.length} bytes`
|
||||
);
|
||||
console.log(`✓ First 100 characters: ${appJsContent.substring(0, 100)}...`);
|
||||
|
||||
|
||||
// Check for syntax errors by parsing
|
||||
try {
|
||||
const result = await terser.minify(appJsContent, {
|
||||
compress: true,
|
||||
mangle: true,
|
||||
sourceMap: false,
|
||||
toplevel: true
|
||||
toplevel: true,
|
||||
});
|
||||
|
||||
|
||||
if (result.error) {
|
||||
console.error('❌ Error during app.js parsing:', result.error);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if (!result.code || result.code.length === 0) {
|
||||
console.error('❌ Minification produced empty output');
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(`✓ app.js minified successfully - result size: ${result.code.length} bytes`);
|
||||
console.log(`✓ First 100 characters of minified: ${result.code.substring(0, 100)}...`);
|
||||
|
||||
|
||||
console.log(
|
||||
`✓ app.js minified successfully - result size: ${result.code.length} bytes`
|
||||
);
|
||||
console.log(
|
||||
`✓ First 100 characters of minified: ${result.code.substring(
|
||||
0,
|
||||
100
|
||||
)}...`
|
||||
);
|
||||
|
||||
// Write to output with explicit name
|
||||
const outputPath = path.join('public', 'app.js');
|
||||
fs.writeFileSync(outputPath, result.code);
|
||||
console.log(`✓ Written minified app.js to: ${outputPath}`);
|
||||
|
||||
|
||||
// Calculate compression ratio
|
||||
const ratio = Math.round((result.code.length / appJsContent.length) * 100);
|
||||
const ratio = Math.round(
|
||||
(result.code.length / appJsContent.length) * 100
|
||||
);
|
||||
console.log(`✓ Compression ratio: ${ratio}% (smaller is better)`);
|
||||
|
||||
} catch (err) {
|
||||
console.error('❌ Terser processing error:', err);
|
||||
}
|
||||
@ -129,48 +144,58 @@ async function debugAppJsMinification() {
|
||||
|
||||
async function appJsMinification() {
|
||||
console.log('Debugging app.js minification specifically...');
|
||||
|
||||
|
||||
if (!fs.existsSync('app.js')) {
|
||||
console.error('❌ app.js not found in root directory');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
const appJsContent = fs.readFileSync('app.js', 'utf8');
|
||||
console.log(`✓ app.js loaded successfully - file size: ${appJsContent.length} bytes`);
|
||||
console.log(
|
||||
`✓ app.js loaded successfully - file size: ${appJsContent.length} bytes`
|
||||
);
|
||||
console.log(`✓ First 100 characters: ${appJsContent.substring(0, 100)}...`);
|
||||
|
||||
|
||||
// Check for syntax errors by parsing
|
||||
try {
|
||||
const result = await terser.minify(appJsContent, {
|
||||
compress: true,
|
||||
mangle: true,
|
||||
sourceMap: false,
|
||||
toplevel: true
|
||||
toplevel: true,
|
||||
});
|
||||
|
||||
|
||||
if (result.error) {
|
||||
console.error('❌ Error during app.js parsing:', result.error);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if (!result.code || result.code.length === 0) {
|
||||
console.error('❌ Minification produced empty output');
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(`✓ app.js minified successfully - result size: ${result.code.length} bytes`);
|
||||
console.log(`✓ First 100 characters of minified: ${result.code.substring(0, 100)}...`);
|
||||
|
||||
|
||||
console.log(
|
||||
`✓ app.js minified successfully - result size: ${result.code.length} bytes`
|
||||
);
|
||||
console.log(
|
||||
`✓ First 100 characters of minified: ${result.code.substring(
|
||||
0,
|
||||
100
|
||||
)}...`
|
||||
);
|
||||
|
||||
// Write to output with explicit name
|
||||
const outputPath = path.join('public', 'app.js');
|
||||
fs.writeFileSync(outputPath, result.code);
|
||||
console.log(`✓ Written minified app.js to: ${outputPath}`);
|
||||
|
||||
|
||||
// Calculate compression ratio
|
||||
const ratio = Math.round((result.code.length / appJsContent.length) * 100);
|
||||
const ratio = Math.round(
|
||||
(result.code.length / appJsContent.length) * 100
|
||||
);
|
||||
console.log(`✓ Compression ratio: ${ratio}% (smaller is better)`);
|
||||
|
||||
} catch (err) {
|
||||
console.error('❌ Terser processing error:', err);
|
||||
}
|
||||
@ -182,46 +207,53 @@ async function appJsMinification() {
|
||||
// Minify JavaScript files
|
||||
async function minifyJS() {
|
||||
console.log('Minifying JavaScript files...');
|
||||
|
||||
|
||||
// Minify root-level JS files (like app.js)
|
||||
const rootFiles = glob.sync(config.rootJs.src, { ignore: config.rootJs.exclude });
|
||||
|
||||
const rootFiles = glob.sync(config.rootJs.src, {
|
||||
ignore: config.rootJs.exclude,
|
||||
});
|
||||
|
||||
console.log(`Found ${rootFiles.length} root JS files to process:`, rootFiles);
|
||||
|
||||
|
||||
for (const file of rootFiles) {
|
||||
// Skip already minified files
|
||||
if (file.endsWith('.min.js')) continue;
|
||||
|
||||
|
||||
try {
|
||||
console.log(`Processing ${file}...`);
|
||||
const content = fs.readFileSync(file, 'utf8');
|
||||
console.log(`- Read ${content.length} bytes`);
|
||||
|
||||
|
||||
// Special handling for app.js with more aggressive options
|
||||
const minifyOptions = file === 'app.js' ? {
|
||||
compress: {
|
||||
dead_code: true,
|
||||
drop_console: false,
|
||||
drop_debugger: true,
|
||||
keep_fargs: false,
|
||||
unused: true
|
||||
},
|
||||
mangle: true,
|
||||
toplevel: true
|
||||
} : {
|
||||
compress: true,
|
||||
mangle: true
|
||||
};
|
||||
|
||||
const minifyOptions =
|
||||
file === 'app.js' ?
|
||||
{
|
||||
compress: {
|
||||
dead_code: true,
|
||||
drop_console: false,
|
||||
drop_debugger: true,
|
||||
keep_fargs: false,
|
||||
unused: true,
|
||||
},
|
||||
mangle: true,
|
||||
toplevel: true,
|
||||
}
|
||||
: {
|
||||
compress: true,
|
||||
mangle: true,
|
||||
};
|
||||
|
||||
const result = await terser.minify(content, minifyOptions);
|
||||
|
||||
|
||||
if (result.error) {
|
||||
console.error(`- Error minifying ${file}:`, result.error);
|
||||
continue;
|
||||
}
|
||||
|
||||
console.log(`- Minified from ${content.length} to ${result.code.length} bytes`);
|
||||
|
||||
|
||||
console.log(
|
||||
`- Minified from ${content.length} to ${result.code.length} bytes`
|
||||
);
|
||||
|
||||
const outputPath = createOutputPath(file, '.', config.rootJs.outputExt);
|
||||
ensureDirectoryExistence(outputPath);
|
||||
fs.writeFileSync(outputPath, result.code);
|
||||
@ -230,21 +262,21 @@ async function minifyJS() {
|
||||
console.error(`Error minifying ${file}:`, err);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Minify JavaScript files in src directory
|
||||
const srcFiles = glob.sync(config.js.src, { ignore: config.js.exclude });
|
||||
|
||||
|
||||
for (const file of srcFiles) {
|
||||
// Skip already minified files
|
||||
if (file.endsWith('.min.js')) continue;
|
||||
|
||||
|
||||
try {
|
||||
const content = fs.readFileSync(file, 'utf8');
|
||||
const result = await terser.minify(content, {
|
||||
compress: true,
|
||||
mangle: true
|
||||
mangle: true,
|
||||
});
|
||||
|
||||
|
||||
const outputPath = createOutputPath(file, '.', config.js.outputExt);
|
||||
ensureDirectoryExistence(outputPath);
|
||||
fs.writeFileSync(outputPath, result.code);
|
||||
@ -259,14 +291,14 @@ async function minifyJS() {
|
||||
function minifyCSS() {
|
||||
console.log('Minifying CSS files...');
|
||||
const files = glob.sync(config.css.src, { ignore: config.css.exclude });
|
||||
|
||||
|
||||
for (const file of files) {
|
||||
if (file.endsWith('.min.css')) continue;
|
||||
|
||||
|
||||
try {
|
||||
const content = fs.readFileSync(file, 'utf8');
|
||||
const result = csso.minify(content);
|
||||
|
||||
|
||||
const outputPath = createOutputPath(file, '.', config.css.outputExt);
|
||||
ensureDirectoryExistence(outputPath);
|
||||
fs.writeFileSync(outputPath, result.css);
|
||||
@ -281,13 +313,13 @@ function minifyCSS() {
|
||||
function minifyJSON() {
|
||||
console.log('Minifying JSON files...');
|
||||
const files = glob.sync(config.json.src);
|
||||
|
||||
|
||||
for (const file of files) {
|
||||
try {
|
||||
const content = fs.readFileSync(file, 'utf8');
|
||||
const jsonData = JSON.parse(content);
|
||||
const minified = JSON.stringify(jsonData);
|
||||
|
||||
|
||||
const outputPath = createOutputPath(file, '.');
|
||||
ensureDirectoryExistence(outputPath);
|
||||
fs.writeFileSync(outputPath, minified);
|
||||
@ -302,11 +334,11 @@ function minifyJSON() {
|
||||
function minifyHTML() {
|
||||
console.log('Minifying HTML files...');
|
||||
const files = glob.sync(config.html.src);
|
||||
|
||||
|
||||
for (const file of files) {
|
||||
try {
|
||||
let content = fs.readFileSync(file, 'utf8');
|
||||
|
||||
|
||||
// Minify HTML
|
||||
const minified = htmlmin.minify(content, {
|
||||
collapseWhitespace: true,
|
||||
@ -315,9 +347,9 @@ function minifyHTML() {
|
||||
minifyCSS: true,
|
||||
removeRedundantAttributes: true,
|
||||
removeEmptyAttributes: true,
|
||||
removeOptionalTags: true
|
||||
removeOptionalTags: true,
|
||||
});
|
||||
|
||||
|
||||
const outputPath = createOutputPath(file, '.');
|
||||
ensureDirectoryExistence(outputPath);
|
||||
fs.writeFileSync(outputPath, minified);
|
||||
@ -332,7 +364,7 @@ function minifyHTML() {
|
||||
function copyImages() {
|
||||
console.log('Copying images...');
|
||||
const files = glob.sync(config.images.src);
|
||||
|
||||
|
||||
for (const file of files) {
|
||||
try {
|
||||
const outputPath = createOutputPath(file, '.');
|
||||
@ -349,7 +381,7 @@ function copyImages() {
|
||||
function copyTypeScriptFiles() {
|
||||
console.log('Copying TypeScript files...');
|
||||
const files = glob.sync(config.typescript.src);
|
||||
|
||||
|
||||
for (const file of files) {
|
||||
try {
|
||||
const outputPath = createOutputPath(file, '.');
|
||||
@ -366,32 +398,32 @@ function copyTypeScriptFiles() {
|
||||
/*
|
||||
function updateAllReferences() {
|
||||
console.log('Updating file references in all files...');
|
||||
|
||||
|
||||
// Get all files that might contain references
|
||||
const files = glob.sync(config.allFiles.src);
|
||||
|
||||
|
||||
// Create a list of file mappings (original to minified)
|
||||
const jsFiles = [
|
||||
...glob.sync(config.rootJs.src, { ignore: config.rootJs.exclude }),
|
||||
...glob.sync(config.js.src, { ignore: config.js.exclude })
|
||||
]
|
||||
.filter(file => !file.endsWith('.min.js') && !file.endsWith('.d.ts') && !file.endsWith('.map') && file !== 'build.js');
|
||||
|
||||
|
||||
const cssFiles = glob.sync(config.css.src, { ignore: config.css.exclude })
|
||||
.filter(file => !file.endsWith('.min.css'));
|
||||
|
||||
|
||||
for (const targetFile of files) {
|
||||
// Skip build.js
|
||||
if (targetFile === 'build.js' || targetFile.includes('build_v')) continue;
|
||||
|
||||
|
||||
// Skip files that likely don't have references or are already processed
|
||||
if (targetFile.endsWith('.min.js') || targetFile.endsWith('.d.ts') ||
|
||||
targetFile.endsWith('.map') || targetFile.endsWith('.min.css')) continue;
|
||||
|
||||
|
||||
try {
|
||||
let content = fs.readFileSync(targetFile, 'utf8');
|
||||
let modified = false;
|
||||
|
||||
|
||||
// Replace JS file references
|
||||
for (const jsFile of jsFiles) {
|
||||
// Get different forms of the path for replacement
|
||||
@ -399,13 +431,13 @@ function updateAllReferences() {
|
||||
const relativePath = './' + normalizedPath; // With leading ./
|
||||
const plainPath = normalizedPath; // Without leading ./
|
||||
const fileNameOnly = path.basename(normalizedPath); // Just the filename
|
||||
|
||||
|
||||
// Create minified versions of each path form
|
||||
const normalizedPathMin = normalizedPath.replace('.js', '.min.js');
|
||||
const relativePathMin = relativePath.replace('.js', '.min.js');
|
||||
const plainPathMin = plainPath.replace('.js', '.min.js');
|
||||
const fileNameOnlyMin = fileNameOnly.replace('.js', '.min.js');
|
||||
|
||||
|
||||
// Attempt different path styles replacements
|
||||
if (content.includes(relativePath)) {
|
||||
content = content.replace(new RegExp(escapeRegExp(relativePath), 'g'), relativePathMin);
|
||||
@ -421,19 +453,19 @@ function updateAllReferences() {
|
||||
modified = true;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Replace CSS file references
|
||||
for (const cssFile of cssFiles) {
|
||||
const normalizedPath = cssFile.replace(/\\/g, '/');
|
||||
const relativePath = './' + normalizedPath;
|
||||
const plainPath = normalizedPath;
|
||||
const fileNameOnly = path.basename(normalizedPath);
|
||||
|
||||
|
||||
const normalizedPathMin = normalizedPath.replace('.css', '.min.css');
|
||||
const relativePathMin = relativePath.replace('.css', '.min.css');
|
||||
const plainPathMin = plainPath.replace('.css', '.min.css');
|
||||
const fileNameOnlyMin = fileNameOnly.replace('.css', '.min.css');
|
||||
|
||||
|
||||
if (content.includes(relativePath)) {
|
||||
content = content.replace(new RegExp(escapeRegExp(relativePath), 'g'), relativePathMin);
|
||||
modified = true;
|
||||
@ -448,7 +480,7 @@ function updateAllReferences() {
|
||||
modified = true;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if (modified) {
|
||||
// Write updated content to the public version
|
||||
const outputPath = createOutputPath(targetFile, '.',
|
||||
@ -471,12 +503,12 @@ function escapeRegExp(string) {
|
||||
// Main function
|
||||
async function build() {
|
||||
console.log('Starting build process...');
|
||||
|
||||
|
||||
cleanPublicDir();
|
||||
|
||||
|
||||
// Run specific app.js debugging and minification
|
||||
await debugAppJsMinification();
|
||||
|
||||
|
||||
await minifyJS();
|
||||
minifyCSS();
|
||||
minifyJSON();
|
||||
@ -485,7 +517,7 @@ async function build() {
|
||||
copyImages();
|
||||
copyTypeScriptFiles();
|
||||
// updateAllReferences();
|
||||
|
||||
|
||||
console.log('Build completed successfully!');
|
||||
}
|
||||
|
||||
@ -499,14 +531,14 @@ async function debugAppJsOnly() {
|
||||
// Check if this script is being run directly
|
||||
if (require.main === module) {
|
||||
const args = process.argv.slice(2);
|
||||
|
||||
|
||||
if (args.includes('--debug-app')) {
|
||||
debugAppJsOnly().catch(err => {
|
||||
debugAppJsOnly().catch((err) => {
|
||||
console.error('Debug failed:', err);
|
||||
process.exit(1);
|
||||
});
|
||||
} else {
|
||||
build().catch(err => {
|
||||
build().catch((err) => {
|
||||
console.error('Build failed:', err);
|
||||
process.exit(1);
|
||||
});
|
||||
@ -516,5 +548,5 @@ if (require.main === module) {
|
||||
// Export functions for external usage
|
||||
module.exports = {
|
||||
build,
|
||||
debugAppJsOnly
|
||||
};
|
||||
debugAppJsOnly,
|
||||
};
|
||||
|
1135
src/js/frontend.js
1135
src/js/frontend.js
File diff suppressed because it is too large
Load Diff
224
src/js/index.d.ts
vendored
224
src/js/index.d.ts
vendored
@ -1,108 +1,162 @@
|
||||
declare enum platforms {
|
||||
All = "all",
|
||||
Activision = "acti",
|
||||
Battlenet = "battle",
|
||||
PSN = "psn",
|
||||
Steam = "steam",
|
||||
Uno = "uno",
|
||||
XBOX = "xbl",
|
||||
ios = "ios",
|
||||
NULL = "_"
|
||||
All = 'all',
|
||||
Activision = 'acti',
|
||||
Battlenet = 'battle',
|
||||
PSN = 'psn',
|
||||
Steam = 'steam',
|
||||
Uno = 'uno',
|
||||
XBOX = 'xbl',
|
||||
ios = 'ios',
|
||||
NULL = '_',
|
||||
}
|
||||
declare enum games {
|
||||
ModernWarfare = "mw",
|
||||
ModernWarfare2 = "mw2",
|
||||
Vanguard = "vg",
|
||||
ColdWar = "cw",
|
||||
NULL = "_"
|
||||
ModernWarfare = 'mw',
|
||||
ModernWarfare2 = 'mw2',
|
||||
Vanguard = 'vg',
|
||||
ColdWar = 'cw',
|
||||
NULL = '_',
|
||||
}
|
||||
declare enum friendActions {
|
||||
Invite = "invite",
|
||||
Uninvite = "uninvite",
|
||||
Remove = "remove",
|
||||
Block = "block",
|
||||
Unblock = "unblock"
|
||||
Invite = 'invite',
|
||||
Uninvite = 'uninvite',
|
||||
Remove = 'remove',
|
||||
Block = 'block',
|
||||
Unblock = 'unblock',
|
||||
}
|
||||
declare const enableDebugMode: () => boolean;
|
||||
declare const disableDebugMode: () => boolean;
|
||||
declare const login: (ssoToken: string) => boolean;
|
||||
declare const telescopeLogin: (username: string, password: string) => Promise<boolean>;
|
||||
declare const telescopeLogin: (
|
||||
username: string,
|
||||
password: string
|
||||
) => Promise<boolean>;
|
||||
declare class WZ {
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
cleanGameMode: (mode: string) => Promise<string>;
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
cleanGameMode: (mode: string) => Promise<string>;
|
||||
}
|
||||
declare class MW {
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
}
|
||||
declare class MW2 {
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
}
|
||||
declare class WZ2 {
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
}
|
||||
declare class MW3 {
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
}
|
||||
declare class WZM {
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
fullData: (unoId: string) => Promise<unknown>;
|
||||
matches: (unoId: string) => Promise<unknown>;
|
||||
matchInfo: (unoId: string, matchId: string) => Promise<unknown>;
|
||||
}
|
||||
declare class CW {
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
}
|
||||
declare class VG {
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (gamertag: string, startTime: number, endTime: number, platform: platforms) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
fullData: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistory: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
combatHistoryWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
breakdown: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
breakdownWithDate: (
|
||||
gamertag: string,
|
||||
startTime: number,
|
||||
endTime: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
seasonloot: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
mapList: (platform: platforms) => Promise<unknown>;
|
||||
matchInfo: (matchId: string, platform: platforms) => Promise<unknown>;
|
||||
}
|
||||
declare class SHOP {
|
||||
purchasableItems: (gameId: string) => Promise<unknown>;
|
||||
bundleInformation: (title: string, bundleId: string) => Promise<unknown>;
|
||||
battlePassLoot: (title: games, season: number, platform: platforms) => Promise<unknown>;
|
||||
purchasableItems: (gameId: string) => Promise<unknown>;
|
||||
bundleInformation: (title: string, bundleId: string) => Promise<unknown>;
|
||||
battlePassLoot: (
|
||||
title: games,
|
||||
season: number,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
}
|
||||
declare class USER {
|
||||
friendFeed: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
eventFeed: () => Promise<unknown>;
|
||||
loggedInIdentities: () => Promise<unknown>;
|
||||
codPoints: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
connectedAccounts: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
settings: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
friendAction: (gamertag: string, platform: platforms, action: friendActions) => Promise<unknown>;
|
||||
friendFeed: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
eventFeed: () => Promise<unknown>;
|
||||
loggedInIdentities: () => Promise<unknown>;
|
||||
codPoints: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
connectedAccounts: (
|
||||
gamertag: string,
|
||||
platform: platforms
|
||||
) => Promise<unknown>;
|
||||
settings: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
friendAction: (
|
||||
gamertag: string,
|
||||
platform: platforms,
|
||||
action: friendActions
|
||||
) => Promise<unknown>;
|
||||
}
|
||||
declare class ALT {
|
||||
search: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
cleanWeapon: (weapon: string) => Promise<string>;
|
||||
search: (gamertag: string, platform: platforms) => Promise<unknown>;
|
||||
cleanWeapon: (weapon: string) => Promise<string>;
|
||||
}
|
||||
declare const Warzone: WZ;
|
||||
declare const ModernWarfare: MW;
|
||||
@ -115,4 +169,22 @@ declare const Vanguard: VG;
|
||||
declare const Store: SHOP;
|
||||
declare const Me: USER;
|
||||
declare const Misc: ALT;
|
||||
export { login, telescopeLogin, platforms, friendActions, Warzone, ModernWarfare, ModernWarfare2, ModernWarfare3, WarzoneMobile, Warzone2, ColdWar, Vanguard, Store, Me, Misc, enableDebugMode, disableDebugMode, };
|
||||
export {
|
||||
login,
|
||||
telescopeLogin,
|
||||
platforms,
|
||||
friendActions,
|
||||
Warzone,
|
||||
ModernWarfare,
|
||||
ModernWarfare2,
|
||||
ModernWarfare3,
|
||||
WarzoneMobile,
|
||||
Warzone2,
|
||||
ColdWar,
|
||||
Vanguard,
|
||||
Store,
|
||||
Me,
|
||||
Misc,
|
||||
enableDebugMode,
|
||||
disableDebugMode,
|
||||
};
|
||||
|
1827
src/js/index.js
1827
src/js/index.js
File diff suppressed because it is too large
Load Diff
File diff suppressed because one or more lines are too long
214
src/js/index.ts
214
src/js/index.ts
@ -1,105 +1,105 @@
|
||||
import { IncomingHttpHeaders } from "http";
|
||||
import { request } from "undici";
|
||||
import weaponMappings from "../data/weapon-ids.json";
|
||||
import wzMappings from "../data/game-modes.json";
|
||||
import { IncomingHttpHeaders } from 'http';
|
||||
import { request } from 'undici';
|
||||
import weaponMappings from '../data/weapon-ids.json';
|
||||
import wzMappings from '../data/game-modes.json';
|
||||
|
||||
const userAgent: string =
|
||||
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.169 Safari/537.36";
|
||||
let baseCookie: string = "new_SiteId=cod;ACT_SSO_LOCALE=en_US;country=US;";
|
||||
let baseSsoToken: string = "";
|
||||
'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/74.0.3729.169 Safari/537.36';
|
||||
let baseCookie: string = 'new_SiteId=cod;ACT_SSO_LOCALE=en_US;country=US;';
|
||||
let baseSsoToken: string = '';
|
||||
let debugMode = false;
|
||||
|
||||
interface CustomHeaders extends IncomingHttpHeaders {
|
||||
"X-XSRF-TOKEN"?: string | undefined;
|
||||
"X-CSRF-TOKEN"?: string | undefined;
|
||||
"Atvi-Auth"?: string | undefined;
|
||||
'X-XSRF-TOKEN'?: string | undefined;
|
||||
'X-CSRF-TOKEN'?: string | undefined;
|
||||
'Atvi-Auth'?: string | undefined;
|
||||
ACT_SSO_COOKIE?: string | undefined;
|
||||
atkn?: string | undefined;
|
||||
cookie?: string | undefined;
|
||||
"content-type"?: string | undefined;
|
||||
'content-type'?: string | undefined;
|
||||
}
|
||||
|
||||
let baseHeaders: CustomHeaders = {
|
||||
"content-type": "application/json",
|
||||
'content-type': 'application/json',
|
||||
cookie: baseCookie,
|
||||
"user-agent": userAgent,
|
||||
'user-agent': userAgent,
|
||||
};
|
||||
|
||||
let baseTelescopeHeaders: CustomHeaders = {
|
||||
accept: "application/json, text/plain, */*",
|
||||
"accept-language": "en-GB,en;q=0.9,en-US;q=0.8,fr;q=0.7,nl;q=0.6,et;q=0.5",
|
||||
"cache-control": "no-cache",
|
||||
pragma: "no-cache",
|
||||
"sec-ch-ua":
|
||||
accept: 'application/json, text/plain, */*',
|
||||
'accept-language': 'en-GB,en;q=0.9,en-US;q=0.8,fr;q=0.7,nl;q=0.6,et;q=0.5',
|
||||
'cache-control': 'no-cache',
|
||||
pragma: 'no-cache',
|
||||
'sec-ch-ua':
|
||||
'"Chromium";v="118", "Microsoft Edge";v="118", "Not=A?Brand";v="99"',
|
||||
"sec-ch-ua-mobile": "?0",
|
||||
"sec-ch-ua-platform": '"Windows"',
|
||||
"sec-fetch-dest": "empty",
|
||||
"sec-fetch-mode": "cors",
|
||||
"sec-fetch-site": "same-site",
|
||||
'sec-ch-ua-mobile': '?0',
|
||||
'sec-ch-ua-platform': '"Windows"',
|
||||
'sec-fetch-dest': 'empty',
|
||||
'sec-fetch-mode': 'cors',
|
||||
'sec-fetch-site': 'same-site',
|
||||
};
|
||||
|
||||
let basePostHeaders: CustomHeaders = {
|
||||
"content-type": "text/plain",
|
||||
'content-type': 'text/plain',
|
||||
cookie: baseCookie,
|
||||
"user-agent": userAgent,
|
||||
'user-agent': userAgent,
|
||||
};
|
||||
|
||||
let baseUrl: string = "https://profile.callofduty.com";
|
||||
let apiPath: string = "/api/papi-client";
|
||||
let baseTelescopeUrl: string = "https://telescope.callofduty.com";
|
||||
let apiTelescopePath: string = "/api/ts-api";
|
||||
let baseUrl: string = 'https://profile.callofduty.com';
|
||||
let apiPath: string = '/api/papi-client';
|
||||
let baseTelescopeUrl: string = 'https://telescope.callofduty.com';
|
||||
let apiTelescopePath: string = '/api/ts-api';
|
||||
let loggedIn: boolean = false;
|
||||
|
||||
enum platforms {
|
||||
All = "all",
|
||||
Activision = "acti",
|
||||
Battlenet = "battle",
|
||||
PSN = "psn",
|
||||
Steam = "steam",
|
||||
Uno = "uno",
|
||||
XBOX = "xbl",
|
||||
ios = "ios",
|
||||
NULL = "_",
|
||||
All = 'all',
|
||||
Activision = 'acti',
|
||||
Battlenet = 'battle',
|
||||
PSN = 'psn',
|
||||
Steam = 'steam',
|
||||
Uno = 'uno',
|
||||
XBOX = 'xbl',
|
||||
ios = 'ios',
|
||||
NULL = '_',
|
||||
}
|
||||
|
||||
enum games {
|
||||
ModernWarfare = "mw",
|
||||
ModernWarfare2 = "mw2",
|
||||
Vanguard = "vg",
|
||||
ColdWar = "cw",
|
||||
NULL = "_",
|
||||
ModernWarfare = 'mw',
|
||||
ModernWarfare2 = 'mw2',
|
||||
Vanguard = 'vg',
|
||||
ColdWar = 'cw',
|
||||
NULL = '_',
|
||||
}
|
||||
|
||||
enum telescopeGames {
|
||||
ModernWarfare2 = "mw2",
|
||||
Warzone2 = "wz2",
|
||||
ModernWarfare3 = "jup",
|
||||
Mobile = "mgl",
|
||||
ModernWarfare2 = 'mw2',
|
||||
Warzone2 = 'wz2',
|
||||
ModernWarfare3 = 'jup',
|
||||
Mobile = 'mgl',
|
||||
}
|
||||
|
||||
enum modes {
|
||||
Multiplayer = "mp",
|
||||
Warzone = "wz",
|
||||
Warzone2 = "wz2",
|
||||
NULL = "_",
|
||||
Multiplayer = 'mp',
|
||||
Warzone = 'wz',
|
||||
Warzone2 = 'wz2',
|
||||
NULL = '_',
|
||||
}
|
||||
|
||||
enum telescopeModes {
|
||||
Multiplayer = "mp",
|
||||
Outbreak = "ob",
|
||||
Multiplayer = 'mp',
|
||||
Outbreak = 'ob',
|
||||
}
|
||||
|
||||
enum friendActions {
|
||||
Invite = "invite",
|
||||
Uninvite = "uninvite",
|
||||
Remove = "remove",
|
||||
Block = "block",
|
||||
Unblock = "unblock",
|
||||
Invite = 'invite',
|
||||
Uninvite = 'uninvite',
|
||||
Remove = 'remove',
|
||||
Block = 'block',
|
||||
Unblock = 'unblock',
|
||||
}
|
||||
|
||||
enum generics {
|
||||
STEAM_UNSUPPORTED = "Steam platform not supported by this game. Try `battle` instead.",
|
||||
STEAM_UNSUPPORTED = 'Steam platform not supported by this game. Try `battle` instead.',
|
||||
UNO_NO_NUMERICAL_ID = `You must use a numerical ID when using the platform 'uno'.\nIf using an Activision ID, please use the platform 'acti'.`,
|
||||
}
|
||||
|
||||
@ -121,7 +121,7 @@ interface telescopeLoginErrorResponse {
|
||||
error: telescopeLoginErrorNestedResponse;
|
||||
}
|
||||
|
||||
let telescopeUnoToken = "";
|
||||
let telescopeUnoToken = '';
|
||||
|
||||
const enableDebugMode = () => (debugMode = true);
|
||||
|
||||
@ -129,7 +129,7 @@ const disableDebugMode = () => (debugMode = false);
|
||||
|
||||
const sendTelescopeRequest = async (url: string) => {
|
||||
try {
|
||||
if (!loggedIn) throw new Error("Not Logged In!");
|
||||
if (!loggedIn) throw new Error('Not Logged In!');
|
||||
let requestUrl = `${baseTelescopeUrl}${apiTelescopePath}${url}`;
|
||||
if (debugMode) console.log(`[DEBUG]`, `Request Uri: ${requestUrl}`);
|
||||
baseTelescopeHeaders.authorization = `Bearer ${telescopeUnoToken}`;
|
||||
@ -152,17 +152,17 @@ const sendTelescopeRequest = async (url: string) => {
|
||||
|
||||
const sendRequest = async (url: string) => {
|
||||
try {
|
||||
if (!loggedIn) throw new Error("Not Logged In.");
|
||||
if (!loggedIn) throw new Error('Not Logged In.');
|
||||
let requestUrl = `${baseUrl}${apiPath}${url}`;
|
||||
|
||||
if (debugMode) console.log(`[DEBUG]`, `Request Uri: ${requestUrl}`);
|
||||
if (debugMode) console.time("Round Trip");
|
||||
if (debugMode) console.time('Round Trip');
|
||||
|
||||
const { body, statusCode } = await request(requestUrl, {
|
||||
headers: baseHeaders,
|
||||
});
|
||||
|
||||
if (debugMode) console.timeEnd("Round Trip");
|
||||
if (debugMode) console.timeEnd('Round Trip');
|
||||
|
||||
if (statusCode >= 500)
|
||||
throw new Error(
|
||||
@ -185,10 +185,10 @@ const sendRequest = async (url: string) => {
|
||||
|
||||
const sendPostRequest = async (url: string, data: string) => {
|
||||
try {
|
||||
if (!loggedIn) throw new Error("Not Logged In.");
|
||||
if (!loggedIn) throw new Error('Not Logged In.');
|
||||
let requestUrl = `${baseUrl}${apiPath}${url}`;
|
||||
const { body, statusCode } = await request(requestUrl, {
|
||||
method: "POST",
|
||||
method: 'POST',
|
||||
headers: basePostHeaders,
|
||||
body: data,
|
||||
});
|
||||
@ -212,41 +212,39 @@ const cleanClientName = (gamertag: string): string => {
|
||||
|
||||
const login = (ssoToken: string): boolean => {
|
||||
if (!ssoToken || ssoToken.trim().length <= 0) return false;
|
||||
let fakeXSRF = "68e8b62e-1d9d-4ce1-b93f-cbe5ff31a041";
|
||||
baseHeaders["X-XSRF-TOKEN"] = fakeXSRF;
|
||||
baseHeaders["X-CSRF-TOKEN"] = fakeXSRF;
|
||||
baseHeaders["Atvi-Auth"] = ssoToken;
|
||||
baseHeaders["ACT_SSO_COOKIE"] = ssoToken;
|
||||
baseHeaders["atkn"] = ssoToken;
|
||||
baseHeaders[
|
||||
"cookie"
|
||||
] = `${baseCookie}ACT_SSO_COOKIE=${ssoToken};XSRF-TOKEN=${fakeXSRF};API_CSRF_TOKEN=${fakeXSRF};ACT_SSO_EVENT="LOGIN_SUCCESS:1644346543228";ACT_SSO_COOKIE_EXPIRY=1645556143194;comid=cod;ssoDevId=63025d09c69f47dfa2b8d5520b5b73e4;tfa_enrollment_seen=true;gtm.custom.bot.flag=human;`;
|
||||
let fakeXSRF = '68e8b62e-1d9d-4ce1-b93f-cbe5ff31a041';
|
||||
baseHeaders['X-XSRF-TOKEN'] = fakeXSRF;
|
||||
baseHeaders['X-CSRF-TOKEN'] = fakeXSRF;
|
||||
baseHeaders['Atvi-Auth'] = ssoToken;
|
||||
baseHeaders['ACT_SSO_COOKIE'] = ssoToken;
|
||||
baseHeaders['atkn'] = ssoToken;
|
||||
baseHeaders['cookie'] =
|
||||
`${baseCookie}ACT_SSO_COOKIE=${ssoToken};XSRF-TOKEN=${fakeXSRF};API_CSRF_TOKEN=${fakeXSRF};ACT_SSO_EVENT="LOGIN_SUCCESS:1644346543228";ACT_SSO_COOKIE_EXPIRY=1645556143194;comid=cod;ssoDevId=63025d09c69f47dfa2b8d5520b5b73e4;tfa_enrollment_seen=true;gtm.custom.bot.flag=human;`;
|
||||
baseSsoToken = ssoToken;
|
||||
basePostHeaders["X-XSRF-TOKEN"] = fakeXSRF;
|
||||
basePostHeaders["X-CSRF-TOKEN"] = fakeXSRF;
|
||||
basePostHeaders["Atvi-Auth"] = ssoToken;
|
||||
basePostHeaders["ACT_SSO_COOKIE"] = ssoToken;
|
||||
basePostHeaders["atkn"] = ssoToken;
|
||||
basePostHeaders[
|
||||
"cookie"
|
||||
] = `${baseCookie}ACT_SSO_COOKIE=${ssoToken};XSRF-TOKEN=${fakeXSRF};API_CSRF_TOKEN=${fakeXSRF};ACT_SSO_EVENT="LOGIN_SUCCESS:1644346543228";ACT_SSO_COOKIE_EXPIRY=1645556143194;comid=cod;ssoDevId=63025d09c69f47dfa2b8d5520b5b73e4;tfa_enrollment_seen=true;gtm.custom.bot.flag=human;`;
|
||||
basePostHeaders['X-XSRF-TOKEN'] = fakeXSRF;
|
||||
basePostHeaders['X-CSRF-TOKEN'] = fakeXSRF;
|
||||
basePostHeaders['Atvi-Auth'] = ssoToken;
|
||||
basePostHeaders['ACT_SSO_COOKIE'] = ssoToken;
|
||||
basePostHeaders['atkn'] = ssoToken;
|
||||
basePostHeaders['cookie'] =
|
||||
`${baseCookie}ACT_SSO_COOKIE=${ssoToken};XSRF-TOKEN=${fakeXSRF};API_CSRF_TOKEN=${fakeXSRF};ACT_SSO_EVENT="LOGIN_SUCCESS:1644346543228";ACT_SSO_COOKIE_EXPIRY=1645556143194;comid=cod;ssoDevId=63025d09c69f47dfa2b8d5520b5b73e4;tfa_enrollment_seen=true;gtm.custom.bot.flag=human;`;
|
||||
loggedIn = true;
|
||||
return loggedIn;
|
||||
};
|
||||
|
||||
const telescope_login_endpoint =
|
||||
"https://wzm-ios-loginservice.prod.demonware.net/v1/login/uno/?titleID=7100&client=shg-cod-jup-bnet";
|
||||
'https://wzm-ios-loginservice.prod.demonware.net/v1/login/uno/?titleID=7100&client=shg-cod-jup-bnet';
|
||||
const telescopeLogin = async (
|
||||
username: string,
|
||||
password: string
|
||||
): Promise<boolean> => {
|
||||
if (!username || !password) return false;
|
||||
const { body, statusCode } = await request(telescope_login_endpoint, {
|
||||
method: "POST",
|
||||
method: 'POST',
|
||||
headers: baseHeaders,
|
||||
body: JSON.stringify({
|
||||
platform: "ios",
|
||||
hardwareType: "ios",
|
||||
platform: 'ios',
|
||||
hardwareType: 'ios',
|
||||
auth: {
|
||||
email: username,
|
||||
password: password,
|
||||
@ -263,14 +261,14 @@ const telescopeLogin = async (
|
||||
} else if (statusCode === 403) {
|
||||
let errorResponse: telescopeLoginErrorResponse =
|
||||
(await body.json()) as telescopeLoginErrorResponse;
|
||||
console.error("Error Logging In:", errorResponse.error.msg);
|
||||
console.error('Error Logging In:', errorResponse.error.msg);
|
||||
}
|
||||
loggedIn = statusCode == 200;
|
||||
return loggedIn;
|
||||
};
|
||||
|
||||
const handleLookupType = (platform: platforms) => {
|
||||
return platform === platforms.Uno ? "id" : "gamer";
|
||||
return platform === platforms.Uno ? 'id' : 'gamer';
|
||||
};
|
||||
|
||||
const checkForValidPlatform = (platform: platforms, gamertag?: string) => {
|
||||
@ -487,7 +485,7 @@ class WZ {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.ModernWarfare,
|
||||
gamertag,
|
||||
@ -500,7 +498,7 @@ class WZ {
|
||||
|
||||
cleanGameMode = async (mode: string): Promise<string> => {
|
||||
//@ts-ignore
|
||||
const foundMode: string = wzMappings["modes"][mode];
|
||||
const foundMode: string = wzMappings['modes'][mode];
|
||||
if (!foundMode) return mode;
|
||||
return foundMode;
|
||||
};
|
||||
@ -604,7 +602,7 @@ class MW {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.ModernWarfare,
|
||||
gamertag,
|
||||
@ -636,7 +634,7 @@ class MW {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.ModernWarfare,
|
||||
gamertag,
|
||||
@ -914,7 +912,7 @@ class CW {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.ColdWar,
|
||||
gamertag,
|
||||
@ -930,7 +928,7 @@ class CW {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.ColdWar,
|
||||
gamertag,
|
||||
@ -1056,7 +1054,7 @@ class VG {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.Vanguard,
|
||||
gamertag,
|
||||
@ -1072,7 +1070,7 @@ class VG {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
games.Vanguard,
|
||||
gamertag,
|
||||
@ -1088,10 +1086,10 @@ class SHOP {
|
||||
purchasableItems = async (gameId: string) => {
|
||||
const endpoint = new Endpoints(
|
||||
games.NULL,
|
||||
"",
|
||||
'',
|
||||
platforms.NULL,
|
||||
modes.NULL,
|
||||
""
|
||||
''
|
||||
);
|
||||
return await sendRequest(endpoint.purchasableItems(gameId));
|
||||
};
|
||||
@ -1099,10 +1097,10 @@ class SHOP {
|
||||
bundleInformation = async (title: string, bundleId: string) => {
|
||||
const endpoint = new Endpoints(
|
||||
games.NULL,
|
||||
"",
|
||||
'',
|
||||
platforms.NULL,
|
||||
modes.NULL,
|
||||
""
|
||||
''
|
||||
);
|
||||
return await sendRequest(endpoint.bundleInformation(title, bundleId));
|
||||
};
|
||||
@ -1116,7 +1114,7 @@ class SHOP {
|
||||
gamertag,
|
||||
_platform: platform,
|
||||
lookupType,
|
||||
} = mapGamertagToPlatform("", platform);
|
||||
} = mapGamertagToPlatform('', platform);
|
||||
const endpoint = new Endpoints(
|
||||
title,
|
||||
gamertag,
|
||||
@ -1148,10 +1146,10 @@ class USER {
|
||||
eventFeed = async () => {
|
||||
const endpoint = new Endpoints(
|
||||
games.NULL,
|
||||
"",
|
||||
'',
|
||||
platforms.NULL,
|
||||
modes.NULL,
|
||||
""
|
||||
''
|
||||
);
|
||||
return await sendRequest(endpoint.eventFeed());
|
||||
};
|
||||
@ -1159,10 +1157,10 @@ class USER {
|
||||
loggedInIdentities = async () => {
|
||||
const endpoint = new Endpoints(
|
||||
games.NULL,
|
||||
"",
|
||||
'',
|
||||
platforms.NULL,
|
||||
modes.NULL,
|
||||
""
|
||||
''
|
||||
);
|
||||
return await sendRequest(endpoint.loggedInIdentities());
|
||||
};
|
||||
@ -1232,7 +1230,7 @@ class USER {
|
||||
modes.NULL,
|
||||
lookupType
|
||||
);
|
||||
return await sendPostRequest(endpoint.friendAction(action), "{}");
|
||||
return await sendPostRequest(endpoint.friendAction(action), '{}');
|
||||
};
|
||||
}
|
||||
|
||||
@ -1255,7 +1253,7 @@ class ALT {
|
||||
|
||||
cleanWeapon = async (weapon: string): Promise<string> => {
|
||||
//@ts-ignore
|
||||
const foundWeapon: string = weaponMappings["All Weapons"][weapon];
|
||||
const foundWeapon: string = weaponMappings['All Weapons'][weapon];
|
||||
if (!foundWeapon) return weapon;
|
||||
return foundWeapon;
|
||||
};
|
||||
|
@ -1,103 +1,109 @@
|
||||
document.addEventListener("DOMContentLoaded", function () {
|
||||
// Fields to save in localStorage with their respective keys
|
||||
const fieldsToSave = [
|
||||
// Stats tab
|
||||
{ id: "username", key: "cod_username" },
|
||||
{ id: "platform", key: "cod_platform" },
|
||||
{ id: "game", key: "cod_game" },
|
||||
{ id: "apiCall", key: "cod_apiCall" },
|
||||
|
||||
// Matches tab
|
||||
{ id: "matchUsername", key: "cod_matchUsername" },
|
||||
{ id: "matchPlatform", key: "cod_matchPlatform" },
|
||||
{ id: "matchGame", key: "cod_matchGame" },
|
||||
{ id: "matchId", key: "cod_matchId" },
|
||||
|
||||
// User tab
|
||||
{ id: "userUsername", key: "cod_userUsername" },
|
||||
{ id: "userPlatform", key: "cod_userPlatform" },
|
||||
{ id: "userCall", key: "cod_userCall" },
|
||||
|
||||
// Other/Search tab
|
||||
{ id: "searchUsername", key: "cod_searchUsername" },
|
||||
{ id: "searchPlatform", key: "cod_searchPlatform" },
|
||||
|
||||
// Format and processing options
|
||||
{ id: "outputFormat", key: "cod_outputFormat" },
|
||||
{ id: "sanitizeOption", key: "cod_sanitizeOption" },
|
||||
{ id: "replaceKeysOption", key: "cod_replaceKeysOption" },
|
||||
{ id: "convertTimeOption", key: "cod_convertTimeOption" },
|
||||
{ id: "timezoneSelect", key: "cod_timezone" }
|
||||
];
|
||||
|
||||
// Load saved values
|
||||
fieldsToSave.forEach(field => {
|
||||
const element = document.getElementById(field.id);
|
||||
if (!element) return; // Skip if element doesn't exist
|
||||
|
||||
const savedValue = localStorage.getItem(field.key);
|
||||
if (savedValue !== null) {
|
||||
// Handle different input types
|
||||
if (element.type === "checkbox") {
|
||||
element.checked = savedValue === "true";
|
||||
} else if (element.tagName === "SELECT") {
|
||||
element.value = savedValue;
|
||||
} else {
|
||||
element.value = savedValue;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Save values on change
|
||||
fieldsToSave.forEach(field => {
|
||||
const element = document.getElementById(field.id);
|
||||
if (!element) return; // Skip if element doesn't exist
|
||||
|
||||
// Different event listener based on input type
|
||||
if (element.type === "checkbox") {
|
||||
element.addEventListener("change", function() {
|
||||
localStorage.setItem(field.key, element.checked);
|
||||
});
|
||||
} else if (element.tagName === "SELECT") {
|
||||
element.addEventListener("change", function() {
|
||||
localStorage.setItem(field.key, element.value);
|
||||
});
|
||||
} else {
|
||||
element.addEventListener("input", function() {
|
||||
localStorage.setItem(field.key, element.value);
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
// Special handling for SSO Token
|
||||
const ssoTokenInput = document.getElementById("ssoToken");
|
||||
const savedSsoToken = localStorage.getItem("cod_ssoToken");
|
||||
|
||||
if (savedSsoToken) {
|
||||
ssoTokenInput.value = savedSsoToken;
|
||||
}
|
||||
document.addEventListener('DOMContentLoaded', function () {
|
||||
// Fields to save in localStorage with their respective keys
|
||||
const fieldsToSave = [
|
||||
// Stats tab
|
||||
{ id: 'username', key: 'cod_username' },
|
||||
{ id: 'platform', key: 'cod_platform' },
|
||||
{ id: 'game', key: 'cod_game' },
|
||||
{ id: 'apiCall', key: 'cod_apiCall' },
|
||||
|
||||
// Ask the user before saving SSO token
|
||||
ssoTokenInput.addEventListener("input", function() {
|
||||
if (confirm("Would you like to save your SSO token? Note: This is stored on your device only.")) {
|
||||
localStorage.setItem("cod_ssoToken", ssoTokenInput.value);
|
||||
// Matches tab
|
||||
{ id: 'matchUsername', key: 'cod_matchUsername' },
|
||||
{ id: 'matchPlatform', key: 'cod_matchPlatform' },
|
||||
{ id: 'matchGame', key: 'cod_matchGame' },
|
||||
{ id: 'matchId', key: 'cod_matchId' },
|
||||
|
||||
// User tab
|
||||
{ id: 'userUsername', key: 'cod_userUsername' },
|
||||
{ id: 'userPlatform', key: 'cod_userPlatform' },
|
||||
{ id: 'userCall', key: 'cod_userCall' },
|
||||
|
||||
// Other/Search tab
|
||||
{ id: 'searchUsername', key: 'cod_searchUsername' },
|
||||
{ id: 'searchPlatform', key: 'cod_searchPlatform' },
|
||||
|
||||
// Format and processing options
|
||||
{ id: 'outputFormat', key: 'cod_outputFormat' },
|
||||
{ id: 'sanitizeOption', key: 'cod_sanitizeOption' },
|
||||
{ id: 'replaceKeysOption', key: 'cod_replaceKeysOption' },
|
||||
{ id: 'convertTimeOption', key: 'cod_convertTimeOption' },
|
||||
{ id: 'timezoneSelect', key: 'cod_timezone' },
|
||||
];
|
||||
|
||||
// Load saved values
|
||||
fieldsToSave.forEach((field) => {
|
||||
const element = document.getElementById(field.id);
|
||||
if (!element) return; // Skip if element doesn't exist
|
||||
|
||||
const savedValue = localStorage.getItem(field.key);
|
||||
if (savedValue !== null) {
|
||||
// Handle different input types
|
||||
if (element.type === 'checkbox') {
|
||||
element.checked = savedValue === 'true';
|
||||
} else if (element.tagName === 'SELECT') {
|
||||
element.value = savedValue;
|
||||
} else {
|
||||
element.value = savedValue;
|
||||
}
|
||||
});
|
||||
|
||||
// Add a clear data button to the UI
|
||||
const container = document.querySelector('.container');
|
||||
const clearButton = document.createElement('button');
|
||||
clearButton.textContent = 'Clear Saved Data';
|
||||
clearButton.className = 'clear-data-btn';
|
||||
clearButton.style.marginTop = '10px';
|
||||
clearButton.addEventListener('click', function() {
|
||||
if (confirm('Are you sure you want to clear all saved form data?')) {
|
||||
fieldsToSave.forEach(field => {
|
||||
localStorage.removeItem(field.key);
|
||||
});
|
||||
localStorage.removeItem("cod_ssoToken");
|
||||
alert('All saved data has been cleared. Refresh the page to see changes.');
|
||||
}
|
||||
});
|
||||
container.appendChild(clearButton);
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
// Save values on change
|
||||
fieldsToSave.forEach((field) => {
|
||||
const element = document.getElementById(field.id);
|
||||
if (!element) return; // Skip if element doesn't exist
|
||||
|
||||
// Different event listener based on input type
|
||||
if (element.type === 'checkbox') {
|
||||
element.addEventListener('change', function () {
|
||||
localStorage.setItem(field.key, element.checked);
|
||||
});
|
||||
} else if (element.tagName === 'SELECT') {
|
||||
element.addEventListener('change', function () {
|
||||
localStorage.setItem(field.key, element.value);
|
||||
});
|
||||
} else {
|
||||
element.addEventListener('input', function () {
|
||||
localStorage.setItem(field.key, element.value);
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
// Special handling for SSO Token
|
||||
const ssoTokenInput = document.getElementById('ssoToken');
|
||||
const savedSsoToken = localStorage.getItem('cod_ssoToken');
|
||||
|
||||
if (savedSsoToken) {
|
||||
ssoTokenInput.value = savedSsoToken;
|
||||
}
|
||||
|
||||
// Ask the user before saving SSO token
|
||||
ssoTokenInput.addEventListener('input', function () {
|
||||
if (
|
||||
confirm(
|
||||
'Would you like to save your SSO token? Note: This is stored on your device only.'
|
||||
)
|
||||
) {
|
||||
localStorage.setItem('cod_ssoToken', ssoTokenInput.value);
|
||||
}
|
||||
});
|
||||
|
||||
// Add a clear data button to the UI
|
||||
const container = document.querySelector('.container');
|
||||
const clearButton = document.createElement('button');
|
||||
clearButton.textContent = 'Clear Saved Data';
|
||||
clearButton.className = 'clear-data-btn';
|
||||
clearButton.style.marginTop = '10px';
|
||||
clearButton.addEventListener('click', function () {
|
||||
if (confirm('Are you sure you want to clear all saved form data?')) {
|
||||
fieldsToSave.forEach((field) => {
|
||||
localStorage.removeItem(field.key);
|
||||
});
|
||||
localStorage.removeItem('cod_ssoToken');
|
||||
alert(
|
||||
'All saved data has been cleared. Refresh the page to see changes.'
|
||||
);
|
||||
}
|
||||
});
|
||||
container.appendChild(clearButton);
|
||||
});
|
||||
|
@ -3,51 +3,51 @@ const fs = require('fs');
|
||||
const path = require('path');
|
||||
|
||||
class Logger {
|
||||
constructor(options = {}) {
|
||||
constructor(options = {}) {
|
||||
// Dynamically determine the base directory
|
||||
const isPackaged = process.pkg !== undefined;
|
||||
let baseDir;
|
||||
|
||||
|
||||
if (isPackaged) {
|
||||
// If running as a packaged executable, use the directory where the executable is located
|
||||
baseDir = path.dirname(process.execPath);
|
||||
// If running as a packaged executable, use the directory where the executable is located
|
||||
baseDir = path.dirname(process.execPath);
|
||||
} else {
|
||||
// In development, use the project root directory
|
||||
baseDir = process.cwd();
|
||||
// In development, use the project root directory
|
||||
baseDir = process.cwd();
|
||||
}
|
||||
|
||||
|
||||
this.options = {
|
||||
logToConsole: options.logToConsole !== false,
|
||||
logToFile: options.logToFile || false,
|
||||
// Use the determined base directory + logs
|
||||
logDirectory: options.logDirectory || path.join(baseDir, 'logs'),
|
||||
userActivityLogFile: options.userActivityLogFile || 'user-activity.log',
|
||||
apiLogFile: options.apiLogFile || 'api.log',
|
||||
minLevel: options.minLevel || 'info'
|
||||
logToConsole: options.logToConsole !== false,
|
||||
logToFile: options.logToFile || false,
|
||||
// Use the determined base directory + logs
|
||||
logDirectory: options.logDirectory || path.join(baseDir, 'logs'),
|
||||
userActivityLogFile: options.userActivityLogFile || 'user-activity.log',
|
||||
apiLogFile: options.apiLogFile || 'api.log',
|
||||
minLevel: options.minLevel || 'info',
|
||||
};
|
||||
|
||||
|
||||
// Create log directory if it doesn't exist and logging to file is enabled
|
||||
if (this.options.logToFile) {
|
||||
try {
|
||||
try {
|
||||
if (!fs.existsSync(this.options.logDirectory)) {
|
||||
fs.mkdirSync(this.options.logDirectory, { recursive: true });
|
||||
fs.mkdirSync(this.options.logDirectory, { recursive: true });
|
||||
}
|
||||
} catch (err) {
|
||||
} catch (err) {
|
||||
console.error(`Failed to create log directory: ${err.message}`);
|
||||
// Fall back to logging in the same directory as the executable if directory creation fails
|
||||
this.options.logDirectory = baseDir;
|
||||
console.log(`Falling back to logging in: ${this.options.logDirectory}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Log levels and their priorities
|
||||
this.levels = {
|
||||
debug: 0,
|
||||
info: 1,
|
||||
warn: 2,
|
||||
error: 3
|
||||
debug: 0,
|
||||
info: 1,
|
||||
warn: 2,
|
||||
error: 3,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
shouldLog(level) {
|
||||
return this.levels[level] >= this.levels[this.options.minLevel];
|
||||
@ -58,7 +58,7 @@ constructor(options = {}) {
|
||||
const logObject = {
|
||||
timestamp,
|
||||
type,
|
||||
message
|
||||
message,
|
||||
};
|
||||
|
||||
if (Object.keys(data).length > 0) {
|
||||
@ -70,11 +70,12 @@ constructor(options = {}) {
|
||||
|
||||
writeToFile(content, isUserActivity = false) {
|
||||
if (!this.options.logToFile) return;
|
||||
|
||||
const logFile = isUserActivity
|
||||
? path.join(this.options.logDirectory, this.options.userActivityLogFile)
|
||||
|
||||
const logFile =
|
||||
isUserActivity ?
|
||||
path.join(this.options.logDirectory, this.options.userActivityLogFile)
|
||||
: path.join(this.options.logDirectory, this.options.apiLogFile);
|
||||
|
||||
|
||||
// Check if the log directory exists before writing
|
||||
try {
|
||||
// Ensure the directory exists
|
||||
@ -82,77 +83,79 @@ constructor(options = {}) {
|
||||
fs.mkdirSync(this.options.logDirectory, { recursive: true });
|
||||
console.log(`Log directory recreated at: ${this.options.logDirectory}`);
|
||||
}
|
||||
|
||||
|
||||
// Now write the log
|
||||
fs.appendFileSync(logFile, content + '\n');
|
||||
} catch (err) {
|
||||
console.error(`Error writing to log file: ${err.message}`);
|
||||
// Fall back to console logging if file writing fails
|
||||
if (this.options.logToConsole) {
|
||||
console.log(`Failed to write to log file, logging to console instead: ${content}`);
|
||||
console.log(
|
||||
`Failed to write to log file, logging to console instead: ${content}`
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
debug(message, data = {}) {
|
||||
if (!this.shouldLog('debug')) return;
|
||||
|
||||
|
||||
const logEntry = this.formatLogEntry('DEBUG', message, data);
|
||||
|
||||
|
||||
if (this.options.logToConsole) {
|
||||
console.debug(logEntry);
|
||||
}
|
||||
|
||||
|
||||
this.writeToFile(logEntry);
|
||||
}
|
||||
|
||||
info(message, data = {}) {
|
||||
if (!this.shouldLog('info')) return;
|
||||
|
||||
|
||||
const logEntry = this.formatLogEntry('INFO', message, data);
|
||||
|
||||
|
||||
if (this.options.logToConsole) {
|
||||
console.log(logEntry);
|
||||
}
|
||||
|
||||
|
||||
this.writeToFile(logEntry);
|
||||
}
|
||||
|
||||
warn(message, data = {}) {
|
||||
if (!this.shouldLog('warn')) return;
|
||||
|
||||
|
||||
const logEntry = this.formatLogEntry('WARN', message, data);
|
||||
|
||||
|
||||
if (this.options.logToConsole) {
|
||||
console.warn(logEntry);
|
||||
}
|
||||
|
||||
|
||||
this.writeToFile(logEntry);
|
||||
}
|
||||
|
||||
error(message, data = {}) {
|
||||
if (!this.shouldLog('error')) return;
|
||||
|
||||
|
||||
const logEntry = this.formatLogEntry('ERROR', message, data);
|
||||
|
||||
|
||||
if (this.options.logToConsole) {
|
||||
console.error(logEntry);
|
||||
}
|
||||
|
||||
|
||||
this.writeToFile(logEntry);
|
||||
}
|
||||
|
||||
// Specialized method for user activity logging
|
||||
userActivity(eventType, data = {}) {
|
||||
const logEntry = this.formatLogEntry('USER_ACTIVITY', eventType, data);
|
||||
|
||||
|
||||
// Don't log user activity to the console
|
||||
/*
|
||||
if (this.options.logToConsole) {
|
||||
console.log(`[USER_ACTIVITY] ${logEntry}`);
|
||||
}
|
||||
*/
|
||||
|
||||
|
||||
this.writeToFile(logEntry, true);
|
||||
}
|
||||
}
|
||||
@ -161,10 +164,10 @@ constructor(options = {}) {
|
||||
const defaultLogger = new Logger({
|
||||
logToConsole: true,
|
||||
logToFile: true,
|
||||
minLevel: process.env.NODE_ENV === 'production' ? 'info' : 'debug'
|
||||
minLevel: process.env.NODE_ENV === 'production' ? 'info' : 'debug',
|
||||
});
|
||||
|
||||
module.exports = {
|
||||
Logger,
|
||||
logger: defaultLogger
|
||||
};
|
||||
logger: defaultLogger,
|
||||
};
|
||||
|
Loading…
x
Reference in New Issue
Block a user