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