diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 629fd787..18877785 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -8,6 +8,9 @@ on: branches: - "*" types: [opened, synchronize, reopened] +concurrency: + group: ${{ github.ref }} + cancel-in-progress: false jobs: build: name: Build binaries @@ -18,17 +21,6 @@ jobs: - Debug - Release steps: - - name: Wait for previous workflows - if: github.event_name == 'push' && (github.ref == 'refs/heads/main' || github.ref == 'refs/heads/develop') - uses: softprops/turnstyle@v1 - with: - poll-interval-seconds: 10 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - - - name: Set up git config - run: git config --global url."https://".insteadOf git:// - - name: Check out files uses: actions/checkout@v3 with: @@ -38,10 +30,9 @@ jobs: lfs: false - name: Add msbuild to PATH - uses: microsoft/setup-msbuild@v1.0.2 + uses: microsoft/setup-msbuild@v1.1.3 - name: Generate project files - #run: tools/premake5 vs2022 --ci-build run: tools/premake5 vs2022 - name: Set up problem matching @@ -51,7 +42,7 @@ jobs: run: msbuild /m /v:minimal /p:Configuration=${{matrix.configuration}} /p:Platform=x64 build/h1-mod.sln - name: Upload ${{matrix.configuration}} binaries - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v3.1.0 with: name: ${{matrix.configuration}} binaries path: | @@ -59,7 +50,7 @@ jobs: build/bin/x64/${{matrix.configuration}}/h1-mod.pdb - name: Upload ${{matrix.configuration}} data artifacts - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v3.1.0 with: name: ${{matrix.configuration}} data artifacts path: | @@ -79,12 +70,12 @@ jobs: run: echo "H1_MOD_MASTER_PATH=${{ secrets.H1_MOD_MASTER_SSH_PATH_DEV }}" >> $GITHUB_ENV - name: Download Release binaries - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v3 with: name: Release binaries - name: Download Release data artifacts - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v3 with: name: Release data artifacts path: data @@ -98,13 +89,6 @@ jobs: - name: Add known hosts run: ssh-keyscan -H ${{ secrets.H1_MOD_MASTER_SSH_ADDRESS }} >> ~/.ssh/known_hosts - - name: Wait for previous workflows - uses: softprops/turnstyle@v1 - with: - poll-interval-seconds: 10 - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} - - name: Remove old data files run: ssh ${{ secrets.H1_MOD_MASTER_SSH_USER }}@${{ secrets.H1_MOD_MASTER_SSH_ADDRESS }} rm -rf ${{ env.H1_MOD_MASTER_PATH }}/h1-mod/* diff --git a/.gitmodules b/.gitmodules index a6a9bd10..62d68e1f 100644 --- a/.gitmodules +++ b/.gitmodules @@ -12,7 +12,7 @@ url = https://github.com/TsudaKageyu/minhook.git [submodule "deps/discord-rpc"] path = deps/discord-rpc - url = https://github.com/discord/discord-rpc.git + url = https://github.com/fedddddd/discord-rpc.git [submodule "deps/asmjit"] path = deps/asmjit url = https://github.com/asmjit/asmjit.git @@ -50,7 +50,7 @@ [submodule "deps/gsc-tool"] path = deps/gsc-tool url = https://github.com/xensik/gsc-tool.git - branch = xlabs + branch = dev [submodule "deps/stb"] path = deps/stb url = https://github.com/nothings/stb.git diff --git a/README.md b/README.md index 46739214..2e731411 100644 --- a/README.md +++ b/README.md @@ -10,9 +10,7 @@

-A client for Call of Duty: Modern Warfare Remastered. - -[This project is based on S1x.](https://github.com/XLabsProject/s1x-client) +NOTE: You must legally own Call of Duty®: Modern Warfare Remastered to run this mod. Cracked/Pirated versions of the game are **NOT** supported. ## Compile from source @@ -27,13 +25,11 @@ A client for Call of Duty: Modern Warfare Remastered. | `--copy-to=PATH` | Optional, copy the EXE to a custom folder after build, define the path here if wanted. | | `--dev-build` | Enable development builds of the client. | -## Credits: +## Credits -- [XLabsProject](https://github.com/XLabsProject) - codebase and iw6x/s1x research -- [quaK](https://github.com/Joelrau) - lots of insight and help -- [fed](https://github.com/fedddddd) - fixed DW/networking, work from [h2-mod](https://github.com/fedddddd/h2-mod) -- [Skull](https://github.com/skkuull) + [mjkzy](https://github.com/mjkzy) - porting code from s1x -- [momo5502](https://github.com/momo5502) - Arxan/Steam research, developer of XLabsProject :D +- [S1x](https://github.com/XLabsProject/s1x-client) - codebase and research (predecessor of MWR) +- [h2-mod](https://github.com/fedddddd/h2-mod) - research (successor of MWR) +- [momo5502](https://github.com/momo5502) - Arxan/Steam research, developer of [XLabsProject](https://github.com/XLabsProject) ## Disclaimer diff --git a/data/cdata/scripts/mp/classes.gsc b/data/cdata/scripts/mp/classes.gsc new file mode 100644 index 00000000..0286f8e7 --- /dev/null +++ b/data/cdata/scripts/mp/classes.gsc @@ -0,0 +1,34 @@ +main() +{ + replacefunc(maps\mp\gametypes\_menus::getclasschoice, ::getclasschoice); +} + +getclasschoice(choice) +{ + if (choice <= 100) + { + if (getdvar("sv_disableCustomClasses") == "1") + { + return "class0"; + } + + choice = "custom" + choice; + } + else if (choice <= 200) + { + choice -= 101; + choice = "class" + choice; + } + else if ( choice <= 206 ) + { + choice -= 200; + choice = "axis_recipe" + choice; + } + else + { + choice -= 206; + choice = "allies_recipe" + choice; + } + + return choice; +} diff --git a/data/cdata/scripts/mp_patches/custom_weapons.gsc b/data/cdata/scripts/mp_patches/custom_weapons.gsc new file mode 100644 index 00000000..cbbd16cd --- /dev/null +++ b/data/cdata/scripts/mp_patches/custom_weapons.gsc @@ -0,0 +1,383 @@ +main() +{ + replacefunc(maps\mp\gametypes\_class::isvalidprimary, ::isvalidprimary); + replacefunc(maps\mp\gametypes\_class::isvalidsecondary, ::isvalidsecondary); + replacefunc(maps\mp\gametypes\_class::isvalidweapon, ::isvalidweapon); + replacefunc(maps\mp\gametypes\_class::buildweaponname, ::buildweaponname); + replacefunc(maps\mp\gametypes\_weapons::watchweaponchange, ::watchweaponchange); +} + +find_in_table(csv, weap) +{ + rows = tablegetrowcount(csv); + + for (i = 0; i < rows; i++) + { + if (tablelookupbyrow(csv, i, 0) == weap) + { + return true; + } + } + + return false; +} + +get_attachment_override(weapon, attachment) +{ + csv = "mp/attachoverrides.csv"; + rows = tablegetrowcount(csv); + + if (!issubstr(weapon, "_mp")) + { + weapon += "_mp"; + } + + for (i = 0; i < rows; i++) + { + if (tablelookupbyrow(csv, i, 0) == weapon && tablelookupbyrow(csv, i, 1) == attachment) + { + return tablelookupbyrow(csv, i, 2); + } + } +} + +get_attachment_name(weapon, attachment) +{ + name = tablelookup("mp/attachkits.csv", 1, attachment, 2); + override = get_attachment_override(weapon, name); + + if (isdefined(override) && override != "") + { + return override; + } + + return name; +} + +is_custom_weapon(weap) +{ + return find_in_table("mp/customweapons.csv", weap); +} + +watchweaponchange() +{ + self endon("death"); + self endon("disconnect"); + self endon("faux_spawn"); + thread maps\mp\gametypes\_weapons::watchstartweaponchange(); + self.lastdroppableweapon = self.currentweaponatspawn; + self.hitsthismag = []; + var_0 = self getcurrentweapon(); + + if (maps\mp\_utility::iscacprimaryweapon(var_0) && !isdefined(self.hitsthismag[var_0])) + { + self.hitsthismag[var_0] = weaponclipsize(var_0); + } + + self.bothbarrels = undefined; + + if (issubstr(var_0, "ranger")) + { + thread maps\mp\gametypes\_weapons::watchrangerusage(var_0); + } + + var_1 = 1; + + for (;;) + { + if (!var_1) + { + self waittill("weapon_change"); + } + + var_1 = 0; + var_0 = self getcurrentweapon(); + + if (var_0 == "none") + { + continue; + } + + var_2 = getweaponattachments(var_0); + self.has_opticsthermal = 0; + self.has_target_enhancer = 0; + self.has_stock = 0; + self.has_laser = 0; + + if (isdefined(var_2)) + { + foreach (var_4 in var_2) + { + if (var_4 == "opticstargetenhancer") + { + self.has_target_enhancer = 1; + continue; + } + + if (var_4 == "stock") + { + self.has_stock = 1; + continue; + } + + if (var_4 == "lasersight") + { + self.has_laser = 1; + continue; + } + + if (issubstr(var_4, "opticsthermal")) + { + self.has_opticsthermal = 1; + } + } + } + + if (maps\mp\_utility::isbombsiteweapon(var_0)) + { + continue; + } + + var_6 = maps\mp\_utility::getweaponnametokens(var_0); + self.bothbarrels = undefined; + + if (issubstr(var_0, "ranger")) + { + thread maps\mp\gametypes\_weapons::watchrangerusage(var_0); + } + + if (var_6[0] == "alt") + { + var_7 = getsubstr(var_0, 4); + var_0 = var_7; + var_6 = maps\mp\_utility::getweaponnametokens(var_0); + } + + if (var_0 != "none" && var_6[0] != "iw5" && var_6[0] != "iw6" && var_6[0] != "h1" && var_6[0] != "h2") + { + if (maps\mp\_utility::iscacprimaryweapon(var_0) && !isdefined(self.hitsthismag[var_0 + "_mp"])) + { + self.hitsthismag[var_0 + "_mp"] = weaponclipsize(var_0 + "_mp"); + } + } + else if (var_0 != "none" && (var_6[0] == "iw5" || var_6[0] == "iw6" || var_6[0] == "h1" || var_6[0] == "h2")) + { + if (maps\mp\_utility::iscacprimaryweapon(var_0) && !isdefined(self.hitsthismag[var_0])) + { + self.hitsthismag[var_0] = weaponclipsize(var_0); + } + } + + if (maps\mp\gametypes\_weapons::maydropweapon(var_0)) + { + self.lastdroppableweapon = var_0; + } + + self.changingweapon = undefined; + } +} + +buildweaponname(var_0, var_1, var_2, var_3, var_4, var_5) +{ + if (!isdefined(var_0) || var_0 == "none" || var_0 == "") + { + return var_0; + } + + if (!isdefined(level.lettertonumber)) + { + level.lettertonumber = maps\mp\gametypes\_class::makeletterstonumbers(); + } + + var_6 = ""; + + if (issubstr(var_0, "iw5_") || issubstr(var_0, "h1_") || issubstr(var_0, "h2_")) + { + var_7 = var_0 + "_mp"; + var_8 = var_0.size; + + if (issubstr(var_0, "h1_") || issubstr(var_0, "h2_")) + { + var_6 = getsubstr(var_0, 3, var_8); + } + else + { + var_6 = getsubstr(var_0, 4, var_8); + } + } + else + { + var_7 = var_0; + var_6 = var_0; + } + + if (var_7 == "h1_junsho_mp") + { + var_1 = "akimbohidden"; + } + + var_9 = isdefined(var_1) && var_1 != "none"; + var_10 = isdefined(var_2) && var_2 != "none"; + + if (!var_10) + { + var_11 = tablelookuprownum("mp/furniturekits/base.csv", 0, var_7); + + if (var_11 >= 0) + { + var_2 = "base"; + var_10 = 1; + } + } + + if (!issubstr(var_0, "h1_")) + { + if (var_9) + { + name = get_attachment_name(var_0, var_1); + if (isdefined(name) && name != "") + { + var_7 += "_" + name; + } + } + } + else if (var_9 || var_10) + { + if (!var_9) + var_1 = "none"; + + if (!var_10) + var_2 = "base"; + + var_7 += ("_a#" + var_1); + var_7 += ("_f#" + var_2); + } + + if (issubstr(var_7, "iw5_") || issubstr(var_7, "h1_") || issubstr(var_7, "h2_")) + { + var_7 = maps\mp\gametypes\_class::buildweaponnamereticle(var_7, var_4); + var_7 = maps\mp\gametypes\_class::buildweaponnameemblem(var_7, var_5); + var_7 = maps\mp\gametypes\_class::buildweaponnamecamo(var_7, var_3); + return var_7; + } + else if (!isvalidweapon(var_7 + "_mp")) + { + return var_0 + "_mp"; + } + else + { + var_7 = maps\mp\gametypes\_class::buildweaponnamereticle(var_7, var_4); + var_7 = maps\mp\gametypes\_class::buildweaponnameemblem(var_7, var_5); + var_7 = maps\mp\gametypes\_class::buildweaponnamecamo(var_7, var_3); + return var_7 + "_mp"; + } +} + +isvalidweapon(var_0, var_1) +{ + if (!isdefined(level.weaponrefs)) + { + level.weaponrefs = []; + + foreach (var_3 in level.weaponlist) + { + level.weaponrefs[var_3] = 1; + } + } + + if (isdefined(level.weaponrefs[var_0])) + { + return 1; + } + + return 0; +} + +isvalidsecondary(var_0, var_1, var_2) +{ + if (maps\mp\_utility::is_true(var_1)) + { + return isvalidprimary(var_0); + } + + if (maps\mp\_utility::islootweapon(var_0)) + { + var_0 = maps\mp\gametypes\_class::getbasefromlootversion(var_0); + } + + if (is_custom_weapon(var_0)) + { + return true; + } + + switch (var_0) + { + case "none": + case "h1_beretta": + case "h1_colt45": + case "h1_deserteagle": + case "h1_deserteagle55": + case "h1_usp": + case "h1_janpst": + case "h1_aprpst": + case "h1_augpst": + case "h1_rpg": + return 1; + default: + return 0; + } + + return 0; +} + + +isvalidprimary(var_0, var_1) +{ + if (is_custom_weapon(var_0)) + { + return true; + } + + switch (var_0) + { + case "h1_ak47": + case "h1_g3": + case "h1_g36c": + case "h1_m14": + case "h1_m16": + case "h1_m4": + case "h1_mp44": + case "h1_xmlar": + case "h1_aprast": + case "h1_augast": + case "h1_ak74u": + case "h1_mp5": + case "h1_p90": + case "h1_skorpion": + case "h1_uzi": + case "h1_febsmg": + case "h1_aprsmg": + case "h1_augsmg": + case "h1_m1014": + case "h1_winchester1200": + case "h1_kam12": + case "h1_junsho": + case "h1_m60e4": + case "h1_rpd": + case "h1_saw": + case "h1_feblmg": + case "h1_junlmg": + case "h1_barrett": + case "h1_dragunov": + case "h1_m21": + case "h1_m40a3": + case "h1_remington700": + case "h1_febsnp": + case "h1_junsnp": + return 1; + default: + return 0; + } + + return 0; +} diff --git a/data/cdata/scripts/sp/battlechatter_patch.gsc b/data/cdata/scripts/sp/battlechatter_patch.gsc new file mode 100644 index 00000000..8dcb59d2 --- /dev/null +++ b/data/cdata/scripts/sp/battlechatter_patch.gsc @@ -0,0 +1,242 @@ +main() +{ + replacefunc(animscripts\battlechatter::init_battlechatter, ::init_battlechatter); +} + +init_battlechatter() +{ + if ( isdefined( anim.chatinitialized ) && anim.chatinitialized ) + return; + + if ( getdvar( "bcs_enable" ) == "" ) + setdvar( "bcs_enable", "on" ); + + if ( getdvar( "bcs_enable" ) == "off" ) + { + anim.chatinitialized = 0; + anim.player.chatinitialized = 0; + return; + } + + anim.chatinitialized = 1; + anim.player.chatinitialized = 0; + + if ( getdvar( "bcs_filterThreat" ) == "" ) + setdvar( "bcs_filterThreat", "off" ); + + if ( getdvar( "bcs_filterInform" ) == "" ) + setdvar( "bcs_filterInform", "off" ); + + if ( getdvar( "bcs_filterOrder" ) == "" ) + setdvar( "bcs_filterOrder", "off" ); + + if ( getdvar( "bcs_filterReaction" ) == "" ) + setdvar( "bcs_filterReaction", "off" ); + + if ( getdvar( "bcs_filterResponse" ) == "" ) + setdvar( "bcs_filterResponse", "off" ); + + if ( getdvar( "bcs_threatLimitTargettedBySelf" ) == "" ) + setdvar( "bcs_threatLimitTargettedBySelf", "off" ); + + if ( getdvar( "bcs_threatLimitTargetingPlayer" ) == "" ) + setdvar( "bcs_threatLimitTargetingPlayer", "off" ); + + if ( getdvar( "bcs_threatLimitInPlayerFOV" ) == "" ) + setdvar( "bcs_threatLimitInPlayerFOV", "on" ); + + if ( getdvar( "bcs_threatLimitInLocation" ) == "" ) + setdvar( "bcs_threatLimitInLocation", "on" ); + + if ( getdvar( "bcs_threatLimitSpeakerDist" ) == "" ) + setdvar( "bcs_threatLimitSpeakerDist", "512" ); + + if ( getdvar( "bcs_threatLimitThreatDist" ) == "" ) + setdvar( "bcs_threatLimitThreatDist", "2048" ); + + if ( getdvar( "bcs_threatPlayerRelative" ) == "" ) + setdvar( "bcs_threatPlayerRelative", "off" ); + + if ( getdvar( "debug_bcprint" ) == "" ) + setdvar( "debug_bcprint", "off" ); + + if ( getdvar( "debug_bcshowqueue" ) == "" ) + setdvar( "debug_bcshowqueue", "off" ); + + if ( getdvar( "debug_bcprintdump" ) == "" ) + setdvar( "debug_bcprintdump", "off" ); + + anim.countryids["british"] = "UK"; + anim.countryids["american"] = "US"; + anim.countryids["russian"] = "RU"; + anim.countryids["arab"] = "AB"; + anim.usedids = []; + anim.usedids["russian"] = []; + anim.usedids["russian"][0] = spawnstruct(); + anim.usedids["russian"][0].count = 0; + anim.usedids["russian"][0].npcid = "1"; + anim.usedids["russian"][1] = spawnstruct(); + anim.usedids["russian"][1].count = 0; + anim.usedids["russian"][1].npcid = "2"; + anim.usedids["russian"][2] = spawnstruct(); + anim.usedids["russian"][2].count = 0; + anim.usedids["russian"][2].npcid = "3"; + anim.usedids["british"] = []; + anim.usedids["british"][0] = spawnstruct(); + anim.usedids["british"][0].count = 0; + anim.usedids["british"][0].npcid = "0"; + anim.usedids["british"][1] = spawnstruct(); + anim.usedids["british"][1].count = 0; + anim.usedids["british"][1].npcid = "1"; + anim.usedids["british"][2] = spawnstruct(); + anim.usedids["british"][2].count = 0; + anim.usedids["british"][2].npcid = "2"; + anim.usedids["british"][3] = spawnstruct(); + anim.usedids["british"][3].count = 0; + anim.usedids["british"][3].npcid = "3"; + anim.usedids["american"] = []; + anim.usedids["american"][0] = spawnstruct(); + anim.usedids["american"][0].count = 0; + anim.usedids["american"][0].npcid = "0"; //0 - US_7 + anim.usedids["american"][1] = spawnstruct(); + anim.usedids["american"][1].count = 0; + anim.usedids["american"][1].npcid = "1"; + anim.usedids["american"][2] = spawnstruct(); + anim.usedids["american"][2].count = 0; + anim.usedids["american"][2].npcid = "2"; + anim.usedids["american"][3] = spawnstruct(); + anim.usedids["american"][3].count = 0; + anim.usedids["american"][3].npcid = "3"; + anim.usedids["american"][4] = spawnstruct(); + anim.usedids["american"][4].count = 0; + anim.usedids["american"][4].npcid = "4"; //4 - US_6 + anim.usedids["arab"] = []; + anim.usedids["arab"][0] = spawnstruct(); + anim.usedids["arab"][0].count = 0; + anim.usedids["arab"][0].npcid = "0"; + anim.usedids["arab"][1] = spawnstruct(); + anim.usedids["arab"][1].count = 0; + anim.usedids["arab"][1].npcid = "1"; + anim.usedids["arab"][2] = spawnstruct(); + anim.usedids["arab"][2].count = 0; + anim.usedids["arab"][2].npcid = "2"; + anim.usedids["arab"][3] = spawnstruct(); + anim.usedids["arab"][3].count = 0; + anim.usedids["arab"][3].npcid = "3"; + anim.eventtypeminwait = []; + anim.eventtypeminwait["threat"] = []; + anim.eventtypeminwait["response"] = []; + anim.eventtypeminwait["reaction"] = []; + anim.eventtypeminwait["order"] = []; + anim.eventtypeminwait["inform"] = []; + anim.eventtypeminwait["custom"] = []; + anim.eventtypeminwait["direction"] = []; + + if ( isdefined( level._stealth ) ) + { + anim.eventactionminwait["threat"]["self"] = 20000; + anim.eventactionminwait["threat"]["squad"] = 30000; + } + else + { + anim.eventactionminwait["threat"]["self"] = 12000; + anim.eventactionminwait["threat"]["squad"] = 8000; + } + + anim.eventactionminwait["response"]["self"] = 1000; + anim.eventactionminwait["response"]["squad"] = 1000; + anim.eventactionminwait["reaction"]["self"] = 1000; + anim.eventactionminwait["reaction"]["squad"] = 1000; + anim.eventactionminwait["order"]["self"] = 8000; + anim.eventactionminwait["order"]["squad"] = 40000; + anim.eventactionminwait["inform"]["self"] = 6000; + anim.eventactionminwait["inform"]["squad"] = 8000; + anim.eventactionminwait["custom"]["self"] = 0; + anim.eventactionminwait["custom"]["squad"] = 5000; + anim.eventtypeminwait["reaction"]["casualty"] = 20000; + anim.eventtypeminwait["reaction"]["taunt"] = 200000; + anim.eventtypeminwait["inform"]["reloading"] = 20000; + anim.eventpriority["threat"]["infantry"] = 0.5; + anim.eventpriority["threat"]["vehicle"] = 0.7; + anim.eventpriority["response"]["killfirm"] = 0.8; + anim.eventpriority["response"]["ack"] = 0.9; + anim.eventpriority["reaction"]["casualty"] = 0.5; + anim.eventpriority["reaction"]["taunt"] = 0.9; + anim.eventpriority["order"]["cover"] = 0.9; + anim.eventpriority["order"]["action"] = 0.5; + anim.eventpriority["order"]["move"] = 0.9; + anim.eventpriority["order"]["displace"] = 0.5; + anim.eventpriority["inform"]["killfirm"] = 0.6; + anim.eventpriority["inform"]["attack"] = 0.9; + anim.eventpriority["inform"]["incoming"] = 0.8; + anim.eventpriority["inform"]["reloading"] = 0.2; + anim.eventpriority["inform"]["suppressed"] = 0.2; + anim.eventpriority["custom"]["generic"] = 1.0; + anim.eventduration["threat"]["infantry"] = 1000; + anim.eventduration["threat"]["vehicle"] = 1000; + anim.eventduration["response"]["killfirm"] = 3000; + anim.eventduration["response"]["ack"] = 2000; + anim.eventduration["reaction"]["casualty"] = 2000; + anim.eventduration["reaction"]["taunt"] = 2000; + anim.eventduration["order"]["cover"] = 3000; + anim.eventduration["order"]["action"] = 3000; + anim.eventduration["order"]["move"] = 3000; + anim.eventduration["order"]["displace"] = 3000; + anim.eventduration["inform"]["killfirm"] = 1000; + anim.eventduration["inform"]["attack"] = 1000; + anim.eventduration["inform"]["incoming"] = 1000; + anim.eventduration["inform"]["reloading"] = 1000; + anim.eventduration["inform"]["suppressed"] = 2000; + anim.eventduration["custom"]["generic"] = 1000; + anim.chatcount = 0; + anim.moveorigin = spawn( "script_origin", ( 0.0, 0.0, 0.0 ) ); + anim.areas = getentarray( "trigger_location", "targetname" ); + anim.locations = getentarray( "trigger_location", "targetname" ); + anim.landmarks = getentarray( "trigger_landmark", "targetname" ); + anim.squadcreatefuncs[anim.squadcreatefuncs.size] = animscripts\battlechatter::init_squadbattlechatter; + anim.squadcreatestrings[anim.squadcreatestrings.size] = "::init_squadBattleChatter"; + anim.isteamspeaking["allies"] = 0; + anim.isteamsaying["allies"]["threat"] = 0; + anim.isteamsaying["allies"]["order"] = 0; + anim.isteamsaying["allies"]["reaction"] = 0; + anim.isteamsaying["allies"]["response"] = 0; + anim.isteamsaying["allies"]["inform"] = 0; + anim.isteamsaying["allies"]["custom"] = 0; + anim.isteamspeaking["axis"] = 0; + anim.isteamsaying["axis"]["threat"] = 0; + anim.isteamsaying["axis"]["order"] = 0; + anim.isteamsaying["axis"]["reaction"] = 0; + anim.isteamsaying["axis"]["response"] = 0; + anim.isteamsaying["axis"]["inform"] = 0; + anim.isteamsaying["axis"]["custom"] = 0; + anim.isteamspeaking["neutral"] = 0; + anim.isteamsaying["neutral"]["threat"] = 0; + anim.isteamsaying["neutral"]["order"] = 0; + anim.isteamsaying["neutral"]["reaction"] = 0; + anim.isteamsaying["neutral"]["response"] = 0; + anim.isteamsaying["neutral"]["inform"] = 0; + anim.isteamsaying["neutral"]["custom"] = 0; + + if ( !isdefined( level.battlechatter ) ) + { + level.battlechatter = []; + level.battlechatter["allies"] = 1; + level.battlechatter["axis"] = 1; + level.battlechatter["neutral"] = 1; + } + + anim.lastteamspeaktime = []; + anim.lastteamspeaktime["allies"] = -5000; + anim.lastteamspeaktime["axis"] = -5000; + + for ( var_0 = 0; var_0 < anim.squadindex.size; var_0++ ) + { + if ( isdefined( anim.squadindex[var_0].chatinitialized ) && anim.squadindex[var_0].chatinitialized ) + continue; + + anim.squadindex[var_0] animscripts\battlechatter::init_squadbattlechatter(); + } + + level notify( "battlechatter initialized" ); + anim notify( "battlechatter initialized" ); +} \ No newline at end of file diff --git a/data/cdata/ui_scripts/classes/__init__.lua b/data/cdata/ui_scripts/classes/__init__.lua new file mode 100644 index 00000000..f1c997ee --- /dev/null +++ b/data/cdata/ui_scripts/classes/__init__.lua @@ -0,0 +1,13 @@ +if game:issingleplayer() or (Engine.InFrontend()) then + return +end + +local getclasscount = Cac.GetCustomClassCount +Cac.GetCustomClassCount = function(...) + local value = Engine.GetDvarBool("sv_disableCustomClasses") + if (value) then + return 0 + end + + return getclasscount(...) +end \ No newline at end of file diff --git a/data/cdata/ui_scripts/custom_weapons/__init__.lua b/data/cdata/ui_scripts/custom_weapons/__init__.lua new file mode 100644 index 00000000..7adcd8fb --- /dev/null +++ b/data/cdata/ui_scripts/custom_weapons/__init__.lua @@ -0,0 +1,26 @@ +if game:issingleplayer() or (not Engine.InFrontend()) then + return +end + +local cols = { + name = 0, + class = 1, +} + +local csv = "mp/customWeapons.csv" +local rows = Engine.TableGetRowCount(csv) +for i = 0, rows do + local weap = Engine.TableLookupByRow(csv, i, cols.name) + local class = Engine.TableLookupByRow(csv, i, cols.class) + if (type(Cac.Weapons.Primary[class]) == "table") then + table.insert(Cac.Weapons.Primary[class], { + weap, + 0 + }) + elseif (type(Cac.Weapons.Secondary[class]) == "table") then + table.insert(Cac.Weapons.Secondary[class], { + weap, + 0 + }) + end +end diff --git a/data/cdata/ui_scripts/discord/__init__.lua b/data/cdata/ui_scripts/discord/__init__.lua index fe988a68..a4d0520f 100644 --- a/data/cdata/ui_scripts/discord/__init__.lua +++ b/data/cdata/ui_scripts/discord/__init__.lua @@ -2,7 +2,7 @@ if (game:issingleplayer() or Engine.InFrontend()) then return end -local container = LUI.UIVerticalList.new({ +local container = LUI.UIElement.new({ topAnchor = true, rightAnchor = true, top = 20, @@ -11,20 +11,6 @@ local container = LUI.UIVerticalList.new({ spacing = 5 }) -function canasktojoin(userid) - history = history or {} - if (history[userid] ~= nil) then - return false - end - - history[userid] = true - game:ontimeout(function() - history[userid] = nil - end, 15000) - - return true -end - function truncatename(name, length) if (#name <= length - 3) then return name @@ -33,27 +19,57 @@ function truncatename(name, length) return name:sub(1, length - 3) .. "..." end +local requestlist = {} +local requestcount = 0 + function addrequest(request) - if (not canasktojoin(request.userid)) then - return + for i = 1, #requestlist do + if (requestlist[i].userid == request.userid or #requestlist > 5) then + return + end end - if (container.temp) then - container:removeElement(container.temp) - container.temp = nil - end + request.id = requestcount + requestcount = requestcount + 1 + local yoffset = #requestlist * (75 + 5) local invite = LUI.UIElement.new({ leftAnchor = true, rightAnchor = true, - height = 75 + height = 75, + top = yoffset }) + local getcurrentindex = function() + for i = 1, #requestlist do + if (requestlist[i].id == request.id) then + return i + end + end + + return 0 + end + + invite:registerEventHandler("update_position", function() + yoffset = (getcurrentindex() - 1) * (75 + 5) + local state = { + leftAnchor = true, + height = 75, + width = 200, + left = -220, + top = yoffset + } + + invite:registerAnimationState("default", state) + invite:animateToState("default", 50) + end) + invite:registerAnimationState("move_in", { leftAnchor = true, height = 75, width = 200, - left = -220 + left = -220, + top = yoffset }) invite:animateToState("move_in", 100) @@ -105,7 +121,7 @@ function addrequest(request) width = 32, height = 32, left = 1, - material = RegisterMaterial(avatarmaterial) + material = avatarmaterial }) local username = LUI.UIText.new({ @@ -119,8 +135,9 @@ function addrequest(request) font = CoD.TextSettings.BodyFontBold.Font }) - username:setText(string.format("%s^7#%s requested to join your game!", truncatename(request.username, 18), - request.discriminator)) + local requesttext = Engine.Localize("LUA_MENU_DISCORD_REQUEST", truncatename(request.displayname, 18)) + + username:setText(requesttext) local buttons = LUI.UIElement.new({ leftAnchor = true, @@ -156,51 +173,54 @@ function addrequest(request) return button end - buttons:addElement(createbutton("[F1] Accept", true)) - buttons:addElement(createbutton("[F2] Deny")) + local accepttext = Engine.Localize("LUA_MENU_DISCORD_ACCEPT", game:getcommandbind("discord_accept")) + local denytext = Engine.Localize("LUA_MENU_DISCORD_DENY", game:getcommandbind("discord_deny")) + + buttons:addElement(createbutton(accepttext, true)) + buttons:addElement(createbutton(denytext)) local fadeouttime = 50 local timeout = 10 * 1000 - fadeouttime local function close() - container:processEvent({ - name = "update_navigation", - dispatchToChildren = true + table.remove(requestlist, getcurrentindex()) + + invite:registerAnimationState("fade_out", { + leftAnchor = true, + rightAnchor = true, + height = 75, + alpha = 0, + left = 0, + top = yoffset }) + invite:animateToState("fade_out", fadeouttime) invite:addElement(LUI.UITimer.new(fadeouttime + 50, "remove")) invite:registerEventHandler("remove", function() container:removeElement(invite) - if (container.temp) then - container:removeElement(container.temp) - container.temp = nil - end - local temp = LUI.UIElement.new({}) - container.temp = temp - container:addElement(temp) + container:processEvent({ + name = "update_position", + dispatchToChildren = true + }) end) end - buttons:registerEventHandler("keydown_", function(element, event) - if (event.key == "F1") then - close() - discord.respond(request.userid, discord.reply.yes) + local closed = false + request.handleresponse = function(event) + if (closed) then + return end - if (event.key == "F2") then - close() + if (event.accept) then + discord.respond(request.userid, discord.reply.yes) + else discord.respond(request.userid, discord.reply.no) end - end) - invite:registerAnimationState("fade_out", { - leftAnchor = true, - rightAnchor = true, - height = 75, - alpha = 0, - left = 0 - }) + closed = true + close() + end invite:addElement(LUI.UITimer.new(timeout, "end_invite")) invite:registerEventHandler("end_invite", function() @@ -236,7 +256,7 @@ function addrequest(request) avatar:registerEventHandler("update", function() local avatarmaterial = discord.getavatarmaterial(request.userid) - avatar:setImage(RegisterMaterial(avatarmaterial)) + avatar:setImage(avatarmaterial) end) avatar:addElement(LUI.UITimer.new(100, "update")) @@ -250,19 +270,17 @@ function addrequest(request) padding:addElement(buttons) container:addElement(invite) + + table.insert(requestlist, request) end -container:registerEventHandler("keydown", function(element, event) - local first = container:getFirstChild() - - if (not first) then +LUI.roots.UIRoot0:registerEventHandler("discord_response", function(element, event) + if (#requestlist <= 0) then return end - first:processEvent({ - name = "keydown_", - key = event.key - }) + local request = requestlist[1] + request.handleresponse(event) end) LUI.roots.UIRoot0:registerEventHandler("discord_join_request", function(element, event) diff --git a/data/cdata/ui_scripts/patches/__init__.lua b/data/cdata/ui_scripts/patches/__init__.lua index 6e0cc090..c8776663 100644 --- a/data/cdata/ui_scripts/patches/__init__.lua +++ b/data/cdata/ui_scripts/patches/__init__.lua @@ -1,5 +1,6 @@ require("language") require("background_effects") +require("pausequit") if game:issingleplayer() then require("sp_unlockall") diff --git a/data/cdata/ui_scripts/patches/language.lua b/data/cdata/ui_scripts/patches/language.lua index 1d322afa..809af8bf 100644 --- a/data/cdata/ui_scripts/patches/language.lua +++ b/data/cdata/ui_scripts/patches/language.lua @@ -49,8 +49,8 @@ LUI.MenuBuilder.registerType("choose_language_menu", function(a1) menu:AddButton(Engine.Localize(string.format("MENU_%s", available_languages[i])), function() LUI.yesnopopup({ title = Engine.Localize("@MENU_NOTICE"), - text = Engine.Localize("MENU_" .. current_language) .. " → " .. - Engine.Localize("MENU_" .. available_languages[i]) .. "\n\n" .. + text = "^2" .. Engine.Localize("MENU_" .. current_language) .. "^7 → ^5" .. + Engine.Localize("MENU_" .. available_languages[i]) .. "\n\n^7" .. Engine.Localize("@LUA_MENU_CONFIRM_LANGUAGE") .. " " .. Engine.Localize("@MENU_APPLY_LANGUAGE_SETTINGS"), callback = function(result) diff --git a/data/cdata/ui_scripts/patches/pausequit.lua b/data/cdata/ui_scripts/patches/pausequit.lua new file mode 100644 index 00000000..224e944a --- /dev/null +++ b/data/cdata/ui_scripts/patches/pausequit.lua @@ -0,0 +1,32 @@ +if (Engine.InFrontend()) then + return +end + +if game:issingleplayer() and Engine.GetDvarString("mapname") == "coup" then + LUI.onmenuopen("sp_pause_menu", function(element) + local menu = element:getFirstChild() + menu:AddButton("@MENU_SP_SKIP_MISSION", function() + Engine.Exec("map blackout") + end) + end) +end + +if game:issingleplayer() then + LUI.onmenuopen("sp_pause_menu", function(element) + local menu = element:getFirstChild() + menu:AddButton("@MENU_QUIT_TO_DESKTOP", function() + LUI.FlowManager.RequestAddMenu(nil, "quit_popmenu") + end) + end) +end + +if not game:issingleplayer() then + local quitToDesktop = function() + LUI.FlowManager.RequestAddMenu(nil, "quit_popmenu") + end + local addQuitButton = function(element) + local menu = element + menu:AddButton("@MENU_QUIT_TO_DESKTOP", quitToDesktop) + end + LUI.onmenuopen("mp_pause_menu", addQuitButton) +end diff --git a/data/cdata/ui_scripts/patches/shader_dialog.lua b/data/cdata/ui_scripts/patches/shader_dialog.lua index d4723fe0..3b18e53a 100644 --- a/data/cdata/ui_scripts/patches/shader_dialog.lua +++ b/data/cdata/ui_scripts/patches/shader_dialog.lua @@ -15,7 +15,7 @@ local function dialog(...) yes_text = Engine.Localize("@MENU_YES"), no_text = Engine.Localize("@MENU_NO_DONT_ASK"), no_action = function() - Engine.SetDvarInt("r_preloadShadersFrontendAllow", 0) + Engine.SetDvarBool("r_preloadShadersFrontendAllow", false) end, default_focus_index = 2, cancel_will_close = false diff --git a/data/cdata/ui_scripts/server_list/lobby.lua b/data/cdata/ui_scripts/server_list/lobby.lua index 8d71585c..e2d48619 100644 --- a/data/cdata/ui_scripts/server_list/lobby.lua +++ b/data/cdata/ui_scripts/server_list/lobby.lua @@ -1,7 +1,7 @@ local Lobby = luiglobals.Lobby local MPLobbyOnline = LUI.mp_menus.MPLobbyOnline -function LeaveLobby(f5_arg0) +function LeaveLobby() LeaveXboxLive() if Lobby.IsInPrivateParty() == false or Lobby.IsPrivatePartyHost() then LUI.FlowManager.RequestLeaveMenuByName("menu_xboxlive") @@ -9,7 +9,7 @@ function LeaveLobby(f5_arg0) end end -function menu_xboxlive(f16_arg0, f16_arg1) +function menu_xboxlive(f16_arg0) local menu = LUI.MPLobbyBase.new(f16_arg0, { menu_title = "@PLATFORM_UI_HEADER_PLAY_MP_CAPS", memberListState = Lobby.MemberListStates.Prelobby @@ -17,7 +17,7 @@ function menu_xboxlive(f16_arg0, f16_arg1) menu:setClass(LUI.MPLobbyOnline) - local serverListButton = menu:AddButton("@LUA_MENU_SERVERLIST", function(a1, a2) + local serverListButton = menu:AddButton("@LUA_MENU_SERVERLIST", function(a1) LUI.FlowManager.RequestAddMenu(a1, "menu_systemlink_join", true, nil) end) serverListButton:setDisabledRefreshRate(500) @@ -27,9 +27,7 @@ function menu_xboxlive(f16_arg0, f16_arg1) menu:AddPersonalizationButton() menu:AddDepotButton() - -- kinda a weird place to do this, but it's whatever - -- add "MODS" button below depot button - local modsButton = menu:AddButton("@MENU_MODS", function(a1, a2) + menu:AddButton("@MENU_MODS", function(a1) LUI.FlowManager.RequestAddMenu(a1, "mods_menu", true, nil) end) end @@ -58,6 +56,8 @@ function menu_xboxlive(f16_arg0, f16_arg1) menu:addElement(self) end + menu:AddMenuDescription(1) + menu:AddMarketingPanel(LUI.MarketingLocation.Featured, LUI.ComScore.ScreenID.PlayOnline) menu.isSignInMenu = true menu:registerEventHandler("gain_focus", LUI.MPLobbyOnline.OnGainFocus) menu:registerEventHandler("player_joined", luiglobals.Cac.PlayerJoinedEvent) @@ -81,7 +81,16 @@ function menu_xboxlive(f16_arg0, f16_arg1) end end) + menu:AddHelp({ + name = "add_button_helper_text", + button_ref = "", + helper_text = " ", + side = "left", + priority = -9000, + clickable = false + }) + return menu end -LUI.MenuBuilder.m_types_build["menu_xboxlive"] = menu_xboxlive +LUI.MenuBuilder.m_types_build["menu_xboxlive"] = menu_xboxlive \ No newline at end of file diff --git a/data/cdata/ui_scripts/server_list/serverlist.lua b/data/cdata/ui_scripts/server_list/serverlist.lua index e6c80757..bf9525e8 100644 --- a/data/cdata/ui_scripts/server_list/serverlist.lua +++ b/data/cdata/ui_scripts/server_list/serverlist.lua @@ -10,19 +10,23 @@ local columns = {{ text = "@MENU_HOST_NAME", dataindex = 0 }, { - offset = 500, + offset = 440, text = "@MENU_MAP", dataindex = 1 }, { - offset = 725, + offset = 615, text = "@MENU_TYPE1", dataindex = 3 }, { - offset = 920, + offset = 780, + text = "@EXE_SV_INFO_MOD", + dataindex = 6 +}, { + offset = 980, text = "@MENU_NUMPLAYERS", dataindex = 2 }, { - offset = 1070, + offset = 1100, text = "@MENU_PING", dataindex = 4 }, { diff --git a/data/zone_source/h1_mod_common.csv b/data/zone_source/h1_mod_common.csv index b69a5031..a0e9c4f5 100644 --- a/data/zone_source/h1_mod_common.csv +++ b/data/zone_source/h1_mod_common.csv @@ -1,2 +1,3 @@ localize,english +ttf,fonts/bank_h1.ttf ttf,fonts/default.otf \ No newline at end of file diff --git a/data/zonetool/fra_h1_mod_common_mp/localizedstrings/french.json b/data/zonetool/fra_h1_mod_common_mp/localizedstrings/french.json index 1b006bfd..fbabba13 100644 --- a/data/zonetool/fra_h1_mod_common_mp/localizedstrings/french.json +++ b/data/zonetool/fra_h1_mod_common_mp/localizedstrings/french.json @@ -2,7 +2,7 @@ "LUA_MENU_SERVERLIST": "Liste des serveurs", "PLATFORM_SYSTEM_LINK_TITLE": "LISTE DES SERVEURS", "MENU_NUMPLAYERS": "Joueurs", - "MENU_PING": "Latence", + "MENU_PING": "Ping", "SERVERLIST_PLAYER_COUNT": "&&1 Joueurs", "SERVERLIST_SERVER_COUNT": "&&1 Serveurs", @@ -25,8 +25,8 @@ "LUA_MENU_EDIT_STATS": "Modifier les statistiques", "UPDATER_POPUP_NO_UPDATES_AVAILABLE": "Aucune mise à jour disponible", - "UPDATER_POPUP_AVAILABLE_UPDATE_TEXT": "Une mise à jour est disponible,\npoursuivre l'installation ?", + "MENU_CCS_NEW_PATCH_NOTICE": "Une mise à jour est disponible,\npoursuivre l'installation ?", "UPDATER_POPUP_SUCCESSFUL": "Mise à jour réussie", - "UPDATER_POPUP_RESTART_POPUP_TEXT": "La mise à jour nécessite un redémarrage", + "MENU_CCS_RESTART_CONFIRMATION_TEXT": "La mise à jour nécessite un redémarrage", "UPDATER_POPUP_CHECKING_FOR_UPDATES": "Vérification des mises à jour..." } \ No newline at end of file diff --git a/data/zonetool/h1_mod_common/fonts/bank_h1.ttf b/data/zonetool/h1_mod_common/fonts/bank_h1.ttf new file mode 100644 index 00000000..71ffc345 Binary files /dev/null and b/data/zonetool/h1_mod_common/fonts/bank_h1.ttf differ diff --git a/data/zonetool/h1_mod_common/fonts/default.otf b/data/zonetool/h1_mod_common/fonts/default.otf index ad4f12ef..6bdf48ec 100644 Binary files a/data/zonetool/h1_mod_common/fonts/default.otf and b/data/zonetool/h1_mod_common/fonts/default.otf differ diff --git a/data/zonetool/h1_mod_common/localizedstrings/english.json b/data/zonetool/h1_mod_common/localizedstrings/english.json index 6ee0187a..989ede95 100644 --- a/data/zonetool/h1_mod_common/localizedstrings/english.json +++ b/data/zonetool/h1_mod_common/localizedstrings/english.json @@ -31,6 +31,7 @@ "PLATFORM_SYSTEM_LINK_TITLE": "SERVER LIST", "MENU_NUMPLAYERS": "Players", "MENU_PING": "Ping", + "MENU_MOD": "Mod", "SERVERLIST_PLAYER_COUNT": "&&1 Players", "SERVERLIST_SERVER_COUNT": "&&1 Servers", "EXE_SAY": "^3Match^7", @@ -55,9 +56,10 @@ "LUA_MENU_EDIT_STATS": "Edit Stats", "UPDATER_POPUP_NO_UPDATES_AVAILABLE": "No updates available", - "UPDATER_POPUP_AVAILABLE_UPDATE_TEXT": "An update is available, proceed with installation?", + "MENU_CCS_NEW_PATCH_NOTICE": "An update is available, proceed with installation?", + "MENU_DOWNLOAD_AUTOUPDATE_PATCH": " ", "UPDATER_POPUP_SUCCESSFUL": "Update successful", - "UPDATER_POPUP_RESTART_POPUP_TEXT": "Update requires restart", + "MENU_CCS_RESTART_CONFIRMATION_TEXT": "Update requires restart", "UPDATER_POPUP_CHECKING_FOR_UPDATES": "Checking for updates...", "MPHUD_FPS": "FPS: ", @@ -93,5 +95,6 @@ "LOCALE_RUSSIAN": "Russian", "LOCALE_SIMPLIFIED_CHINESE": "Simplified Chinese", "LOCALE_SPANISH": "Spanish", - "LOCALE_TRADITIONAL_CHINESE": "Traditional Chinese" + "LOCALE_TRADITIONAL_CHINESE": "Traditional Chinese", + "MENU_QUIT_TO_DESKTOP": "Quit to Desktop" } \ No newline at end of file diff --git a/data/zonetool/localizedstrings/english.json b/data/zonetool/localizedstrings/english.json index f0b9b8dc..726e5cf0 100644 --- a/data/zonetool/localizedstrings/english.json +++ b/data/zonetool/localizedstrings/english.json @@ -2,5 +2,10 @@ "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Unlock All Missions and Intel", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Cancel Unlock All Missions", "LUA_MENU_CHOOSE_LANGUAGE_DESC": "Choose your language.", - "MENU_APPLY_LANGUAGE_SETTINGS": "Apply language settings?" + "MENU_APPLY_LANGUAGE_SETTINGS": "Apply language settings?", + + "LUA_MENU_DISCORD_REQUEST": "&&1^7 requested to join your game!", + "LUA_MENU_DISCORD_REQUEST_DISCRIMINATOR": "&&1^7#&&2 requested to join your game!", + "LUA_MENU_DISCORD_ACCEPT": "[&&1] Accept", + "LUA_MENU_DISCORD_DENY": "[&&1] Deny" } \ No newline at end of file diff --git a/data/zonetool/localizedstrings/french.json b/data/zonetool/localizedstrings/french.json index f32229e3..d29a48fc 100644 --- a/data/zonetool/localizedstrings/french.json +++ b/data/zonetool/localizedstrings/french.json @@ -9,7 +9,7 @@ "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Débloquer toutes les missions", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Annuler déblocage", - "MENU_MODS": "MODS", + "MENU_MODS": "Mods", "MENU_MODS_DESC": "Charger les mods installés.", "LUA_MENU_MOD_DESC_DEFAULT": "Charger &&1.", "LUA_MENU_MOD_DESC": "&&1\nAuteur: &&2\nVersion: &&3", @@ -22,9 +22,9 @@ "MENU_NO_DONT_ASK": "Non, ne plus me le demander", "UPDATER_POPUP_NO_UPDATES_AVAILABLE": "Aucune mise à jour disponible", - "UPDATER_POPUP_AVAILABLE_UPDATE_TEXT": "Une mise à jour est disponible,\npoursuivre l'installation ?", + "MENU_CCS_NEW_PATCH_NOTICE": "Une mise à jour est disponible,\npoursuivre l'installation ?", "UPDATER_POPUP_SUCCESSFUL": "Mise à jour réussie", - "UPDATER_POPUP_RESTART_POPUP_TEXT": "La mise à jour nécessite un redémarrage", + "MENU_CCS_RESTART_CONFIRMATION_TEXT": "La mise à jour nécessite un redémarrage", "UPDATER_POPUP_CHECKING_FOR_UPDATES": "Vérification des mises à jour...", "MPHUD_FPS": "IPS: ", @@ -45,6 +45,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "Chinois simplifié", "LOCALE_SPANISH": "Espagnol", "LOCALE_TRADITIONAL_CHINESE": "Chinois traditionnel", + "MENU_QUIT_TO_DESKTOP": "Retour au bureau", "LUA_MENU_CHOOSE_LANGUAGE": "Choisissez la langue", "LUA_MENU_CHOOSE_LANGUAGE_DESC": "Choisissez la langue." diff --git a/data/zonetool/localizedstrings/german.json b/data/zonetool/localizedstrings/german.json index ab9a9770..c7d87f34 100644 --- a/data/zonetool/localizedstrings/german.json +++ b/data/zonetool/localizedstrings/german.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "Vereinfachtes Chinesisch", "LOCALE_SPANISH": "Spanisch", "LOCALE_TRADITIONAL_CHINESE": "Traditionelles Chinesisch", + "MENU_QUIT_TO_DESKTOP": "Zum Desktop", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Alle Missionen freischalten", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Freischalten abbrechen" } \ No newline at end of file diff --git a/data/zonetool/localizedstrings/italian.json b/data/zonetool/localizedstrings/italian.json index af0a304d..cd50ace6 100644 --- a/data/zonetool/localizedstrings/italian.json +++ b/data/zonetool/localizedstrings/italian.json @@ -26,5 +26,6 @@ "LOCALE_RUSSIAN": "Russo", "LOCALE_SIMPLIFIED_CHINESE": "Cinese semplificato", "LOCALE_SPANISH": "Spagnolo", - "LOCALE_TRADITIONAL_CHINESE": "Cinese tradizionale" + "LOCALE_TRADITIONAL_CHINESE": "Cinese tradizionale", + "MENU_QUIT_TO_DESKTOP": "Esci e torna al desktop" } \ No newline at end of file diff --git a/data/zonetool/localizedstrings/japanese_partial.json b/data/zonetool/localizedstrings/japanese_partial.json index e9066310..dacb82da 100644 --- a/data/zonetool/localizedstrings/japanese_partial.json +++ b/data/zonetool/localizedstrings/japanese_partial.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "簡体字中国語", "LOCALE_SPANISH": "スペイン語", "LOCALE_TRADITIONAL_CHINESE": "繁体字中国語", + "MENU_QUIT_TO_DESKTOP": "デスクトップに戻る", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "全ミッションをアンロック", "LUA_MENU_CANCEL_UNLOCK_CAPS": "アンロックをキャンセル" diff --git a/data/zonetool/localizedstrings/korean.json b/data/zonetool/localizedstrings/korean.json index 96972e89..00a32abc 100644 --- a/data/zonetool/localizedstrings/korean.json +++ b/data/zonetool/localizedstrings/korean.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "중국어(간체)", "LOCALE_SPANISH": "스페인어", "LOCALE_TRADITIONAL_CHINESE": "중국어(번체)", + "MENU_QUIT_TO_DESKTOP": "데스크탑으로 나가기", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "모든 임무 잠금 해제", "LUA_MENU_CANCEL_UNLOCK_CAPS": "잠금 해제 취소" diff --git a/data/zonetool/localizedstrings/polish.json b/data/zonetool/localizedstrings/polish.json index e8323a37..30318415 100644 --- a/data/zonetool/localizedstrings/polish.json +++ b/data/zonetool/localizedstrings/polish.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "Chiński uproszczony", "LOCALE_SPANISH": "Hiszpański", "LOCALE_TRADITIONAL_CHINESE": "Chiński tradycyjny", + "MENU_QUIT_TO_DESKTOP": "Wyjdź do pulpitu", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Odblokuj wszystkie misje", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Anuluj odblokowanie" diff --git a/data/zonetool/localizedstrings/portuguese.json b/data/zonetool/localizedstrings/portuguese.json index cfcbf86d..aed9edfb 100644 --- a/data/zonetool/localizedstrings/portuguese.json +++ b/data/zonetool/localizedstrings/portuguese.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "Chinês simplificado", "LOCALE_SPANISH": "Español", "LOCALE_TRADITIONAL_CHINESE": "Chinês tradicional", + "MENU_QUIT_TO_DESKTOP": "Sair para a área de trabalho", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Desbloquear todas as missões", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Cancelar desbloqueio" diff --git a/data/zonetool/localizedstrings/russian.json b/data/zonetool/localizedstrings/russian.json index 5ac4fe31..a49b64ba 100644 --- a/data/zonetool/localizedstrings/russian.json +++ b/data/zonetool/localizedstrings/russian.json @@ -11,7 +11,7 @@ "LUA_MENU_CHOOSE_LANGUAGE": "Выбор языка", "LUA_MENU_CHOOSE_LANGUAGE_DESC": "Поменять язык интерфейса и озвучки.", - "MENU_MODS": "МОДЫ", + "MENU_MODS": "Моды", "MENU_MODS_DESC": "Запуск установленных модов.", "LUA_MENU_MOD_DESC_DEFAULT": "Запустить &&1.", "LUA_MENU_MOD_DESC": "&&1\nАвтор: &&2\nВерсия: &&3.", @@ -24,9 +24,9 @@ "MENU_NO_DONT_ASK": "Нет, больше не спрашивать", "UPDATER_POPUP_NO_UPDATES_AVAILABLE": "У вас установлены все последние обновления", - "UPDATER_POPUP_AVAILABLE_UPDATE_TEXT": "Доступно обновление клиента игры,\nначать установку сейчас?", + "MENU_CCS_NEW_PATCH_NOTICE": "Доступно обновление клиента игры,\nначать установку сейчас?", "UPDATER_POPUP_SUCCESSFUL": "Обновление завершено", - "UPDATER_POPUP_RESTART_POPUP_TEXT": "Для применения изменений необходим перезапуск игры", + "MENU_CCS_RESTART_CONFIRMATION_TEXT": "Для применения изменений необходим перезапуск игры", "UPDATER_POPUP_CHECKING_FOR_UPDATES": "Проверка наличия обновлений...", "MPHUD_FPS": "К/С: ", @@ -48,6 +48,7 @@ "LOCALE_SPANISH": "Испанский", "LOCALE_TRADITIONAL_CHINESE": "Китайский традиционный", "LUA_MENU_DOWNLOAD": "Скачать", + "MENU_QUIT_TO_DESKTOP": "Выйти на рабочий стол", "MPUI_MP44": "MP-44", "WEAPON_AT4": "AT4", diff --git a/data/zonetool/localizedstrings/simplified_chinese.json b/data/zonetool/localizedstrings/simplified_chinese.json index 5560337e..069d21a5 100644 --- a/data/zonetool/localizedstrings/simplified_chinese.json +++ b/data/zonetool/localizedstrings/simplified_chinese.json @@ -2,7 +2,7 @@ "LUA_MENU_SERVERLIST": "服务器列表", "PLATFORM_SYSTEM_LINK_TITLE": "服务器列表", "MENU_NUMPLAYERS": "玩家", - "MENU_PING": "Ping", + "MENU_PING": "时延", "LOCALE_ENGLISH": "英语", "LOCALE_ENGLISH_SAFE": "英语 (审查制度)", "LOCALE_FRENCH": "法语", @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "简体中文", "LOCALE_SPANISH": "西班牙语", "LOCALE_TRADITIONAL_CHINESE": "繁体中文", + "MENU_QUIT_TO_DESKTOP": "退至桌面", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "解锁全部任务", "LUA_MENU_CANCEL_UNLOCK_CAPS": "取消解锁" diff --git a/data/zonetool/localizedstrings/spanish.json b/data/zonetool/localizedstrings/spanish.json index bdebeb16..d4d420be 100644 --- a/data/zonetool/localizedstrings/spanish.json +++ b/data/zonetool/localizedstrings/spanish.json @@ -16,6 +16,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "Chino simplificado", "LOCALE_SPANISH": "Español", "LOCALE_TRADITIONAL_CHINESE": "Chino tradicional", + "MENU_QUIT_TO_DESKTOP": "Salir al escritorio", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "Desbloquear todas las misiones", "LUA_MENU_CANCEL_UNLOCK_CAPS": "Cancelar desbloqueo" diff --git a/data/zonetool/localizedstrings/traditional_chinese.json b/data/zonetool/localizedstrings/traditional_chinese.json index 16f3ca3d..7c7ea96f 100644 --- a/data/zonetool/localizedstrings/traditional_chinese.json +++ b/data/zonetool/localizedstrings/traditional_chinese.json @@ -2,7 +2,7 @@ "LUA_MENU_SERVERLIST": "伺服器列表", "PLATFORM_SYSTEM_LINK_TITLE": "伺服器列表", "MENU_NUMPLAYERS": "玩家", - "MENU_PING": "網路延遲", + "MENU_PING": "延遲", "MENU_TYPE1": "類型", "LOCALE_ENGLISH": "英文", "LOCALE_ENGLISH_SAFE": "英文 (審查制度)", @@ -17,6 +17,7 @@ "LOCALE_SIMPLIFIED_CHINESE": "簡體中文", "LOCALE_SPANISH": "西班牙文", "LOCALE_TRADITIONAL_CHINESE": "繁體中文", + "MENU_QUIT_TO_DESKTOP": "返回桌面", "LUA_MENU_CAMPAIGN_UNLOCKED_ALL_TITLE": "解鎖所有任務", "LUA_MENU_CANCEL_UNLOCK_CAPS": "取消解鎖" diff --git a/data/zonetool/rus_h1_mod_common_mp/localizedstrings/russian.json b/data/zonetool/rus_h1_mod_common_mp/localizedstrings/russian.json index b568d035..d2973591 100644 --- a/data/zonetool/rus_h1_mod_common_mp/localizedstrings/russian.json +++ b/data/zonetool/rus_h1_mod_common_mp/localizedstrings/russian.json @@ -117,6 +117,7 @@ "EXE_SV_INFO_FRIENDLY_FIRE": "Огонь по своим", "EXE_SV_INFO_GAMETYPE": "Режим игры", "EXE_SV_INFO_KILLCAM": "Повтор", + "EXE_SV_INFO_MOD": "Мод", "EXE_SV_INFO_NAME": "имя", "EXE_SV_INFO_PASSWORD": "С паролем", "EXE_SV_INFO_PING": "пинг", diff --git a/deps/GSL b/deps/GSL index 6c6111ac..e64c97fc 160000 --- a/deps/GSL +++ b/deps/GSL @@ -1 +1 @@ -Subproject commit 6c6111acb7b5d687ac006969ac96e5b1f21374cd +Subproject commit e64c97fc2cfc11992098bb38eda932de275e3f4d diff --git a/deps/asmjit b/deps/asmjit index 5b5b0b38..7c10a14d 160000 --- a/deps/asmjit +++ b/deps/asmjit @@ -1 +1 @@ -Subproject commit 5b5b0b38775938df4d3779604ff1db60b9a9dcbf +Subproject commit 7c10a14d347879f889c6d11a9398f1d453acc690 diff --git a/deps/curl b/deps/curl index 4ab601d9..102de7aa 160000 --- a/deps/curl +++ b/deps/curl @@ -1 +1 @@ -Subproject commit 4ab601d93a07cee665ec2458a51fccd0767c03f1 +Subproject commit 102de7aa8d5bfc6ed5fe85e89c7b943d0c186f03 diff --git a/deps/discord-rpc b/deps/discord-rpc index 963aa9f3..b3383798 160000 --- a/deps/discord-rpc +++ b/deps/discord-rpc @@ -1 +1 @@ -Subproject commit 963aa9f3e5ce81a4682c6ca3d136cddda614db33 +Subproject commit b3383798b353c31ea6770fee673740c27f6e3489 diff --git a/deps/extra/gsc-tool/interface.cpp b/deps/extra/gsc-tool/interface.cpp deleted file mode 100644 index 28b7ad73..00000000 --- a/deps/extra/gsc-tool/interface.cpp +++ /dev/null @@ -1,30 +0,0 @@ -#include "stdafx.hpp" - -#include - -#include "interface.hpp" - -namespace gsc -{ - std::unique_ptr compiler() - { - auto compiler = std::make_unique(); - compiler->mode(xsk::gsc::build::prod); - return compiler; - } - - std::unique_ptr decompiler() - { - return std::make_unique(); - } - - std::unique_ptr assembler() - { - return std::make_unique(); - } - - std::unique_ptr disassembler() - { - return std::make_unique(); - } -} diff --git a/deps/extra/gsc-tool/interface.hpp b/deps/extra/gsc-tool/interface.hpp deleted file mode 100644 index 133e6ae2..00000000 --- a/deps/extra/gsc-tool/interface.hpp +++ /dev/null @@ -1,9 +0,0 @@ -#pragma once - -namespace gsc -{ - std::unique_ptr compiler(); - std::unique_ptr decompiler(); - std::unique_ptr assembler(); - std::unique_ptr disassembler(); -} diff --git a/deps/gsc-tool b/deps/gsc-tool index 7d374025..cbfcce1d 160000 --- a/deps/gsc-tool +++ b/deps/gsc-tool @@ -1 +1 @@ -Subproject commit 7d374025b7675bada64c247ebe9378dd335a33da +Subproject commit cbfcce1dd67534c2115331e41d4cb6893e96196c diff --git a/deps/json b/deps/json index 4c6cde72..a259ecc5 160000 --- a/deps/json +++ b/deps/json @@ -1 +1 @@ -Subproject commit 4c6cde72e533158e044252718c013a48bcff346c +Subproject commit a259ecc51e1951e12f757ce17db958e9881e9c6c diff --git a/deps/libtomcrypt b/deps/libtomcrypt index 29986d04..7e863d21 160000 --- a/deps/libtomcrypt +++ b/deps/libtomcrypt @@ -1 +1 @@ -Subproject commit 29986d04f2dca985ee64fbca1c7431ea3e3422f4 +Subproject commit 7e863d21429f94ed6a720e24499a12a3f852bb31 diff --git a/deps/libtommath b/deps/libtommath index 03de03de..8314bde5 160000 --- a/deps/libtommath +++ b/deps/libtommath @@ -1 +1 @@ -Subproject commit 03de03dee753442d4b23166982514639c4ccbc39 +Subproject commit 8314bde5e5c8e5d9331460130a9d1066e324f091 diff --git a/deps/lua b/deps/lua index be908a7d..7923dbbf 160000 --- a/deps/lua +++ b/deps/lua @@ -1 +1 @@ -Subproject commit be908a7d4d8130264ad67c5789169769f824c5d1 +Subproject commit 7923dbbf72da303ca1cca17efd24725668992f15 diff --git a/deps/minhook b/deps/minhook index 49d03ad1..f5485b84 160000 --- a/deps/minhook +++ b/deps/minhook @@ -1 +1 @@ -Subproject commit 49d03ad118cf7f6768c79a8f187e14b8f2a07f94 +Subproject commit f5485b8454544c2f034c78f8f127c1d03dea3636 diff --git a/deps/premake/gsc-tool.lua b/deps/premake/gsc-tool.lua index 08da07de..87cc40c5 100644 --- a/deps/premake/gsc-tool.lua +++ b/deps/premake/gsc-tool.lua @@ -1,5 +1,5 @@ gsc_tool = { - source = path.join(dependencies.basePath, "gsc-tool/src") + source = path.join(dependencies.basePath, "gsc-tool") } function gsc_tool.import() @@ -9,60 +9,54 @@ end function gsc_tool.includes() includedirs { - path.join(gsc_tool.source, "utils"), - path.join(gsc_tool.source, "h1"), - path.join(dependencies.basePath, "extra/gsc-tool") -- https://github.com/GEEKiDoS/open-teknomw3/blob/master/deps/extra/gsc-tool + path.join(gsc_tool.source, "include") } end --- https://github.com/xensik/gsc-tool/blob/dev/premake5.lua#L95 function gsc_tool.project() project "xsk-gsc-utils" - kind "StaticLib" - language "C++" + kind "StaticLib" + language "C++" + warnings "Off" - pchheader "stdafx.hpp" - pchsource(path.join(gsc_tool.source, "utils/stdafx.cpp")) + files { + path.join(gsc_tool.source, "include/xsk/utils/*.hpp"), + path.join(gsc_tool.source, "src/utils/*.cpp") + } - files { - path.join(gsc_tool.source, "utils/**.h"), - path.join(gsc_tool.source, "utils/**.hpp"), - path.join(gsc_tool.source, "utils/**.cpp") - } + includedirs { + path.join(gsc_tool.source, "include") + } - includedirs { - path.join(gsc_tool.source, "utils"), - gsc_tool.source - } - - zlib.includes() + zlib.includes() project "xsk-gsc-h1" - kind "StaticLib" - language "C++" + kind "StaticLib" + language "C++" + warnings "Off" - pchheader "stdafx.hpp" - pchsource(path.join(gsc_tool.source, "h1/stdafx.cpp")) - - files { - path.join(gsc_tool.source, "h1/**.h"), - path.join(gsc_tool.source, "h1/**.hpp"), - path.join(gsc_tool.source, "h1/**.cpp"), - path.join(dependencies.basePath, "extra/gsc-tool/interface.cpp") - } - - includedirs { - path.join(gsc_tool.source, "h1"), - gsc_tool.source, - path.join(dependencies.basePath, "extra/gsc-tool") - } - - -- https://github.com/xensik/gsc-tool/blob/dev/premake5.lua#L25 - -- adding these build options fixes a bunch of parser stuff filter "action:vs*" - buildoptions "/bigobj" buildoptions "/Zc:__cplusplus" filter {} + + files { + path.join(gsc_tool.source, "include/xsk/stdinc.hpp"), + + path.join(gsc_tool.source, "include/xsk/gsc/engine/h1.hpp"), + path.join(gsc_tool.source, "src/gsc/engine/h1.cpp"), + + path.join(gsc_tool.source, "src/gsc/engine/h1_code.cpp"), + path.join(gsc_tool.source, "src/gsc/engine/h1_func.cpp"), + path.join(gsc_tool.source, "src/gsc/engine/h1_meth.cpp"), + path.join(gsc_tool.source, "src/gsc/engine/h1_token.cpp"), path.join(gsc_tool.source, "src/gsc/*.cpp"), + + path.join(gsc_tool.source, "src/gsc/common/*.cpp"), + path.join(gsc_tool.source, "include/xsk/gsc/common/*.hpp") + } + + includedirs { + path.join(gsc_tool.source, "include") + } end table.insert(dependencies, gsc_tool) diff --git a/deps/protobuf b/deps/protobuf index 7ce9c415..5a3dac89 160000 --- a/deps/protobuf +++ b/deps/protobuf @@ -1 +1 @@ -Subproject commit 7ce9c415455c098409222702b3b4572b47232882 +Subproject commit 5a3dac894157bf3618b2c906a8b9073b4cad62b6 diff --git a/deps/rapidjson b/deps/rapidjson index a98e9999..6089180e 160000 --- a/deps/rapidjson +++ b/deps/rapidjson @@ -1 +1 @@ -Subproject commit a98e99992bd633a2736cc41f96ec85ef0c50e44d +Subproject commit 6089180ecb704cb2b136777798fa1be303618975 diff --git a/deps/sol2 b/deps/sol2 index f81643aa..9c882a28 160000 --- a/deps/sol2 +++ b/deps/sol2 @@ -1 +1 @@ -Subproject commit f81643aa0c0c507c0cd8400b8cfedc74a34a19f6 +Subproject commit 9c882a28fdb6f4ad79a53a4191b43ce48a661175 diff --git a/deps/stb b/deps/stb index 8b5f1f37..f4a71b13 160000 --- a/deps/stb +++ b/deps/stb @@ -1 +1 @@ -Subproject commit 8b5f1f37b5b75829fc72d38e7b5d4bcbf8a26d55 +Subproject commit f4a71b13373436a2866c5d68f8f80ac6f0bc1ffe diff --git a/deps/zlib b/deps/zlib index e5546956..643e17b7 160000 --- a/deps/zlib +++ b/deps/zlib @@ -1 +1 @@ -Subproject commit e554695638228b846d49657f31eeff0ca4680e8a +Subproject commit 643e17b7498d12ab8d15565662880579692f769d diff --git a/src/client/component/branding.cpp b/src/client/component/branding.cpp index 342b8b0f..8e6b6512 100644 --- a/src/client/component/branding.cpp +++ b/src/client/component/branding.cpp @@ -31,20 +31,30 @@ namespace branding void draw_branding() { const auto font = game::R_RegisterFont("fonts/fira_mono_bold.ttf", 22); - if (font) + if (!font) { -#ifdef DEBUG - game::R_AddCmdDrawText("h1-mod: " VERSION " (" __DATE__ " " __TIME__ ")", - 0x7FFFFFFF, font, 10.f, - 5.f + static_cast(font->pixelHeight), - 1.f, 1.f, 0.0f, color, 0); -#else - game::R_AddCmdDrawText("h1-mod", - 0x7FFFFFFF, font, 10.f, - 5.f + static_cast(font->pixelHeight), - 1.f, 1.f, 0.0f, color, 0); -#endif + return; } + +#ifdef DEBUG + const auto text = "h1-mod: " VERSION " (" __DATE__ " " __TIME__ ")"; +#else + const auto text = "h1-mod: " VERSION; +#endif + + const auto placement = game::ScrPlace_GetViewPlacement(); + float text_color[4] = {0.6f, 0.6f, 0.6f, 0.6f}; + + game::rectDef_s rect{}; + rect.x = 0; + rect.y = 0; + rect.w = 500; + rect.horzAlign = 0; + rect.vertAlign = 0; + + game::rectDef_s text_rect{}; + + game::UI_DrawWrappedText(placement, text, &rect, font, -102.5f, 10.f, 0.17f, text_color, 0, 0, &text_rect, 0); } } diff --git a/src/client/component/command.cpp b/src/client/component/command.cpp index 58c0918d..b3b576c0 100644 --- a/src/client/component/command.cpp +++ b/src/client/component/command.cpp @@ -4,11 +4,12 @@ #include "command.hpp" #include "console.hpp" #include "dvars.hpp" -#include "game_console.hpp" #include "fastfiles.hpp" #include "filesystem.hpp" -#include "scheduler.hpp" +#include "game_console.hpp" #include "logfile.hpp" +#include "mods.hpp" +#include "scheduler.hpp" #include "game/game.hpp" #include "game/dvars.hpp" @@ -28,8 +29,6 @@ namespace command std::unordered_map> handlers; std::unordered_map> handlers_sv; - std::optional saved_fs_game; - void main_handler() { params params = {}; @@ -557,43 +556,11 @@ namespace command } } - void register_fs_game_path() - { - const auto* fs_game = game::Dvar_FindVar("fs_game"); - const auto new_mod_path = fs_game->current.string; - - // check if the last saved fs_game value isn't empty and if it doesn't equal the new fs_game - if (saved_fs_game.has_value() && saved_fs_game != new_mod_path) - { - // unregister path to be used as a fs directory - filesystem::unregister_path(saved_fs_game.value()); - } - - if (new_mod_path && !new_mod_path[0]) - { - return; - } - - // register fs_game value as a fs directory used for many things - filesystem::register_path(new_mod_path); - saved_fs_game = new_mod_path; - } - class component final : public component_interface { public: void post_unpack() override { - // it might be overdone to change the filesystem path on every new value change, but to be fair, - // for the mods that don't need full restarts, this is good because it'll adjust and work like so - // in my opinion, this is fine. if a user tries to modify the dvar themselves, they'll have problems - // but i seriously doubt it'll be bad. - dvars::callback::on_new_value("fs_game", []() - { - console::warn("fs_game value changed, filesystem paths will be adjusted to new dvar value."); - register_fs_game_path(); - }); - if (game::environment::is_sp()) { add_commands_sp(); @@ -702,7 +669,6 @@ namespace command const auto name = params.get(1); const auto dvar = game::Dvar_FindVar(name); - if (dvar == nullptr) { console::info("%s doesn't exist\n", name); diff --git a/src/client/component/command.hpp b/src/client/component/command.hpp index a891b802..06228c9d 100644 --- a/src/client/component/command.hpp +++ b/src/client/component/command.hpp @@ -49,6 +49,4 @@ namespace command void add_sv(const char* name, std::function callback); void execute(std::string command, bool sync = false); - - void register_fs_game_path(); } \ No newline at end of file diff --git a/src/client/component/dedicated_info.cpp b/src/client/component/dedicated_info.cpp index 8164345b..d8fbb1d6 100644 --- a/src/client/component/dedicated_info.cpp +++ b/src/client/component/dedicated_info.cpp @@ -21,7 +21,7 @@ namespace dedicated_info scheduler::loop([] { - auto* sv_running = game::Dvar_FindVar("sv_running"); + const auto sv_running = game::Dvar_FindVar("sv_running"); if (!sv_running || !sv_running->current.enabled || (*game::mp::svs_clients) == nullptr) { SetConsoleTitle("H1-Mod Dedicated Server"); diff --git a/src/client/component/discord.cpp b/src/client/component/discord.cpp index 7677662f..689d0006 100644 --- a/src/client/component/discord.cpp +++ b/src/client/component/discord.cpp @@ -5,7 +5,6 @@ #include "command.hpp" #include "discord.hpp" #include "materials.hpp" -#include "network.hpp" #include "party.hpp" #include "scheduler.hpp" @@ -28,119 +27,181 @@ namespace discord { namespace { - DiscordRichPresence discord_presence; - - void update_discord() + struct discord_presence_state_t { - if (!game::CL_IsCgameInitialized() || game::VirtualLobby_Loaded()) + int start_timestamp; + int party_size; + int party_max; + }; + + struct discord_presence_strings_t + { + std::string state; + std::string details; + std::string small_image_key; + std::string small_image_text; + std::string large_image_key; + std::string large_image_text; + std::string party_id; + std::string join_secret; + }; + + DiscordRichPresence discord_presence{}; + discord_presence_strings_t discord_strings; + + std::mutex avatar_map_mutex; + std::unordered_map avatar_material_map; + game::Material* default_avatar_material{}; + + void update_discord_frontend() + { + discord_presence.details = SELECT_VALUE("Singleplayer", "Multiplayer"); + discord_presence.startTimestamp = 0; + + static const auto in_firing_range = game::Dvar_FindVar("virtualLobbyInFiringRange"); + if (in_firing_range != nullptr && in_firing_range->current.enabled == 1) { - discord_presence.details = SELECT_VALUE("Singleplayer", "Multiplayer"); - discord_presence.state = "Main Menu"; - - const auto in_firing_range = game::Dvar_FindVar("virtualLobbyInFiringRange"); - if (in_firing_range && in_firing_range->current.enabled == 1) - { - discord_presence.state = "Firing Range"; - } - - discord_presence.partySize = 0; - discord_presence.partyMax = 0; - discord_presence.startTimestamp = 0; - discord_presence.largeImageKey = SELECT_VALUE("menu_singleplayer", "menu_multiplayer"); - - // set to blank when in lobby - discord_presence.matchSecret = ""; - discord_presence.joinSecret = ""; - discord_presence.partyId = ""; + discord_presence.state = "Firing Range"; + discord_presence.largeImageKey = "mp_vlobby_room"; } else { - static char details[0x80] = {0}; - const auto map = game::Dvar_FindVar("mapname")->current.string; - const auto key = utils::string::va("PRESENCE_%s%s", SELECT_VALUE("SP_", ""), map); - const char* mapname = map; - - if (game::DB_XAssetExists(game::ASSET_TYPE_LOCALIZE, key) && !game::DB_IsXAssetDefault(game::ASSET_TYPE_LOCALIZE, key)) - { - mapname = game::UI_SafeTranslateString(key); - } - - if (game::environment::is_mp()) - { - static char clean_gametype[0x80] = {0}; - const auto gametype = game::UI_GetGameTypeDisplayName( - game::Dvar_FindVar("g_gametype")->current.string); - utils::string::strip(gametype, - clean_gametype, sizeof(clean_gametype)); - strcpy_s(details, 0x80, utils::string::va("%s on %s", clean_gametype, mapname)); - - static char clean_hostname[0x80] = {0}; - utils::string::strip(game::Dvar_FindVar("sv_hostname")->current.string, - clean_hostname, sizeof(clean_hostname)); - auto max_clients = party::server_client_count(); - - if (game::SV_Loaded()) - { - strcpy_s(clean_hostname, "Private Match"); - max_clients = game::Dvar_FindVar("sv_maxclients")->current.integer; - discord_presence.partyPrivacy = DISCORD_PARTY_PRIVATE; - } - else - { - const auto server_net_info = party::get_state_host(); - const auto server_ip_port = utils::string::va("%i.%i.%i.%i:%i", - static_cast(server_net_info.ip[0]), - static_cast(server_net_info.ip[1]), - static_cast(server_net_info.ip[2]), - static_cast(server_net_info.ip[3]), - static_cast(ntohs(server_net_info.port)) - ); - - static char join_secret[0x80] = {0}; - strcpy_s(join_secret, 0x80, server_ip_port); - - static char party_id[0x80] = {0}; - const auto server_ip_port_hash = utils::cryptography::sha1::compute(server_ip_port, true).substr(0, 8); - strcpy_s(party_id, 0x80, server_ip_port_hash.data()); - - discord_presence.partyId = party_id; - discord_presence.joinSecret = join_secret; - discord_presence.partyPrivacy = DISCORD_PARTY_PUBLIC; - } - - const auto client_state = *game::mp::client_state; - if (client_state != nullptr) - { - discord_presence.partySize = client_state->num_players; - } - else - { - discord_presence.partySize = 0; - } - - discord_presence.partyMax = max_clients; - discord_presence.state = clean_hostname; - discord_presence.largeImageKey = map; - } - else if (game::environment::is_sp()) - { - discord_presence.state = ""; - discord_presence.largeImageKey = map; - strcpy_s(details, 0x80, mapname); - } - - discord_presence.details = details; - - if (!discord_presence.startTimestamp) - { - discord_presence.startTimestamp = std::chrono::duration_cast( - std::chrono::system_clock::now().time_since_epoch()).count(); - } + discord_presence.state = "Main Menu"; + discord_presence.largeImageKey = SELECT_VALUE("menu_singleplayer", "menu_multiplayer"); } Discord_UpdatePresence(&discord_presence); } + void update_discord_ingame() + { + static const auto mapname_dvar = game::Dvar_FindVar("mapname"); + auto mapname = mapname_dvar->current.string; + + discord_strings.large_image_key = mapname; + + const auto presence_key = utils::string::va("PRESENCE_%s%s", SELECT_VALUE("SP_", ""), mapname); + if (game::DB_XAssetExists(game::ASSET_TYPE_LOCALIZE, presence_key) && + !game::DB_IsXAssetDefault(game::ASSET_TYPE_LOCALIZE, presence_key)) + { + mapname = game::UI_SafeTranslateString(presence_key); + } + + if (game::environment::is_mp()) + { + static const auto gametype_dvar = game::Dvar_FindVar("g_gametype"); + static const auto max_clients_dvar = game::Dvar_FindVar("sv_maxclients"); + static const auto hostname_dvar = game::Dvar_FindVar("sv_hostname"); + + const auto gametype_display_name = game::UI_GetGameTypeDisplayName(gametype_dvar->current.string); + const auto gametype = utils::string::strip(gametype_display_name); + + discord_strings.details = std::format("{} on {}", gametype, mapname); + + const auto client_state = *game::mp::client_state; + if (client_state != nullptr) + { + discord_presence.partySize = client_state->num_players; + } + + if (game::SV_Loaded()) + { + discord_strings.state = "Private Match"; + discord_presence.partyMax = max_clients_dvar->current.integer; + discord_presence.partyPrivacy = DISCORD_PARTY_PRIVATE; + } + else + { + discord_strings.state = utils::string::strip(hostname_dvar->current.string); + + const auto server_connection_state = party::get_server_connection_state(); + const auto server_ip_port = std::format("{}.{}.{}.{}:{}", + static_cast(server_connection_state.host.ip[0]), + static_cast(server_connection_state.host.ip[1]), + static_cast(server_connection_state.host.ip[2]), + static_cast(server_connection_state.host.ip[3]), + static_cast(ntohs(server_connection_state.host.port)) + ); + + discord_strings.party_id = utils::cryptography::sha1::compute(server_ip_port, true).substr(0, 8); + discord_presence.partyMax = server_connection_state.max_clients; + discord_presence.partyPrivacy = DISCORD_PARTY_PUBLIC; + discord_strings.join_secret = server_ip_port; + } + + auto server_discord_info = party::get_server_discord_info(); + if (server_discord_info.has_value()) + { + discord_strings.small_image_key = server_discord_info->image; + discord_strings.small_image_text = server_discord_info->image_text; + } + } + else if (game::environment::is_sp()) + { + discord_strings.details = mapname; + } + + if (discord_presence.startTimestamp == 0) + { + discord_presence.startTimestamp = std::chrono::duration_cast( + std::chrono::system_clock::now().time_since_epoch()).count(); + } + + discord_presence.state = discord_strings.state.data(); + discord_presence.details = discord_strings.details.data(); + discord_presence.smallImageKey = discord_strings.small_image_key.data(); + discord_presence.smallImageText = discord_strings.small_image_text.data(); + discord_presence.largeImageKey = discord_strings.large_image_key.data(); + discord_presence.largeImageText = discord_strings.large_image_text.data(); + discord_presence.partyId = discord_strings.party_id.data(); + discord_presence.joinSecret = discord_strings.join_secret.data(); + + Discord_UpdatePresence(&discord_presence); + } + + void update_discord() + { + const auto saved_time = discord_presence.startTimestamp; + discord_presence = {}; + discord_presence.startTimestamp = saved_time; + + if (!game::CL_IsCgameInitialized() || game::VirtualLobby_Loaded()) + { + update_discord_frontend(); + } + else + { + update_discord_ingame(); + } + } + + game::Material* create_avatar_material(const std::string& name, const std::string& data) + { + const auto material = materials::create_material(name); + try + { + if (!materials::setup_material_image(material, data)) + { + materials::free_material(material); + return nullptr; + } + + { + std::lock_guard _0(avatar_map_mutex); + avatar_material_map.insert(std::make_pair(name, material)); + } + + return material; + } + catch (const std::exception& e) + { + materials::free_material(material); + console::error("Failed to load user avatar image: %s\n", e.what()); + } + + return nullptr; + } + void download_user_avatar(const std::string& id, const std::string& avatar) { const auto data = utils::http::get_data( @@ -156,10 +217,10 @@ namespace discord return; } - materials::add(utils::string::va(AVATAR, id.data()), value.buffer); + const auto name = utils::string::va(AVATAR, id.data()); + create_avatar_material(name, value.buffer); } - bool has_default_avatar = false; void download_default_avatar() { const auto data = utils::http::get_data(DEFAULT_AVATAR_URL); @@ -174,25 +235,146 @@ namespace discord return; } - has_default_avatar = true; - materials::add(DEFAULT_AVATAR, value.buffer); + default_avatar_material = create_avatar_material(DEFAULT_AVATAR, value.buffer); + } + + void ready(const DiscordUser* request) + { + DiscordRichPresence presence{}; + presence.instance = 1; + presence.state = ""; + console::info("Discord: Ready on %s (%s)\n", request->username, request->userId); + Discord_UpdatePresence(&presence); + } + + void errored(const int error_code, const char* message) + { + console::error("Discord: %s (%i)\n", message, error_code); + } + + void join_game(const char* join_secret) + { + console::debug("Discord: join_game called with secret '%s'\n", join_secret); + + scheduler::once([=] + { + game::netadr_s target{}; + if (game::NET_StringToAdr(join_secret, &target)) + { + console::info("Discord: Connecting to server '%s'\n", join_secret); + party::connect(target); + } + }, scheduler::pipeline::main); + } + + std::string get_display_name(const DiscordUser* user) + { + if (user->discriminator != nullptr && user->discriminator != "0"s) + { + return std::format("{}#{}", user->username, user->discriminator); + } + else if (user->globalName[0] != 0) + { + return user->globalName; + } + else + { + return user->username; + } + } + + void join_request(const DiscordUser* request) + { + console::debug("Discord: Join request from %s (%s)\n", request->username, request->userId); + + if (game::Com_InFrontend() || !ui_scripting::lui_running()) + { + Discord_Respond(request->userId, DISCORD_REPLY_IGNORE); + return; + } + + static std::unordered_map last_requests; + + const std::string user_id = request->userId; + const std::string avatar = request->avatar; + const std::string discriminator = request->discriminator; + const std::string username = request->username; + const auto display_name = get_display_name(request); + + const auto now = std::chrono::high_resolution_clock::now(); + auto iter = last_requests.find(user_id); + if (iter != last_requests.end()) + { + if ((now - iter->second) < 15s) + { + return; + } + else + { + iter->second = now; + } + } + else + { + last_requests.insert(std::make_pair(user_id, now)); + } + + scheduler::once([=] + { + const ui_scripting::table request_table{}; + request_table.set("avatar", avatar); + request_table.set("discriminator", discriminator); + request_table.set("userid", user_id); + request_table.set("username", username); + request_table.set("displayname", display_name); + + ui_scripting::notify("discord_join_request", + { + {"request", request_table} + }); + }, scheduler::pipeline::lui); + + const auto material_name = utils::string::va(AVATAR, user_id.data()); + if (!avatar.empty() && !avatar_material_map.contains(material_name)) + { + download_user_avatar(user_id, avatar); + } + } + + void set_default_bindings() + { + const auto set_binding = [](const std::string& command, const game::keyNum_t key) + { + const auto binding = game::Key_GetBindingForCmd(command.data()); + for (auto i = 0; i < 256; i++) + { + if (game::playerKeys[0].keys[i].binding == binding) + { + return; + } + } + + if (game::playerKeys[0].keys[key].binding == 0) + { + game::Key_SetBinding(0, key, binding); + } + }; + + set_binding("discord_accept", game::K_F1); + set_binding("discord_deny", game::K_F2); } } - std::string get_avatar_material(const std::string& id) + game::Material* get_avatar_material(const std::string& id) { - const auto avatar_name = utils::string::va(AVATAR, id.data()); - if (materials::exists(avatar_name)) + const auto material_name = utils::string::va(AVATAR, id.data()); + const auto iter = avatar_material_map.find(material_name); + if (iter == avatar_material_map.end()) { - return avatar_name; + return default_avatar_material; } - if (has_default_avatar) - { - return DEFAULT_AVATAR; - } - - return "black"; + return iter->second; } void respond(const std::string& id, int reply) @@ -206,15 +388,14 @@ namespace discord class component final : public component_interface { public: - void post_load() override + void post_unpack() override { if (game::environment::is_dedi()) { return; } - DiscordEventHandlers handlers; - ZeroMemory(&handlers, sizeof(handlers)); + DiscordEventHandlers handlers{}; handlers.ready = ready; handlers.errored = errored; handlers.disconnected = errored; @@ -233,16 +414,29 @@ namespace discord Discord_Initialize("947125042930667530", &handlers, 1, nullptr); - scheduler::once(download_default_avatar, scheduler::pipeline::async); - - scheduler::once([] + if (game::environment::is_mp()) { - scheduler::once(update_discord, scheduler::pipeline::async); - scheduler::loop(update_discord, scheduler::pipeline::async, 5s); - scheduler::loop(Discord_RunCallbacks, scheduler::pipeline::async, 1s); - }, scheduler::pipeline::main); + scheduler::on_game_initialized([] + { + scheduler::once(download_default_avatar, scheduler::async); + set_default_bindings(); + }, scheduler::main); + } + + scheduler::loop(Discord_RunCallbacks, scheduler::async, 500ms); + scheduler::loop(update_discord, scheduler::async, 5s); initialized_ = true; + + command::add("discord_accept", []() + { + ui_scripting::notify("discord_response", {{"accept", true}}); + }); + + command::add("discord_deny", []() + { + ui_scripting::notify("discord_response", {{"accept", false}}); + }); } void pre_destroy() override @@ -257,67 +451,6 @@ namespace discord private: bool initialized_ = false; - - static void ready(const DiscordUser* request) - { - ZeroMemory(&discord_presence, sizeof(discord_presence)); - discord_presence.instance = 1; - console::info("Discord: Ready on %s (%s)\n", request->username, request->userId); - Discord_UpdatePresence(&discord_presence); - } - - static void errored(const int error_code, const char* message) - { - console::error("Discord: Error (%i): %s\n", error_code, message); - } - - static void join_game(const char* join_secret) - { - scheduler::once([=] - { - game::netadr_s target{}; - if (game::NET_StringToAdr(join_secret, &target)) - { - console::info("Discord: Connecting to server: %s\n", join_secret); - party::connect(target); - } - }, scheduler::pipeline::main); - } - - static void join_request(const DiscordUser* request) - { - console::info("Discord: Join request from %s (%s)\n", request->username, request->userId); - - if (game::Com_InFrontend() || !ui_scripting::lui_running()) - { - Discord_Respond(request->userId, DISCORD_REPLY_IGNORE); - return; - } - - std::string user_id = request->userId; - std::string avatar = request->avatar; - std::string discriminator = request->discriminator; - std::string username = request->username; - - scheduler::once([=] - { - const ui_scripting::table request_table{}; - request_table.set("avatar", avatar); - request_table.set("discriminator", discriminator); - request_table.set("userid", user_id); - request_table.set("username", username); - - ui_scripting::notify("discord_join_request", - { - {"request", request_table} - }); - }, scheduler::pipeline::lui); - - if (!materials::exists(utils::string::va(AVATAR, user_id.data()))) - { - download_user_avatar(user_id, avatar); - } - } }; } diff --git a/src/client/component/discord.hpp b/src/client/component/discord.hpp index 5399f952..54c75ca9 100644 --- a/src/client/component/discord.hpp +++ b/src/client/component/discord.hpp @@ -1,7 +1,9 @@ #pragma once +#include "game/game.hpp" + namespace discord { - std::string get_avatar_material(const std::string& id); + game::Material* get_avatar_material(const std::string& id); void respond(const std::string& id, int reply); } diff --git a/src/client/component/download.cpp b/src/client/component/download.cpp index 9d4ce938..18e152ee 100644 --- a/src/client/component/download.cpp +++ b/src/client/component/download.cpp @@ -8,6 +8,8 @@ #include "game/ui_scripting/execution.hpp" +#include "utils/hash.hpp" + #include #include #include @@ -23,10 +25,16 @@ namespace download bool active{}; }; + std::atomic_bool kill_downloads = false; utils::concurrency::container globals; bool download_aborted() { + if (kill_downloads) + { + return true; + } + return globals.access([](globals_t& globals_) { return globals_.abort; @@ -159,10 +167,10 @@ namespace download const auto url = utils::string::va("%s/%s", base.data(), file.name.data()); console::debug("Downloading %s from %s: %s\n", file.name.data(), base.data(), url); - const auto data = utils::http::get_data(url, {}, {}, &progress_callback); + auto data = utils::http::get_data(url, {}, {}, &progress_callback); if (!data.has_value()) { - menu_error("Download failed: An unknown error occurred, please try again."); + menu_error(utils::string::va("Download failed: An unknown error occurred when getting data from '%s', please try again.", url)); return; } @@ -171,9 +179,16 @@ namespace download return; } - const auto& result = data.value(); + auto& result = data.value(); if (result.code != CURLE_OK) { + if (result.code == CURLE_COULDNT_CONNECT) + { + menu_error(utils::string::va("Download failed: Couldn't connect to server '%s' (%i)\n", + url, result.code)); + return; + } + menu_error(utils::string::va("Download failed: %s (%i)\n", curl_easy_strerror(result.code), result.code)); return; @@ -181,15 +196,15 @@ namespace download if (result.response_code >= 400) { - menu_error(utils::string::va("Download failed: Server returned bad response code %i\n", + menu_error(utils::string::va("Download failed: Server returned bad response code (%i)\n", result.response_code)); return; } - const auto hash = utils::cryptography::sha1::compute(result.buffer, true); + const auto hash = utils::hash::get_buffer_hash(result.buffer, file.name); if (hash != file.hash) { - menu_error(utils::string::va("Download failed: file hash doesn't match the server's (%s: %s != %s)\n", + menu_error(utils::string::va("Download failed: File hash doesn't match the server's (%s: %s != %s)\n", file.name.data(), hash.data(), file.hash.data())); return; } @@ -225,7 +240,18 @@ namespace download scheduler::once([] { ui_scripting::notify("mod_download_done", {}); - party::menu_error("Download for server mod has been cancelled."); + party::menu_error("Download failed: Aborted"); }, scheduler::pipeline::lui); } + + class component final : public component_interface + { + public: + void pre_destroy() override + { + kill_downloads = true; + } + }; } + +REGISTER_COMPONENT(download::component) diff --git a/src/client/component/dvars.cpp b/src/client/component/dvars.cpp index 3fec1fd4..d6cfc5aa 100644 --- a/src/client/component/dvars.cpp +++ b/src/client/component/dvars.cpp @@ -2,8 +2,10 @@ #include "loader/component_loader.hpp" #include "dvars.hpp" +#include "console.hpp" #include "game/game.hpp" +#include "game/dvars.hpp" #include @@ -255,13 +257,13 @@ namespace dvars namespace callback { - static std::unordered_map> new_value_callbacks; + static std::unordered_map> dvar_new_value_callbacks; static std::unordered_map> dvar_on_register_function_map; - void on_new_value(const std::string& name, const std::function callback) + void on_new_value(const std::string& name, const std::function callback) { - new_value_callbacks[game::generateHashValue(name.data())] = callback; + dvar_new_value_callbacks[game::generateHashValue(name.data())] = callback; } void on_register(const std::string& name, const std::function& callback) @@ -532,15 +534,34 @@ namespace dvars { dvar_set_variant_hook.invoke(dvar, value, source); - if (callback::new_value_callbacks.find(dvar->hash) != callback::new_value_callbacks.end()) + if (callback::dvar_new_value_callbacks.contains(dvar->hash)) { - callback::new_value_callbacks[dvar->hash](); + callback::dvar_new_value_callbacks[dvar->hash](value); } } class component final : public component_interface { public: + void post_start() override + { + try + { + const auto list_json = utils::nt::load_resource(DVAR_LIST); + const auto list = nlohmann::json::parse(list_json); + for (const auto& [_0, dvar_info] : list.items()) + { + const auto name = dvar_info[0].get(); + const auto description = dvar_info[1].get(); + dvars::insert_dvar_info(name, description); + } + } + catch (const std::exception& e) + { + console::error("Failed to parse dvar list: %s\n", e.what()); + } + } + void post_unpack() override { dvar_register_bool_hook.create(SELECT_VALUE(0x419220_b, 0x182340_b), &dvar_register_bool); diff --git a/src/client/component/dvars.hpp b/src/client/component/dvars.hpp index 3c9831c8..707f229a 100644 --- a/src/client/component/dvars.hpp +++ b/src/client/component/dvars.hpp @@ -1,5 +1,7 @@ #pragma once +#include "game/game.hpp" + namespace dvars { namespace disable @@ -29,7 +31,7 @@ namespace dvars namespace callback { - void on_new_value(const std::string& name, const std::function callback); + void on_new_value(const std::string& name, const std::function callback); void on_register(const std::string& name, const std::function& callback); } diff --git a/src/client/component/experimental.cpp b/src/client/component/experimental.cpp new file mode 100644 index 00000000..e9fc4d21 --- /dev/null +++ b/src/client/component/experimental.cpp @@ -0,0 +1,30 @@ +#include +#include "loader/component_loader.hpp" + +#include "dvars.hpp" +#include "scheduler.hpp" + +#include "game/game.hpp" +#include "game/dvars.hpp" + +#include + +namespace experimental +{ + class component final : public component_interface + { + public: + void post_unpack() override + { + // fix static model's lighting going black sometimes + //dvars::override::register_int("r_smodelInstancedThreshold", 0, 0, 128, 0x0); + + // change minimum cap to -2000 instead of -1000 (culling issue) + dvars::override::register_float("r_lodBiasRigid", 0, -2000, 0, game::DVAR_FLAG_SAVED); + } + }; +} + +#ifdef DEBUG +REGISTER_COMPONENT(experimental::component) +#endif \ No newline at end of file diff --git a/src/client/component/fastfiles.cpp b/src/client/component/fastfiles.cpp index a85bfb8b..36e04289 100644 --- a/src/client/component/fastfiles.cpp +++ b/src/client/component/fastfiles.cpp @@ -5,6 +5,8 @@ #include "console.hpp" #include "fastfiles.hpp" #include "filesystem.hpp" +#include "imagefiles.hpp" +#include "weapon.hpp" #include "game/dvars.hpp" @@ -22,13 +24,17 @@ namespace fastfiles { utils::hook::detour db_try_load_x_file_internal_hook; utils::hook::detour db_find_xasset_header_hook; - game::dvar_t* g_dump_scripts; - std::vector fastfile_handles; + game::dvar_t* g_dump_scripts; + game::dvar_t* db_print_default_assets; + + utils::concurrency::container> fastfile_handles; + bool is_mod_pre_gfx = false; void db_try_load_x_file_internal(const char* zone_name, const int flags) { console::info("Loading fastfile %s\n", zone_name); + is_mod_pre_gfx = zone_name == "mod_pre_gfx"s; current_fastfile.access([&](std::string& fastfile) { fastfile = zone_name; @@ -68,12 +74,13 @@ namespace fastfiles dump_gsc_script(name, result); } + const std::string override_asset_name = "override/"s + name; + if (type == game::XAssetType::ASSET_TYPE_RAWFILE) { if (result.rawfile) { - const std::string override_rawfile_name = "override/"s + name; - const auto override_rawfile = db_find_xasset_header_hook.invoke(type, override_rawfile_name.data(), 0); + const auto override_rawfile = db_find_xasset_header_hook.invoke(type, override_asset_name.data(), 0); if (override_rawfile.rawfile) { result.rawfile = override_rawfile.rawfile; @@ -82,6 +89,25 @@ namespace fastfiles } } + if (type == game::XAssetType::ASSET_TYPE_STRINGTABLE) + { + if (result.stringTable) + { + const auto override_stringtable = db_find_xasset_header_hook.invoke(type, override_asset_name.data(), 0); + if (override_stringtable.stringTable) + { + result.stringTable = override_stringtable.stringTable; + console::debug("using override asset for stringtable: \"%s\"\n", name); + } + } + } + + if (db_print_default_assets->current.enabled && game::DB_IsXAssetDefault(type, name)) + { + console::warn("Waited %i msec for default asset \"%s\" of type \"%s\"\n", + diff, name, game::g_assetNames[type]); + } + if (diff > 100) { console::print( @@ -214,7 +240,10 @@ namespace fastfiles FILE_FLAG_OVERLAPPED | FILE_FLAG_NO_BUFFERING, nullptr); if (handle != INVALID_HANDLE_VALUE) { - fastfile_handles.push_back(handle); + fastfile_handles.access([&](std::vector& handles) + { + handles.push_back(handle); + }); } return handle; @@ -231,8 +260,9 @@ namespace fastfiles const auto& usermap_value = usermap.value(); const std::string usermap_file = utils::string::va("%s.ff", usermap_value.data()); const std::string usermap_load_file = utils::string::va("%s_load.ff", usermap_value.data()); + const std::string usermap_pak_file = utils::string::va("%s.pak", usermap_value.data()); - if (mapname == usermap_file || mapname == usermap_load_file) + if (mapname == usermap_file || mapname == usermap_load_file || mapname == usermap_pak_file) { const auto path = utils::string::va("usermaps\\%s\\%s", usermap_value.data(), mapname.data()); @@ -288,7 +318,7 @@ namespace fastfiles } } - if (name.ends_with(".ff")) + if (name.ends_with(".ff") || name.ends_with(".pak")) { handle = find_fastfile(name, true); } @@ -336,11 +366,17 @@ namespace fastfiles void load_pre_gfx_zones(game::XZoneInfo* zoneInfo, unsigned int zoneCount, game::DBSyncMode syncMode) { + imagefiles::close_custom_handles(); + std::vector data; merge(&data, zoneInfo, zoneCount); // code_pre_gfx + weapon::clear_modifed_enums(); + try_load_zone("mod_pre_gfx", true); + try_load_zone("h1_mod_pre_gfx", true); + game::DB_LoadXAssets(data.data(), static_cast(data.size()), syncMode); } @@ -429,6 +465,642 @@ namespace fastfiles console::warn("No aipaths found for this map\n"); } } + + int format_bsp_name(char* filename, int size, const char* mapname) + { + std::string name = mapname; + auto fmt = "maps/%s.d3dbsp"; + if (name.starts_with("mp_")) + { + fmt = "maps/mp/%s.d3dbsp"; + } + + return game::Com_sprintf(filename, size, fmt, mapname); + } + + void get_bsp_filename_stub(char* filename, int size, const char* mapname) + { + auto base_mapname = mapname; + game::Com_IsAddonMap(mapname, &base_mapname); + format_bsp_name(filename, size, base_mapname); + } + + utils::hook::detour image_file_decrypt_value_hook; + bool image_file_decrypt_value_stub(char* value, int size, char* buffer) + { + auto is_all_zero = true; + for (auto i = 0; i < size; i++) + { + if (value[i] != 0) + { + is_all_zero = false; + } + } + + if (is_all_zero) + { + return true; + } + + return image_file_decrypt_value_hook.invoke(value, size, buffer); + } + + int com_sprintf_stub(char* dest, int size, const char* /*fmt*/, const char* mapname) + { + return format_bsp_name(dest, size, mapname); + } + + const char* get_zone_name(const unsigned int index) + { + if (game::environment::is_sp()) + { + return game::sp::g_zones[index].name; + } + else + { + return game::mp::g_zones[index].name; + } + } + + utils::hook::detour db_unload_x_zones_hook; + void db_unload_x_zones_stub(const unsigned short* unload_zones, + const unsigned int unload_count, const bool create_default) + { + for (auto i = 0u; i < unload_count; i++) + { + const auto zone_name = get_zone_name(unload_zones[i]); + if (zone_name[0] != '\0') + { + imagefiles::close_handle(zone_name); + } + } + + db_unload_x_zones_hook.invoke(unload_zones, unload_count, create_default); + } + + namespace mp + { + constexpr unsigned int get_asset_type_size(const game::XAssetType type) + { + constexpr int asset_type_sizes[] = + { + 96, 88, 128, 56, 40, 216, + 56, 680, 592, 32, 32, 32, + 32, 32, 2112, 1936, 104, + 32, 24, 152, 152, 152, 16, + 64, 640, 40, 16, 136, 24, + 296, 176, 2864, 48, 0, 24, + 200, 88, 16, 144, 3616, 56, + 64, 16, 16, 0, 0, 0, 0, 24, + 40, 24, 48, 40, 24, 16, 80, + 128, 2256, 136, 32, 72, + 24, 64, 88, 48, 32, 96, 152, + 64, 32, 32, + }; + + return asset_type_sizes[type]; + } + + constexpr unsigned int get_pool_type_size(const game::XAssetType type) + { + constexpr int asset_pool_sizes[] = + { + 128, 256, 16, 1, 128, 5000, + 5248, 4352, 17536, 256, 49152, + 12288, 12288, 72864, 512, + 2750, 23264, 16000, 256, 64, + 64, 64, 64, 8000, 1, 1, 1, 1, + 1, 2, 1, 1, 32, 0, 128, 910, + 16, 14100, 128, 200, 1, 2048, + 4, 6, 0, 0, 0, 0, 1024, 768, + 400, 128, 128, 24, 24, 24, + 32, 32, 2, 128, 64, 384, 128, + 1, 128, 64, 32, 32, 16, 32, 16 + }; + + return asset_pool_sizes[type]; + } + + template + char* reallocate_asset_pool() + { + constexpr auto element_size = get_asset_type_size(Type); + static char new_pool[element_size * Size] = {0}; + static_assert(element_size != 0); + assert(element_size == game::DB_GetXAssetTypeSize(Type)); + + std::memmove(new_pool, game::g_assetPool[Type], game::g_poolSize[Type] * element_size); + + game::g_assetPool[Type] = new_pool; + game::g_poolSize[Type] = Size; + + return new_pool; + } + + template + char* reallocate_asset_pool_multiplier() + { + constexpr auto pool_size = get_pool_type_size(Type); + return reallocate_asset_pool(); + } + +#define RVA(ptr) static_cast(reinterpret_cast(ptr) - 0_b) + + struct buffer_info + { + void* ptr; + size_t size; + }; + + std::vector string_buffers; + void memset_stub(void* place, int value, size_t size) + { + for (const auto& buffer : string_buffers) + { + std::memset(buffer.ptr, 0, buffer.size); + } + + std::memset(place, value, size); + } + + void reallocate_weapon_pool() + { + constexpr auto multiplier = 2; + constexpr auto pool_size = get_pool_type_size(game::ASSET_TYPE_WEAPON) * multiplier; + static void* weapon_complete_defs[pool_size]{}; + static void* weapon_strings[pool_size]{}; + + string_buffers.emplace_back(weapon_strings, pool_size * sizeof(void*)); + + utils::hook::set(0x1186A4_b + 4, RVA(weapon_strings)); + utils::hook::set(0x1186B5_b + 4, RVA(weapon_strings)); + utils::hook::set(0x104BD2_b + 4, RVA(weapon_strings) - 0x38F1750); + + reallocate_asset_pool(); + + utils::hook::inject(0x2E3005_b + 3, + reinterpret_cast(reinterpret_cast(weapon_complete_defs) + 8)); + + utils::hook::inject(0xED734_b + 3, weapon_complete_defs); + utils::hook::inject(0x1D59F4_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DCEDB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7BB5_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7D35_b + 3, weapon_complete_defs); + utils::hook::inject(0x2ECCD0_b + 3, weapon_complete_defs); + utils::hook::inject(0x429B84_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1DFD_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E21AB_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E8BC9_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DDBA6_b + 3, weapon_complete_defs); + utils::hook::inject(0x549FF0_b + 3, weapon_complete_defs); + utils::hook::inject(0x563D20_b + 3, weapon_complete_defs); + utils::hook::inject(0x563E04_b + 3, weapon_complete_defs); + utils::hook::inject(0x618464_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DB218_b + 3, weapon_complete_defs); + utils::hook::inject(0x41ECDC_b + 3, weapon_complete_defs); + utils::hook::inject(0x42C882_b + 3, weapon_complete_defs); + utils::hook::inject(0xEFE22_b + 3, weapon_complete_defs); + utils::hook::inject(0x1199DD_b + 3, weapon_complete_defs); + utils::hook::inject(0x11D857_b + 3, weapon_complete_defs); + utils::hook::inject(0x128E28_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DB83B_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DC5BC_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2549_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E29DF_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6337_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7963_b + 3, weapon_complete_defs); + utils::hook::inject(0x2F0BA3_b + 3, weapon_complete_defs); + utils::hook::inject(0x3044C3_b + 3, weapon_complete_defs); + utils::hook::inject(0x305118_b + 3, weapon_complete_defs); + utils::hook::inject(0x41B385_b + 3, weapon_complete_defs); + utils::hook::inject(0x42544A_b + 3, weapon_complete_defs); + utils::hook::inject(0x425EAB_b + 3, weapon_complete_defs); + utils::hook::inject(0x426971_b + 3, weapon_complete_defs); + utils::hook::inject(0x42ACA7_b + 3, weapon_complete_defs); + utils::hook::inject(0x10A173_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D922C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DD2D0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DE6C4_b + 3, weapon_complete_defs); + utils::hook::inject(0x2F041F_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E2BC_b + 3, weapon_complete_defs); + utils::hook::inject(0x41F054_b + 3, weapon_complete_defs); + utils::hook::inject(0x427487_b + 3, weapon_complete_defs); + utils::hook::inject(0x461657_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A351_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A524_b + 3, weapon_complete_defs); + utils::hook::inject(0x567328_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DE83F_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DF050_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EBFE0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DF290_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A6A0_b + 3, weapon_complete_defs); + utils::hook::inject(0xE97A0_b + 3, weapon_complete_defs); + utils::hook::inject(0xE97F0_b + 3, weapon_complete_defs); + utils::hook::inject(0xE9900_b + 3, weapon_complete_defs); + utils::hook::inject(0xE9954_b + 3, weapon_complete_defs); + utils::hook::inject(0xEDAEC_b + 3, weapon_complete_defs); + utils::hook::inject(0x1BA6FC_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E99B2_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E9AD2_b + 3, weapon_complete_defs); + utils::hook::inject(0x2818CA_b + 3, weapon_complete_defs); + utils::hook::inject(0x2845FD_b + 3, weapon_complete_defs); + utils::hook::inject(0x284C2D_b + 3, weapon_complete_defs); + utils::hook::inject(0x285694_b + 3, weapon_complete_defs); + utils::hook::inject(0x285C2C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2C606A_b + 3, weapon_complete_defs); + utils::hook::inject(0x2CD275_b + 3, weapon_complete_defs); + utils::hook::inject(0x2CD2B4_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D50A4_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D8B20_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DC824_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DCDE1_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DEA7C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1463_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E14EF_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E15EB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E17FB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E18EB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E19EB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E339E_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E360A_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E3CE0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E56C8_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E5840_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E58BB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E5FE2_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6890_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E68F0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6960_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6AB0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6CA0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7640_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E76A0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7700_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7760_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E77C0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7A80_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E8753_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EA650_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EB870_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EC488_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EFD84_b + 3, weapon_complete_defs); + utils::hook::inject(0x427D5C_b + 3, weapon_complete_defs); + utils::hook::inject(0x4288F8_b + 3, weapon_complete_defs); + utils::hook::inject(0x428C89_b + 3, weapon_complete_defs); + utils::hook::inject(0x43748E_b + 3, weapon_complete_defs); + utils::hook::inject(0x4376AE_b + 3, weapon_complete_defs); + utils::hook::inject(0x43796E_b + 3, weapon_complete_defs); + utils::hook::inject(0x54953B_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A21F_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A5F0_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A7E7_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A8D9_b + 3, weapon_complete_defs); + utils::hook::inject(0x54ADA0_b + 3, weapon_complete_defs); + utils::hook::inject(0x54BAC0_b + 3, weapon_complete_defs); + utils::hook::inject(0x472198_b + 3, weapon_complete_defs); + utils::hook::inject(0x285FF2_b + 3, weapon_complete_defs); + utils::hook::inject(0x2C3154_b + 3, weapon_complete_defs); + utils::hook::inject(0x2C3AC0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DD193_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DECC4_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DEE68_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E16EB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1ACB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1CFB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1EDB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2015_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E20AB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7530_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E8950_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EBA5C_b + 3, weapon_complete_defs); + utils::hook::inject(0x30307B_b + 3, weapon_complete_defs); + utils::hook::inject(0x30308E_b + 3, weapon_complete_defs); + utils::hook::inject(0x30917E_b + 3, weapon_complete_defs); + utils::hook::inject(0x41AE27_b + 3, weapon_complete_defs); + utils::hook::inject(0x549354_b + 3, weapon_complete_defs); + utils::hook::inject(0x54A867_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D4FDB_b + 3, weapon_complete_defs); + utils::hook::inject(0xEB3BA_b + 3, weapon_complete_defs); + utils::hook::inject(0xFDC77_b + 3, weapon_complete_defs); + utils::hook::inject(0x1072EB_b + 3, weapon_complete_defs); + utils::hook::inject(0x11C14E_b + 3, weapon_complete_defs); + utils::hook::inject(0x1270D5_b + 3, weapon_complete_defs); + utils::hook::inject(0x12868F_b + 3, weapon_complete_defs); + utils::hook::inject(0x128848_b + 3, weapon_complete_defs); + utils::hook::inject(0x2C4160_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7B12_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EA7C6_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EAE75_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EB077_b + 3, weapon_complete_defs); + utils::hook::inject(0x41BB9D_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E64B_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E868_b + 3, weapon_complete_defs); + utils::hook::inject(0x41EBCB_b + 3, weapon_complete_defs); + utils::hook::inject(0x426172_b + 3, weapon_complete_defs); + utils::hook::inject(0x4262A0_b + 3, weapon_complete_defs); + utils::hook::inject(0x439669_b + 3, weapon_complete_defs); + utils::hook::inject(0x45E912_b + 3, weapon_complete_defs); + utils::hook::inject(0x46284E_b + 3, weapon_complete_defs); + utils::hook::inject(0x46D658_b + 3, weapon_complete_defs); + utils::hook::inject(0x46DF93_b + 3, weapon_complete_defs); + utils::hook::inject(0xD597B_b + 3, weapon_complete_defs); + utils::hook::inject(0xF3375_b + 3, weapon_complete_defs); + utils::hook::inject(0x121F3A_b + 3, weapon_complete_defs); + utils::hook::inject(0x1BA9C8_b + 3, weapon_complete_defs); + utils::hook::inject(0x1D3F28_b + 3, weapon_complete_defs); + utils::hook::inject(0x2C4220_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D70DB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DB108_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2FC1_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EB8D9_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EBB85_b + 3, weapon_complete_defs); + utils::hook::inject(0x40C304_b + 3, weapon_complete_defs); + utils::hook::inject(0x42A795_b + 3, weapon_complete_defs); + utils::hook::inject(0x472530_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E9939_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E7F79_b + 3, weapon_complete_defs); + utils::hook::inject(0x117129_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DE589_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E4D2E_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E4E73_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E578E_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6686_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6DCD_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E05A_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E1D3_b + 3, weapon_complete_defs); + utils::hook::inject(0x428ECE_b + 3, weapon_complete_defs); + utils::hook::inject(0x54BB26_b + 3, weapon_complete_defs); + utils::hook::inject(0x6183A6_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D6FF6_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EBCED_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E041A_b + 3, weapon_complete_defs); + utils::hook::inject(0xF38FE_b + 3, weapon_complete_defs); + utils::hook::inject(0xF7880_b + 3, weapon_complete_defs); + utils::hook::inject(0x102153_b + 3, weapon_complete_defs); + utils::hook::inject(0x1021FB_b + 3, weapon_complete_defs); + utils::hook::inject(0x10415B_b + 3, weapon_complete_defs); + utils::hook::inject(0x1168F5_b + 3, weapon_complete_defs); + utils::hook::inject(0x126C09_b + 3, weapon_complete_defs); + utils::hook::inject(0x180552_b + 3, weapon_complete_defs); + utils::hook::inject(0x1CCFD0_b + 3, weapon_complete_defs); + utils::hook::inject(0x1D929F_b + 3, weapon_complete_defs); + utils::hook::inject(0x1D9575_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E8E0E_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E98CC_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D4EF3_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D638B_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DC023_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E31D7_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E3EC8_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E9364_b + 3, weapon_complete_defs); + utils::hook::inject(0x2FCEDF_b + 3, weapon_complete_defs); + utils::hook::inject(0x324376_b + 3, weapon_complete_defs); + utils::hook::inject(0x440497_b + 3, weapon_complete_defs); + utils::hook::inject(0x460237_b + 3, weapon_complete_defs); + utils::hook::inject(0x46025A_b + 3, weapon_complete_defs); + utils::hook::inject(0x461200_b + 3, weapon_complete_defs); + utils::hook::inject(0x46EBE4_b + 3, weapon_complete_defs); + utils::hook::inject(0x46EE70_b + 3, weapon_complete_defs); + utils::hook::inject(0x46F4E1_b + 3, weapon_complete_defs); + utils::hook::inject(0x46FD48_b + 3, weapon_complete_defs); + utils::hook::inject(0x5F2479_b + 3, weapon_complete_defs); + utils::hook::inject(0x6641D_b + 3, weapon_complete_defs); + utils::hook::inject(0x1074F2_b + 3, weapon_complete_defs); + utils::hook::inject(0x1C7B8D_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DE40E_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DE8B5_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DF63C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DF7DF_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E0CD9_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2ADD_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2B7C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E2DB1_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E4C10_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E9F50_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EA0B0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EA535_b + 3, weapon_complete_defs); + utils::hook::inject(0x2ED0A5_b + 3, weapon_complete_defs); + utils::hook::inject(0x305581_b + 3, weapon_complete_defs); + utils::hook::inject(0x3236BD_b + 3, weapon_complete_defs); + utils::hook::inject(0x3F2CDA_b + 3, weapon_complete_defs); + utils::hook::inject(0x407BE5_b + 3, weapon_complete_defs); + utils::hook::inject(0x4155E0_b + 3, weapon_complete_defs); + utils::hook::inject(0x41C61D_b + 3, weapon_complete_defs); + utils::hook::inject(0x41D96E_b + 3, weapon_complete_defs); + utils::hook::inject(0x41DA71_b + 3, weapon_complete_defs); + utils::hook::inject(0x42A442_b + 3, weapon_complete_defs); + utils::hook::inject(0x42F56F_b + 3, weapon_complete_defs); + utils::hook::inject(0x5482BB_b + 3, weapon_complete_defs); + utils::hook::inject(0x54AA1D_b + 3, weapon_complete_defs); + utils::hook::inject(0x1075A9_b + 3, weapon_complete_defs); + utils::hook::inject(0xC5394_b + 3, weapon_complete_defs); + utils::hook::inject(0xEEC4D_b + 3, weapon_complete_defs); + utils::hook::inject(0x11C2E2_b + 3, weapon_complete_defs); + utils::hook::inject(0x11E6F2_b + 3, weapon_complete_defs); + utils::hook::inject(0x1994C0_b + 3, weapon_complete_defs); + utils::hook::inject(0x1E9DE0_b + 3, weapon_complete_defs); + utils::hook::inject(0x285E44_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E1BE1_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E3F34_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E474C_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E6B2E_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E9530_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E95B0_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EC08D_b + 3, weapon_complete_defs); + utils::hook::inject(0x2ECC4D_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EFD59_b + 3, weapon_complete_defs); + utils::hook::inject(0x2FCEFE_b + 3, weapon_complete_defs); + utils::hook::inject(0x2FCF15_b + 3, weapon_complete_defs); + utils::hook::inject(0x328444_b + 3, weapon_complete_defs); + utils::hook::inject(0x40F47E_b + 3, weapon_complete_defs); + utils::hook::inject(0x4169AB_b + 3, weapon_complete_defs); + utils::hook::inject(0x41C481_b + 3, weapon_complete_defs); + utils::hook::inject(0x41E742_b + 3, weapon_complete_defs); + utils::hook::inject(0x41EE62_b + 3, weapon_complete_defs); + utils::hook::inject(0x41EF3E_b + 3, weapon_complete_defs); + utils::hook::inject(0x45FD83_b + 3, weapon_complete_defs); + utils::hook::inject(0x46181E_b + 3, weapon_complete_defs); + utils::hook::inject(0x46EDD8_b + 3, weapon_complete_defs); + utils::hook::inject(0x54924D_b + 3, weapon_complete_defs); + utils::hook::inject(0x54AFA5_b + 3, weapon_complete_defs); + utils::hook::inject(0x617F77_b + 3, weapon_complete_defs); + utils::hook::inject(0xF30DF_b + 3, weapon_complete_defs); + utils::hook::inject(0x2BC095_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D605B_b + 3, weapon_complete_defs); + utils::hook::inject(0x2D8CD1_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DA7DB_b + 3, weapon_complete_defs); + utils::hook::inject(0x2DC379_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E3121_b + 3, weapon_complete_defs); + utils::hook::inject(0x2E5953_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EA8C2_b + 3, weapon_complete_defs); + utils::hook::inject(0x2EE9F7_b + 3, weapon_complete_defs); + utils::hook::inject(0x303318_b + 3, weapon_complete_defs); + utils::hook::inject(0x424D33_b + 3, weapon_complete_defs); + utils::hook::inject(0x429149_b + 3, weapon_complete_defs); + utils::hook::inject(0x4299F4_b + 3, weapon_complete_defs); + utils::hook::inject(0x4417C9_b + 3, weapon_complete_defs); + utils::hook::inject(0x471083_b + 3, weapon_complete_defs); + + utils::hook::inject(0x2E0440_b + 3, weapon_complete_defs); + + utils::hook::set(0x427EB1_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x4240A8_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x54B8B1_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2D274B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x311DA8_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x311EB8_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x323BD1_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2E0864_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2F170C_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42CB3B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x136327_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x4671FF_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x46722F_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x46754C_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x565FA2_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x56600E_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2EA352_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2EA369_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x56483F_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x2EA337_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x402261_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x4022A9_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x41CED5_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42B540_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42B560_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x5660CB_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42B523_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x117C82_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x411438_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x12AB34_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x129F9B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x12AC16_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x12AC9D_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x424087_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x54B897_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x129F5C_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42406A_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x54B87B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x565D1B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x123FF5_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42CB1B_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x42CAFE_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x136310_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x46752E_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x1362F3_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0xF454D_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x41CC61_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x41D7FB_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x427E8F_b + 4, RVA(&weapon_complete_defs)); + utils::hook::set(0x427E72_b + 4, RVA(&weapon_complete_defs)); + + utils::hook::set(0x11865F_b + 4, RVA(&weapon_complete_defs)); + } + + void reallocate_attachment_pool() + { + constexpr auto multiplier = 2; + constexpr auto pool_size = get_pool_type_size(game::ASSET_TYPE_ATTACHMENT) * multiplier; + reallocate_asset_pool(); + + static void* attachment_array[pool_size]{}; + static void* attachment_strings[pool_size]{}; + + string_buffers.emplace_back(attachment_strings, pool_size * sizeof(void*)); + + utils::hook::inject(0x118599_b + 3, attachment_strings); + utils::hook::inject(0x441187_b + 3, attachment_strings); + utils::hook::set(0x104C8A_b + 4, RVA(attachment_strings) - 0x38F1750); + + const auto sub_118540_stub = [](utils::hook::assembler& a) + { + a.mov(rax, reinterpret_cast(&attachment_array[0])); + a.mov(r8d, pool_size); + a.mov(rdx, rax); + a.mov(ecx, game::ASSET_TYPE_ATTACHMENT); + a.call(0x59D460_b); + a.jmp(0x118573_b); + }; + + const auto loc_1185A0_stub = [](utils::hook::assembler& a) + { + a.mov(rax, reinterpret_cast(&attachment_array[0])); + a.push(rbx); + a.imul(rbx, 8); + a.mov(rcx, qword_ptr(rax, rbx)); + a.pop(rbx); + a.cmp(qword_ptr(rcx, 8), 0); + a.lea(rsi, qword_ptr(rcx, 8)); + a.jmp(0x1185AE_b); + }; + + utils::hook::jump(0x11855F_b, utils::hook::assemble(sub_118540_stub), true); + utils::hook::jump(0x1185A0_b, utils::hook::assemble(loc_1185A0_stub), true); + } + + void reallocate_attachment_and_weapon() + { + // weapon & attachment strings are reset here (we need to also reset the reallocated ones) + utils::hook::call(0x13ABBA_b, memset_stub); + utils::hook::call(0x13AC5C_b, memset_stub); + utils::hook::call(0x13ACF0_b, memset_stub); + utils::hook::call(0x17D1C5_b, memset_stub); + + reallocate_weapon_pool(); + reallocate_attachment_pool(); + } + + void reallocate_sound_pool() + { + constexpr auto original_pool_size = get_pool_type_size(game::ASSET_TYPE_SOUND); + constexpr auto multiplier = 2; + constexpr auto pool_size = original_pool_size * multiplier; + + const auto pool = reallocate_asset_pool(); + utils::hook::inject(0x39621D_b + 3, reinterpret_cast(reinterpret_cast(pool) + 8)); + + static unsigned short net_const_string_sound_map[pool_size]{}; + utils::hook::inject(0x2B0CEA_b + 3, net_const_string_sound_map); + utils::hook::inject(0x2B0F52_b + 3, net_const_string_sound_map); + utils::hook::inject(0x2B1866_b + 3, net_const_string_sound_map); + utils::hook::inject(0x2B1CC7_b + 3, net_const_string_sound_map); + } + + void reallocate_asset_pools() + { + reallocate_attachment_and_weapon(); + reallocate_sound_pool(); + reallocate_asset_pool_multiplier(); + reallocate_asset_pool_multiplier(); + reallocate_asset_pool_multiplier(); + } + } + + void reallocate_asset_pools() + { + if (!game::environment::is_sp()) + { + mp::reallocate_asset_pools(); + } + } + + utils::hook::detour db_link_x_asset_entry_hook; + game::XAssetEntry* db_link_x_asset_entry_stub(game::XAssetType type, game::XAssetHeader* header) + { + if (!is_mod_pre_gfx) + { + return db_link_x_asset_entry_hook.invoke(type, header); + } + + static game::XAssetEntry entry{}; + + if (type != game::ASSET_TYPE_STRINGTABLE) + { + return &entry; + } + + return db_link_x_asset_entry_hook.invoke(type, header); + } } bool exists(const std::string& zone, bool ignore_usermap) @@ -466,10 +1138,14 @@ namespace fastfiles void close_fastfile_handles() { - for (const auto& handle : fastfile_handles) + fastfile_handles.access([&](std::vector& handles) { - CloseHandle(handle); - } + for (const auto& handle : handles) + { + CloseHandle(handle); + } + }); + } void set_usermap(const std::string& usermap) @@ -521,12 +1197,28 @@ namespace fastfiles SELECT_VALUE(0x1F5700_b, 0x39A620_b), &db_try_load_x_file_internal); db_find_xasset_header_hook.create(game::DB_FindXAssetHeader, db_find_xasset_header_stub); + db_unload_x_zones_hook.create(SELECT_VALUE(0x1F6040_b, + 0x39B3C0_b), db_unload_x_zones_stub); + + db_print_default_assets = dvars::register_bool("db_printDefaultAssets", + false, game::DVAR_FLAG_SAVED, "Print default asset usage"); + + if (!game::environment::is_sp()) + { + db_link_x_asset_entry_hook.create(0x396E80_b, db_link_x_asset_entry_stub); + } + g_dump_scripts = dvars::register_bool("g_dumpScripts", false, game::DVAR_FLAG_NONE, "Dump GSC scripts"); - // Allow loading of unsigned fastfiles + reallocate_asset_pools(); + + // Allow loading of unsigned fastfiles & imagefiles if (!game::environment::is_sp()) { utils::hook::nop(0x368153_b, 2); // DB_InflateInit + + image_file_decrypt_value_hook.create(0x367520_b, image_file_decrypt_value_stub); + utils::hook::set(0x366F00_b, 0xC301B0); } if (game::environment::is_sp()) @@ -536,6 +1228,12 @@ namespace fastfiles // Don't sys_error if aipaths are missing utils::hook::call(0x2F8EE9_b, db_find_aipaths_stub); } + else + { + // Allow loading sp maps on mp + utils::hook::jump(0x15AFC0_b, get_bsp_filename_stub); + utils::hook::call(0x112ED8_b, com_sprintf_stub); + } // Allow loading of mixed compressor types utils::hook::nop(SELECT_VALUE(0x1C4BE7_b, 0x3687A7_b), 2); @@ -595,6 +1293,53 @@ namespace fastfiles console::warn("loadzone: zone \"%s\" could not be found!\n", name); } }); + + command::add("poolUsages", []() + { + for (auto i = 0; i < game::ASSET_TYPE_COUNT; i++) + { + auto count = 0; + enum_assets(static_cast(i), [&](game::XAssetHeader header) + { + count++; + }, true); + + console::info("%i %s: %i / %i\n", i, game::g_assetNames[i], count, game::g_poolSize[i]); + } + }); + + command::add("poolUsage", [](const command::params& params) + { + if (params.size() < 2) + { + console::info("Usage: poolUsage \n"); + return; + } + + const auto type = static_cast(std::atoi(params.get(1))); + + auto count = 0; + enum_assets(type, [&](game::XAssetHeader header) + { + count++; + }, true); + + console::info("%i %s: %i / %i\n", type, game::g_assetNames[type], count, game::g_poolSize[type]); + }); + + command::add("assetCount", [](const command::params& params) + { + auto count = 0; + for (auto i = 0; i < game::ASSET_TYPE_COUNT; i++) + { + enum_assets(static_cast(i), [&](game::XAssetHeader header) + { + count++; + }, true); + } + + console::info("assets: %i / %i\n", count, 155000); + }); } }; } diff --git a/src/client/component/filesystem.cpp b/src/client/component/filesystem.cpp index e007b036..8f231216 100644 --- a/src/client/component/filesystem.cpp +++ b/src/client/component/filesystem.cpp @@ -5,6 +5,7 @@ #include "console.hpp" #include "filesystem.hpp" #include "localized_strings.hpp" +#include "mods.hpp" #include "updater.hpp" #include "game/game.hpp" @@ -39,11 +40,10 @@ namespace filesystem void fs_startup_stub(const char* name) { - console::debug("[FS] Startup\n"); + console::info("[FS] Startup\n"); initialized = true; - // hardcoded paths filesystem::register_path(utils::properties::get_appdata_path() / CLIENT_DATA_FOLDER); filesystem::register_path(L"."); filesystem::register_path(L"h1-mod"); @@ -53,9 +53,13 @@ namespace filesystem filesystem::register_path(L"raw"); filesystem::register_path(L"main"); - fs_startup_hook.invoke(name); + const auto mod_path = utils::flags::get_flag("mod"); + if (mod_path.has_value()) + { + mods::set_mod(mod_path.value()); + } - command::register_fs_game_path(); + fs_startup_hook.invoke(name); } std::vector get_paths(const std::filesystem::path& path) @@ -175,7 +179,7 @@ namespace filesystem { if (can_insert_path(path_)) { - console::debug("[FS] Registering path '%s'\n", path_.generic_string().data()); + console::info("[FS] Registering path '%s'\n", path_.generic_string().data()); get_search_paths_internal().push_front(path_); } } diff --git a/src/client/component/game_console.cpp b/src/client/component/game_console.cpp index 91684ee2..49649f60 100644 --- a/src/client/component/game_console.cpp +++ b/src/client/component/game_console.cpp @@ -181,12 +181,12 @@ namespace game_console { input = utils::string::to_lower(input); - for (const auto& dvar : dvars::dvar_list) + for (const auto& [hash, dvar] : dvars::dvar_map) { auto name = utils::string::to_lower(dvar.name); if (game::Dvar_FindVar(name.data()) && utils::string::match_compare(input, name, exact)) { - suggestions.push_back(dvar); + suggestions.emplace_back(dvar); } if (exact && suggestions.size() > 1) @@ -197,7 +197,7 @@ namespace game_console if (suggestions.size() == 0 && game::Dvar_FindVar(input.data())) { - suggestions.push_back({input, ""}); + suggestions.emplace_back(input, ""); } game::cmd_function_s* cmd = (*game::cmd_functions); @@ -209,7 +209,7 @@ namespace game_console if (utils::string::match_compare(input, name, exact)) { - suggestions.push_back({cmd->name, ""}); + suggestions.emplace_back(cmd->name, ""); } if (exact && suggestions.size() > 1) @@ -269,17 +269,13 @@ namespace game_console if (matches.size() > 24) { draw_hint_box(1, dvars::con_inputHintBoxColor->current.vector); - draw_hint_text(0, utils::string::va("%i matches (too many to show here). Press SHIFT + TAB to show more", matches.size()), - dvars::con_inputDvarMatchColor->current.vector); - - if (game::playerKeys[0].keys[game::keyNum_t::K_SHIFT].down && game::playerKeys[0].keys[game::keyNum_t::K_TAB].down) - { - console::info("]%s\n", con.buffer); - for (size_t i = 0; i < matches.size(); i++) - { - console::info("\t%s\n", matches[i].name.data()); - } - } + draw_hint_text(0, + utils::string::va( + "%i matches (too many to show here, " + "press shift+tilde to open full console, " + "press tab to print them all)", + matches.size() + ), dvars::con_inputDvarMatchColor->current.vector); } else if (matches.size() == 1) { @@ -717,6 +713,15 @@ namespace game_console clear(); } + + if (key == game::keyNum_t::K_TAB && std::strlen(con.buffer) >= 2 && matches.size() > 24) + { + console::info("]%s\n", con.buffer); + for (const auto& match : matches) + { + console::info("\t%s\n", match.name.data()); + } + } } } diff --git a/src/client/component/gsc/script_error.cpp b/src/client/component/gsc/script_error.cpp index 20af68f2..6a1a90d6 100644 --- a/src/client/component/gsc/script_error.cpp +++ b/src/client/component/gsc/script_error.cpp @@ -2,11 +2,15 @@ #include "loader/component_loader.hpp" #include "game/game.hpp" +#include "script_extension.hpp" #include "script_error.hpp" #include "component/scripting.hpp" #include +#include + +using namespace utils::string; namespace gsc { @@ -18,6 +22,37 @@ namespace gsc std::string unknown_function_error; + std::array var_typename = + { + "undefined", + "object", + "string", + "localized string", + "vector", + "float", + "int", + "codepos", + "precodepos", + "function", + "builtin function", + "builtin method", + "stack", + "animation", + "pre animation", + "thread", + "thread", + "thread", + "thread", + "struct", + "removed entity", + "entity", + "array", + "removed thread", + "", + "thread list", + "endon list", + }; + void scr_emit_function_stub(std::uint32_t filename, std::uint32_t thread_name, char* code_pos) { current_filename = filename; @@ -43,13 +78,14 @@ namespace gsc { const auto& pos = function.value(); unknown_function_error = std::format( - "while processing function '{}' in script '{}':\nunknown script '{}'", - pos.first, pos.second, scripting::current_file + "while processing function '{}' in script '{}':\nunknown script '{}' ({})", + pos.first, pos.second, scripting::current_file, scripting::current_file_id ); } else { - unknown_function_error = std::format("unknown script '{}'", scripting::current_file); + unknown_function_error = std::format("unknown script '{}' ({})", + scripting::current_file, scripting::current_file_id); } } @@ -59,29 +95,212 @@ namespace gsc const auto name = scripting::get_token(thread_name); unknown_function_error = std::format( - "while processing script '{}':\nunknown function '{}::{}'", + "while processing script '{}':\nunknown function '{}::{}'", scripting::current_file, filename, name ); } - void unknown_function_stub(const char* code_pos) + void compile_error_stub(const char* code_pos, [[maybe_unused]] const char* msg) { get_unknown_function_error(code_pos); - game::Com_Error(game::ERR_DROP, "script link error\n%s", - unknown_function_error.data()); + game::Com_Error(game::ERR_SCRIPT_DROP, "script link error\n%s", unknown_function_error.data()); } - + std::uint32_t find_variable_stub(std::uint32_t parent_id, std::uint32_t thread_name) { const auto res = game::FindVariable(parent_id, thread_name); if (!res) { get_unknown_function_error(thread_name); - game::Com_Error(game::ERR_DROP, "script link error\n%s", - unknown_function_error.data()); + game::Com_Error(game::ERR_SCRIPT_DROP, "script link error\n%s", unknown_function_error.data()); } return res; } + + unsigned int scr_get_object(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (value->type == game::VAR_POINTER) + { + return value->u.pointerValue; + } + + scr_error(va("Type %s is not an object", var_typename[value->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + unsigned int scr_get_const_string(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (game::Scr_CastString(value)) + { + assert(value->type == game::VAR_STRING); + return value->u.stringValue; + } + + game::Scr_ErrorInternal(); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + unsigned int scr_get_const_istring(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (value->type == game::VAR_ISTRING) + { + return value->u.stringValue; + } + + scr_error(va("Type %s is not a localized string", var_typename[value->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + void scr_validate_localized_string_ref(int parm_index, const char* token, int token_len) + { + assert(token); + assert(token_len >= 0); + + if (token_len < 2) + { + return; + } + + for (auto char_iter = 0; char_iter < token_len; ++char_iter) + { + if (!std::isalnum(static_cast(token[char_iter])) && token[char_iter] != '_') + { + scr_error(va("Illegal localized string reference: %s must contain only alpha-numeric characters and underscores", token)); + } + } + } + + void scr_get_vector(unsigned int index, float* vector_value) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (value->type == game::VAR_VECTOR) + { + std::memcpy(vector_value, value->u.vectorValue, sizeof(std::float_t[3])); + return; + } + + scr_error(va("Type %s is not a vector", var_typename[value->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + } + + int scr_get_int(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (value->type == game::VAR_INTEGER) + { + return value->u.intValue; + } + + scr_error(va("Type %s is not an int", var_typename[value->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + float scr_get_float(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + auto* value = game::scr_VmPub->top - index; + if (value->type == game::VAR_FLOAT) + { + return value->u.floatValue; + } + + if (value->type == game::VAR_INTEGER) + { + return static_cast(value->u.intValue); + } + + scr_error(va("Type %s is not a float", var_typename[value->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0.0f; + } + + int scr_get_pointer_type(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + if ((game::scr_VmPub->top - index)->type == game::VAR_POINTER) + { + return static_cast(game::GetObjectType((game::scr_VmPub->top - index)->u.uintValue)); + } + + scr_error(va("Type %s is not an object", var_typename[(game::scr_VmPub->top - index)->type])); + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + int scr_get_type(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + return (game::scr_VmPub->top - index)->type; + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return 0; + } + + const char* scr_get_type_name(unsigned int index) + { + if (index < game::scr_VmPub->outparamcount) + { + return var_typename[(game::scr_VmPub->top - index)->type]; + } + + scr_error(va("Parameter %u does not exist", index + 1)); + return nullptr; + } + + template + void safe_func() + { + static utils::hook::detour hook; + static const auto stub = []() + { + __try + { + hook.invoke(); + } + __except (EXCEPTION_EXECUTE_HANDLER) + { + game::Scr_ErrorInternal(); + } + }; + + const auto ptr = rva + 0_b; + hook.create(reinterpret_cast(ptr), stub); + } } std::optional> find_function(const char* pos) @@ -108,9 +327,27 @@ namespace gsc { scr_emit_function_hook.create(SELECT_VALUE(0x3BD680_b, 0x504660_b), &scr_emit_function_stub); - utils::hook::call(SELECT_VALUE(0x3BD626_b, 0x504606_b), unknown_function_stub); // CompileError (LinkFile) - utils::hook::call(SELECT_VALUE(0x3BD672_b, 0x504652_b), unknown_function_stub); // ^ - utils::hook::call(SELECT_VALUE(0x3BD75A_b, 0x50473A_b), find_variable_stub); // Scr_EmitFunction + utils::hook::call(SELECT_VALUE(0x3BD626_b, 0x504606_b), compile_error_stub); // CompileError (LinkFile) + utils::hook::call(SELECT_VALUE(0x3BD672_b, 0x504652_b), compile_error_stub); // ^ + utils::hook::call(SELECT_VALUE(0x3BD75A_b, 0x50473A_b), find_variable_stub); // Scr_EmitFunction + + // Restore basic error messages for commonly used scr functions + utils::hook::jump(SELECT_VALUE(0x3C89F0_b, 0x50F9E0_b), scr_get_object); + utils::hook::jump(SELECT_VALUE(0x3C84C0_b, 0x50F560_b), scr_get_const_string); + utils::hook::jump(SELECT_VALUE(0x3C8280_b, 0x50F320_b), scr_get_const_istring); + utils::hook::jump(SELECT_VALUE(0x2D6950_b, 0x452EF0_b), scr_validate_localized_string_ref); + utils::hook::jump(SELECT_VALUE(0x3C8F30_b, 0x50FF20_b), scr_get_vector); + utils::hook::jump(SELECT_VALUE(0x3C8930_b, 0x50F920_b), scr_get_int); + utils::hook::jump(SELECT_VALUE(0x3C87D0_b, 0x50F870_b), scr_get_float); + + utils::hook::jump(SELECT_VALUE(0x3C8C10_b, 0x50FC00_b), scr_get_pointer_type); + utils::hook::jump(SELECT_VALUE(0x3C8DE0_b, 0x50FDD0_b), scr_get_type); + utils::hook::jump(SELECT_VALUE(0x3C8E50_b, 0x50FE40_b), scr_get_type_name); + + if (!game::environment::is_sp()) + { + safe_func<0xBA7A0>(); // fix vlobby cac crash + } } void pre_destroy() override diff --git a/src/client/component/gsc/script_error.hpp b/src/client/component/gsc/script_error.hpp index e8742026..1cee00d2 100644 --- a/src/client/component/gsc/script_error.hpp +++ b/src/client/component/gsc/script_error.hpp @@ -1,4 +1,3 @@ - #pragma once namespace gsc diff --git a/src/client/component/gsc/script_extension.cpp b/src/client/component/gsc/script_extension.cpp index 9afcfd0b..5e174beb 100644 --- a/src/client/component/gsc/script_extension.cpp +++ b/src/client/component/gsc/script_extension.cpp @@ -3,23 +3,17 @@ #include "game/dvars.hpp" #include "game/game.hpp" -#include "game/scripting/execution.hpp" -#include "game/scripting/function.hpp" -#include "game/scripting/functions.hpp" -#include "game/scripting/lua/error.hpp" - -#include +#include "component/logfile.hpp" #include "component/command.hpp" #include "component/console.hpp" #include "component/scripting.hpp" -#include "component/logfile.hpp" -#include -#include - -#include "script_extension.hpp" #include "script_error.hpp" +#include "script_extension.hpp" +#include "script_loading.hpp" + +#include namespace gsc { @@ -33,8 +27,8 @@ namespace gsc namespace { - std::unordered_map functions; - std::unordered_map methods; + std::unordered_map functions; + std::unordered_map methods; bool force_error_print = false; std::optional gsc_error_msg; @@ -70,10 +64,15 @@ namespace gsc reinterpret_cast(pos - 2)); } + game::scr_entref_t get_entity_id_stub(std::uint32_t ent_id) + { + const auto ref = game::Scr_GetEntityIdRef(ent_id); + saved_ent_ref = ref; + return ref; + } + void execute_custom_function(const std::uint16_t id) { - auto error = false; - try { const auto& function = functions[id]; @@ -85,23 +84,33 @@ namespace gsc return_value(result); } } - catch (const std::exception& e) + catch (const std::exception& ex) { - error = true; - force_error_print = true; - gsc_error_msg = e.what(); + scr_error(ex.what()); + } + } + + void vm_call_builtin_function_stub(builtin_function func) + { + const auto function_id = get_function_id(); + const auto custom = functions.contains(static_cast(function_id)); + if (custom) + { + execute_custom_function(function_id); + return; } - if (error) + if (func == nullptr) { - game::Scr_ErrorInternal(); + scr_error(utils::string::va("builtin function \"%s\" doesn't exist", gsc_ctx->func_name(function_id).data()), true); + return; } + + func(); } void execute_custom_method(const std::uint16_t id) { - auto error = false; - try { const auto& method = methods[id]; @@ -113,52 +122,29 @@ namespace gsc return_value(result); } } - catch (const std::exception& e) + catch (const std::exception& ex) { - error = true; - force_error_print = true; - gsc_error_msg = e.what(); - } - - if (error) - { - game::Scr_ErrorInternal(); + scr_error(ex.what()); } } - void vm_call_builtin_function_stub(builtin_function function) + void vm_call_builtin_method_stub(builtin_method meth) { - const auto function_id = get_function_id(); - - if (!functions.contains(function_id)) + const auto method_id = get_function_id(); + const auto custom = methods.contains(static_cast(method_id)); + if (custom) { - function(); + execute_custom_method(method_id); + return; } - else - { - execute_custom_function(function_id); - } - } - game::scr_entref_t get_entity_id_stub(std::uint32_t ent_id) - { - const auto ref = game::Scr_GetEntityIdRef(ent_id); - saved_ent_ref = ref; - return ref; - } - - void vm_call_builtin_method_stub(builtin_method method) - { - const auto function_id = get_function_id(); - - if (!methods.contains(function_id)) + if (meth == nullptr) { - method(saved_ent_ref); - } - else - { - execute_custom_method(function_id); + scr_error(utils::string::va("builtin method \"%s\" doesn't exist", gsc_ctx->meth_name(method_id).data()), true); + return; } + + meth(saved_ent_ref); } void builtin_call_error(const std::string& error) @@ -167,13 +153,11 @@ namespace gsc if (function_id > 0x1000) { - console::warn("in call to builtin method \"%s\"%s", - xsk::gsc::h1::resolver::method_name(function_id).data(), error.data()); + console::warn("in call to builtin method \"%s\"%s", gsc_ctx->meth_name(function_id).data(), error.data()); } else { - console::warn("in call to builtin function \"%s\"%s", - xsk::gsc::h1::resolver::function_name(function_id).data(), error.data()); + console::warn("in call to builtin function \"%s\"%s", gsc_ctx->func_name(function_id).data(), error.data()); } } @@ -181,7 +165,8 @@ namespace gsc { try { - return {xsk::gsc::h1::resolver::opcode_name(opcode)}; + const auto index = gsc_ctx->opcode_enum(opcode); + return { gsc_ctx->opcode_name(index) }; } catch (...) { @@ -209,7 +194,8 @@ namespace gsc void vm_error_stub(int mark_pos) { - if (!developer_script->current.enabled && !force_error_print) + const bool dev_script = developer_script ? developer_script->current.enabled : false; + if (!dev_script && !force_error_print) { utils::hook::invoke(SELECT_VALUE(0x415C90_b, 0x59DDA0_b), mark_pos); return; @@ -266,19 +252,27 @@ namespace gsc } } + void scr_error(const char* error, const bool force_print) + { + force_error_print = force_print; + gsc_error_msg = error; + + game::Scr_ErrorInternal(); + } + namespace function { void add(const std::string& name, script_function function) { - if (xsk::gsc::h1::resolver::find_function(name)) + if (gsc_ctx->func_exists(name)) { - const auto id = xsk::gsc::h1::resolver::function_id(name); + const auto id = gsc_ctx->func_id(name); functions[id] = function; } else { const auto id = ++function_id_start; - xsk::gsc::h1::resolver::add_function(name, static_cast(id)); + gsc_ctx->func_add(name, static_cast(id)); functions[id] = function; } } @@ -288,15 +282,15 @@ namespace gsc { void add(const std::string& name, script_method method) { - if (xsk::gsc::h1::resolver::find_method(name)) + if (gsc_ctx->meth_exists(name)) { - const auto id = xsk::gsc::h1::resolver::method_id(name); + const auto id = gsc_ctx->meth_id(name); methods[id] = method; } else { const auto id = ++method_id_start; - xsk::gsc::h1::resolver::add_method(name, static_cast(id)); + gsc_ctx->meth_add(name, static_cast(id)); methods[id] = method; } } @@ -332,6 +326,8 @@ namespace gsc public: void post_unpack() override { + developer_script = dvars::register_bool("developer_script", false, 0, "Enable developer script comments"); + utils::hook::set(SELECT_VALUE(0x3BD86C_b, 0x50484C_b), 0x1000); // change builtin func count utils::hook::set(SELECT_VALUE(0x3BD872_b, 0x504852_b) + 4, @@ -348,8 +344,6 @@ namespace gsc utils::hook::inject(SELECT_VALUE(0x3BDC36_b, 0x504C66_b) + 3, &meth_table); utils::hook::set(SELECT_VALUE(0x3BDC3F_b, 0x504C6F_b), sizeof(meth_table)); - developer_script = dvars::register_bool("developer_script", false, 0, "Enable developer script comments"); - utils::hook::nop(SELECT_VALUE(0x3CB723_b, 0x512783_b), 8); utils::hook::call(SELECT_VALUE(0x3CB723_b, 0x512783_b), vm_call_builtin_function_stub); @@ -358,7 +352,7 @@ namespace gsc utils::hook::nop(SELECT_VALUE(0x3CBA4E_b, 0x512AAE_b), 2); utils::hook::call(SELECT_VALUE(0x3CBA46_b, 0x512AA6_b), vm_call_builtin_method_stub); - utils::hook::call(SELECT_VALUE(0x3CC9F3_b, 0x513A53_b), vm_error_stub); + utils::hook::call(SELECT_VALUE(0x3CC9F3_b, 0x513A53_b), vm_error_stub); // LargeLocalResetToMark if (game::environment::is_dedi()) { @@ -424,7 +418,7 @@ namespace gsc if (what.type != game::VAR_FUNCTION || with.type != game::VAR_FUNCTION) { - throw std::runtime_error("replaceFunc: parameter 1 must be a function"); + throw std::runtime_error("replacefunc: parameter 1 must be a function"); } logfile::set_gsc_hook(what.u.codePosValue, with.u.codePosValue); @@ -455,8 +449,7 @@ namespace gsc function::add("executecommand", [](const function_args& args) { - const auto cmd = args[0].as(); - command::execute(cmd, true); + command::execute(args[0].as(), false); return scripting::script_value{}; }); diff --git a/src/client/component/gsc/script_extension.hpp b/src/client/component/gsc/script_extension.hpp index 2aae4a2e..ad57e2b5 100644 --- a/src/client/component/gsc/script_extension.hpp +++ b/src/client/component/gsc/script_extension.hpp @@ -34,6 +34,8 @@ namespace gsc extern const game::dvar_t* developer_script; + void scr_error(const char* error, const bool force_print = false); + namespace function { void add(const std::string& name, script_function function); diff --git a/src/client/component/gsc/script_loading.cpp b/src/client/component/gsc/script_loading.cpp index 5d9c2cf5..ab24a55b 100644 --- a/src/client/component/gsc/script_loading.cpp +++ b/src/client/component/gsc/script_loading.cpp @@ -6,7 +6,7 @@ #include "component/filesystem.hpp" #include "component/logfile.hpp" #include "component/scripting.hpp" -#include "component/gsc/script_loading.hpp" +#include "component/memory.hpp" #include "game/dvars.hpp" @@ -14,39 +14,34 @@ #include "game/scripting/execution.hpp" #include "game/scripting/function.hpp" -#include -#include -#include -#include -#include -#include -#include -#include +#include "script_extension.hpp" +#include "script_loading.hpp" +#include #include #include #include namespace gsc { + std::unique_ptr gsc_ctx = std::make_unique();; + namespace { - auto compiler = ::gsc::compiler(); - auto decompiler = ::gsc::decompiler(); - auto assembler = ::gsc::assembler(); - auto disassembler = ::gsc::disassembler(); + utils::hook::detour scr_begin_load_scripts_hook; + utils::hook::detour scr_end_load_scripts_hook; std::unordered_map main_handles; std::unordered_map init_handles; utils::memory::allocator scriptfile_allocator; - std::unordered_map loaded_scripts; + std::unordered_map loaded_scripts; struct { char* buf = nullptr; char* pos = nullptr; - unsigned int size = 0x1000000; + const unsigned int size = memory::custom_script_mem_size; } script_memory; char* allocate_buffer(size_t size) @@ -83,15 +78,14 @@ namespace gsc free_script_memory(); } - bool read_script_file(const std::string& name, std::string* data) + bool read_raw_script_file(const std::string& name, std::string* data) { if (filesystem::read_file(name, data)) { return true; } - const auto name_str = name.data(); - + const auto* name_str = name.data(); if (game::DB_XAssetExists(game::ASSET_TYPE_RAWFILE, name_str) && !game::DB_IsXAssetDefault(game::ASSET_TYPE_RAWFILE, name_str)) { @@ -110,41 +104,75 @@ namespace gsc return false; } + bool force_load = false; + game::ScriptFile* load_custom_script(const char* file_name, const std::string& real_name) { - if (game::VirtualLobby_Loaded()) - { - return nullptr; - } - - if (const auto itr = loaded_scripts.find(real_name); itr != loaded_scripts.end()) + if (const auto itr = loaded_scripts.find(file_name); itr != loaded_scripts.end()) { return itr->second; } - std::string source_buffer; - if (!read_script_file(real_name + ".gsc", &source_buffer) || source_buffer.empty()) + if (game::VirtualLobby_Loaded() && !force_load) { return nullptr; } - if (game::DB_XAssetExists(game::ASSET_TYPE_SCRIPTFILE, file_name) && + std::string source_buffer{}; + if (!read_raw_script_file(real_name + ".gsc", &source_buffer) || source_buffer.empty()) + { + return nullptr; + } + + // filter out "GSC rawfiles" that were used for development usage and are not meant for us. + // each "GSC rawfile" has a ScriptFile counterpart to be used instead + if (game::DB_XAssetExists(game::ASSET_TYPE_SCRIPTFILE, file_name) && !game::DB_IsXAssetDefault(game::ASSET_TYPE_SCRIPTFILE, file_name)) { - // filter out gsc rawfiles that contain developer code (has ScriptFile counterparts for ship, won't compile either) - if ((real_name.starts_with("maps/createfx") || real_name.starts_with("maps/createart") || real_name.starts_with("maps/mp")) + if ((real_name.starts_with("maps/createfx") || real_name.starts_with("maps/createart") || real_name.starts_with("maps/mp")) && (real_name.ends_with("_fx") || real_name.ends_with("_fog") || real_name.ends_with("_hdr"))) { + console::debug("Refusing to compile rawfile '%s'\n", real_name.data()); return game::DB_FindXAssetHeader(game::ASSET_TYPE_SCRIPTFILE, file_name, false).scriptfile; } } - std::vector data; - data.assign(source_buffer.begin(), source_buffer.end()); + console::debug("Loading custom gsc '%s.gsc'", real_name.data()); try { - compiler->compile(real_name, data); + auto& compiler = gsc_ctx->compiler(); + auto& assembler = gsc_ctx->assembler(); + + std::vector data; + data.assign(source_buffer.begin(), source_buffer.end()); + + const auto assembly_ptr = compiler.compile(real_name, data); + const auto output_script = assembler.assemble(*assembly_ptr); + + const auto bytecode = output_script.first; // formerly named "script" + const auto stack = output_script.second; + + const auto script_file_ptr = static_cast(scriptfile_allocator.allocate(sizeof(game::ScriptFile))); + script_file_ptr->name = file_name; + + script_file_ptr->len = static_cast(stack.size); + script_file_ptr->bytecodeLen = static_cast(bytecode.size); + + const auto stack_size = static_cast(stack.size + 1); + const auto byte_code_size = static_cast(bytecode.size + 1); + + script_file_ptr->buffer = static_cast(scriptfile_allocator.allocate(stack_size)); + std::memcpy(const_cast(script_file_ptr->buffer), stack.data, stack.size); + + script_file_ptr->bytecode = allocate_buffer(byte_code_size); + std::memcpy(script_file_ptr->bytecode, bytecode.data, bytecode.size); + + script_file_ptr->compressedLen = 0; + + loaded_scripts[file_name] = script_file_ptr; + + return script_file_ptr; } catch (const std::exception& e) { @@ -153,46 +181,21 @@ namespace gsc console::error("**********************************************\n"); return nullptr; } + } - auto assembly = compiler->output(); - - try + std::string get_raw_script_file_name(const std::string& name) + { + if (name.ends_with(".gsh")) { - assembler->assemble(real_name, assembly); - } - catch (const std::exception& e) - { - console::error("*********** script compile error *************\n"); - console::error("failed to assemble '%s':\n%s", real_name.data(), e.what()); - console::error("**********************************************\n"); - return nullptr; + return name; } - const auto script_file_ptr = scriptfile_allocator.allocate(); - script_file_ptr->name = file_name; - - const auto stack = assembler->output_stack(); - script_file_ptr->len = static_cast(stack.size()); - - const auto script = assembler->output_script(); - script_file_ptr->bytecodeLen = static_cast(script.size()); - - script_file_ptr->buffer = game::Hunk_AllocateTempMemoryHigh(stack.size() + 1); - std::memcpy(script_file_ptr->buffer, stack.data(), stack.size()); - - script_file_ptr->bytecode = allocate_buffer(script.size() + 1); - std::memcpy(script_file_ptr->bytecode, script.data(), script.size()); - - script_file_ptr->compressedLen = 0; - - loaded_scripts[real_name] = script_file_ptr; - - return script_file_ptr; + return name + ".gsc"; } std::string get_script_file_name(const std::string& name) { - const auto id = xsk::gsc::h1::resolver::token_id(name); + const auto id = gsc_ctx->token_id(name); if (!id) { return name; @@ -201,27 +204,25 @@ namespace gsc return std::to_string(id); } - std::vector decompile_script_file(const std::string& name, const std::string& real_name) + std::pair> read_compiled_script_file(const std::string& name, const std::string& real_name) { const auto* script_file = game::DB_FindXAssetHeader(game::ASSET_TYPE_SCRIPTFILE, name.data(), false).scriptfile; - if (!script_file) + if (script_file == nullptr) { throw std::runtime_error(std::format("Could not load scriptfile '{}'", real_name)); } - console::info("Decompiling scriptfile '%s'\n", real_name.data()); + console::debug("Decompiling scriptfile '%s'\n", real_name.data()); - std::vector stack{script_file->buffer, script_file->buffer + script_file->len}; - std::vector bytecode{script_file->bytecode, script_file->bytecode + script_file->bytecodeLen}; + const auto len = script_file->compressedLen; + const std::string stack{script_file->buffer, static_cast(len)}; - auto decompressed_stack = xsk::utils::zlib::decompress(stack, static_cast(stack.size())); + const auto decompressed_stack = utils::compression::zlib::decompress(stack); - disassembler->disassemble(name, bytecode, decompressed_stack); - auto output = disassembler->output(); + std::vector stack_data; + stack_data.assign(decompressed_stack.begin(), decompressed_stack.end()); - decompiler->decompile(name, output); - - return decompiler->output(); + return {{reinterpret_cast(script_file->bytecode), static_cast(script_file->bytecodeLen)}, stack_data}; } void load_script(const std::string& name) @@ -231,31 +232,31 @@ namespace gsc return; } - const auto main_handle = game::Scr_GetFunctionHandle(name.data(), xsk::gsc::h1::resolver::token_id("main")); - const auto init_handle = game::Scr_GetFunctionHandle(name.data(), xsk::gsc::h1::resolver::token_id("init")); + const auto main_handle = game::Scr_GetFunctionHandle(name.data(), gsc_ctx->token_id("main")); + const auto init_handle = game::Scr_GetFunctionHandle(name.data(), gsc_ctx->token_id("init")); if (main_handle) { - console::info("Loaded '%s::main'\n", name.data()); + console::debug("Loaded '%s::main'\n", name.data()); main_handles[name] = main_handle; } if (init_handle) { - console::info("Loaded '%s::init'\n", name.data()); + console::debug("Loaded '%s::init'\n", name.data()); init_handles[name] = init_handle; } } - void load_scripts(const std::filesystem::path& root_dir, const std::filesystem::path& script_dir) + void load_scripts(const std::filesystem::path& root_dir, const std::filesystem::path& subfolder) { - std::filesystem::path script_dir_path = root_dir / script_dir; - if (!utils::io::directory_exists(script_dir_path.generic_string())) + std::filesystem::path script_dir = root_dir / subfolder; + if (!utils::io::directory_exists(script_dir.generic_string())) { return; } - const auto scripts = utils::io::list_files(script_dir_path.generic_string()); + const auto scripts = utils::io::list_files(script_dir.generic_string()); for (const auto& script : scripts) { if (!script.ends_with(".gsc")) @@ -281,25 +282,31 @@ namespace gsc return game::DB_IsXAssetDefault(type, name); } - void gscr_load_gametype_script_stub(void* a1, void* a2) + void load_gametype_script_stub(void* a1, void* a2) { utils::hook::invoke(SELECT_VALUE(0x2B9DA0_b, 0x18BC00_b), a1, a2); - if (game::VirtualLobby_Loaded()) - { - return; - } - for (const auto& path : filesystem::get_search_paths()) { - load_scripts(path, "scripts/"); if (game::environment::is_sp()) { load_scripts(path, "scripts/sp/"); + load_scripts(path, "scripts/"); } else { - load_scripts(path, "scripts/mp/"); + if (!game::VirtualLobby_Loaded()) + { + load_scripts(path, "scripts/mp/"); + load_scripts(path, "scripts/"); + } + + force_load = true; + const auto _0 = gsl::finally([&] + { + force_load = false; + }); + load_scripts(path, "scripts/mp_patches/"); } } } @@ -313,28 +320,48 @@ namespace gsc return; } - utils::hook::invoke(SELECT_VALUE(0x1F1E00_b, 0x396080_b), rawfile, buf, size); + game::DB_GetRawBuffer(rawfile, buf, size); } - void pmem_init_stub() + void scr_begin_load_scripts_stub() { - utils::hook::invoke(SELECT_VALUE(0x420260_b, 0x5A5590_b)); + const bool dev_script = developer_script ? developer_script->current.enabled : false; + const auto comp_mode = dev_script ? + xsk::gsc::build::dev: + xsk::gsc::build::prod; - const auto type_0 = &game::g_scriptmem[0]; - const auto type_1 = &game::g_scriptmem[1]; + gsc_ctx->init(comp_mode, [](const std::string& include_name) + -> std::pair> + { + const auto real_name = get_raw_script_file_name(include_name); - const auto size_0 = 0x100000; // default size - const auto size_1 = 0x100000 + script_memory.size; + std::string file_buffer; + if (!read_raw_script_file(real_name, &file_buffer) || file_buffer.empty()) + { + const auto name = get_script_file_name(include_name); + if (game::DB_XAssetExists(game::ASSET_TYPE_SCRIPTFILE, name.data())) + { + return read_compiled_script_file(name, real_name); + } - const auto block = reinterpret_cast(VirtualAlloc(NULL, size_0 + size_1, MEM_RESERVE, PAGE_READWRITE)); + throw std::runtime_error(std::format("Could not load gsc file '{}'", real_name)); + } - type_0->buf = block; - type_0->size = size_0; + std::vector script_data; + script_data.assign(file_buffer.begin(), file_buffer.end()); - type_1->buf = block + size_0; - type_1->size = size_1; + return {{}, script_data}; + }); - utils::hook::set(SELECT_VALUE(0x420252_b, 0x5A5582_b), size_0 + size_1); + scr_begin_load_scripts_hook.invoke(); + } + + void scr_end_load_scripts_stub() + { + // cleanup the compiler + gsc_ctx->cleanup(); + + scr_end_load_scripts_hook.invoke(); } } @@ -342,7 +369,7 @@ namespace gsc { for (auto& function_handle : main_handles) { - console::info("Executing '%s::main'\n", function_handle.first.data()); + console::debug("Executing '%s::main'\n", function_handle.first.data()); game::RemoveRefToObject(game::Scr_ExecThread(function_handle.second, 0)); } } @@ -351,7 +378,7 @@ namespace gsc { for (auto& function_handle : init_handles) { - console::info("Executing '%s::init'\n", function_handle.first.data()); + console::debug("Executing '%s::init'\n", function_handle.first.data()); game::RemoveRefToObject(game::Scr_ExecThread(function_handle.second, 0)); } } @@ -362,7 +389,7 @@ namespace gsc const auto id = static_cast(std::atoi(name)); if (id) { - real_name = xsk::gsc::h1::resolver::token_name(id); + real_name = gsc_ctx->token_name(id); } auto* script = load_custom_script(name, real_name); @@ -379,53 +406,36 @@ namespace gsc public: void post_unpack() override { - // allow custom scripts to include other custom scripts - xsk::gsc::h1::resolver::init([](const auto& include_name) - { - const auto real_name = include_name + ".gsc"; + // Load our scripts with an uncompressed stack + utils::hook::call(SELECT_VALUE(0x3C7280_b, 0x50E3C0_b), db_get_raw_buffer_stub); - std::string file_buffer; - if (!read_script_file(real_name, &file_buffer) || file_buffer.empty()) - { - const auto name = get_script_file_name(include_name); - if (game::DB_XAssetExists(game::ASSET_TYPE_SCRIPTFILE, name.data())) - { - return decompile_script_file(name, real_name); - } - else - { - throw std::runtime_error(std::format("Could not load gsc file '{}'", real_name)); - } - } + scr_begin_load_scripts_hook.create(SELECT_VALUE(0x3BDB90_b, 0x504BC0_b), scr_begin_load_scripts_stub); + scr_end_load_scripts_hook.create(SELECT_VALUE(0x3BDCC0_b, 0x504CF0_b), scr_end_load_scripts_stub); - std::vector result; - result.assign(file_buffer.begin(), file_buffer.end()); - - return result; - }); - - // hook xasset functions to return our own custom scripts + // ProcessScript: hook xasset functions to return our own custom scripts utils::hook::call(SELECT_VALUE(0x3C7217_b, 0x50E357_b), find_script); utils::hook::call(SELECT_VALUE(0x3C7227_b, 0x50E367_b), db_is_x_asset_default); - // GScr_LoadScripts - utils::hook::call(SELECT_VALUE(0x2BA152_b, 0x18C325_b), gscr_load_gametype_script_stub); + // GScr_LoadScripts: initial loading of scripts + utils::hook::call(SELECT_VALUE(0x2BA152_b, 0x18C325_b), load_gametype_script_stub); - // loads scripts with an uncompressed stack - utils::hook::call(SELECT_VALUE(0x3C7280_b, 0x50E3C0_b), db_get_raw_buffer_stub); - - // Increase script memory - utils::hook::call(SELECT_VALUE(0x38639C_b, 0x15C4D6_b), pmem_init_stub); + // main is called from scripting.cpp + // init is called from scripting.cpp scripting::on_shutdown([](bool free_scripts, bool post_shutdown) { if (free_scripts && post_shutdown) { - xsk::gsc::h1::resolver::cleanup(); clear(); } }); } + + void pre_destroy() override + { + scr_begin_load_scripts_hook.clear(); + scr_end_load_scripts_hook.clear(); + } }; } diff --git a/src/client/component/gsc/script_loading.hpp b/src/client/component/gsc/script_loading.hpp index e42a7e45..8fcdff2c 100644 --- a/src/client/component/gsc/script_loading.hpp +++ b/src/client/component/gsc/script_loading.hpp @@ -1,7 +1,10 @@ #pragma once +#include namespace gsc { + extern std::unique_ptr gsc_ctx; + void load_main_handles(); void load_init_handles(); game::ScriptFile* find_script(game::XAssetType type, const char* name, int allow_create_default); diff --git a/src/client/component/imagefiles.cpp b/src/client/component/imagefiles.cpp new file mode 100644 index 00000000..bd1cdf72 --- /dev/null +++ b/src/client/component/imagefiles.cpp @@ -0,0 +1,230 @@ +#include +#include "loader/component_loader.hpp" + +#include "images.hpp" +#include "console.hpp" +#include "filesystem.hpp" +#include "fastfiles.hpp" +#include "scheduler.hpp" +#include "imagefiles.hpp" + +#include "game/game.hpp" + +#include +#include +#include +#include + +#define CUSTOM_IMAGE_FILE_INDEX 96 + +namespace imagefiles +{ + namespace + { + utils::memory::allocator image_file_allocator; + std::unordered_map image_file_handles; + + std::string get_image_file_name() + { + return fastfiles::get_current_fastfile(); + } + + namespace mp + { + struct image_file_unk + { + char __pad0[120]; + }; + + std::unordered_map image_file_unk_map; + + void* get_image_file_unk_mp(unsigned int index) + { + if (index != CUSTOM_IMAGE_FILE_INDEX) + { + return &reinterpret_cast( + SELECT_VALUE(0x4802090_b, 0x6306770_b))[index]; + } + + const auto name = get_image_file_name(); + if (image_file_unk_map.find(name) == image_file_unk_map.end()) + { + const auto unk = image_file_allocator.allocate(); + image_file_unk_map[name] = unk; + return unk; + } + + return image_file_unk_map[name]; + } + } + + namespace sp + { + struct image_file_unk + { + char __pad0[96]; + }; + + std::unordered_map image_file_unk_map; + + void* get_image_file_unk_mp(unsigned int index) + { + if (index != CUSTOM_IMAGE_FILE_INDEX) + { + return &reinterpret_cast( + SELECT_VALUE(0x4802090_b, 0x6306770_b))[index]; + } + + const auto name = get_image_file_name(); + if (image_file_unk_map.find(name) == image_file_unk_map.end()) + { + const auto unk = image_file_allocator.allocate(); + image_file_unk_map[name] = unk; + return unk; + } + + return image_file_unk_map[name]; + } + } + + game::DB_IFileSysFile* get_image_file_handle(unsigned int index) + { + if (index != CUSTOM_IMAGE_FILE_INDEX) + { + return reinterpret_cast( + SELECT_VALUE(0x4801D80_b, 0x6306180_b))[index]; + } + + const auto name = get_image_file_name(); + return image_file_handles[name]; + } + + void db_create_gfx_image_stream_stub(utils::hook::assembler& a) + { + const auto check_image_file_handle = a.newLabel(); + const auto handle_is_open = a.newLabel(); + + a.movzx(eax, cx); + a.push(rax); + a.push(rax); + a.pushad64(); + a.mov(rcx, rax); + a.call_aligned(SELECT_VALUE(sp::get_image_file_unk_mp, mp::get_image_file_unk_mp)); + a.mov(qword_ptr(rsp, 0x80), rax); + a.popad64(); + a.pop(rax); + a.mov(rsi, rax); + a.pop(rax); + + a.push(rax); + a.push(rax); + a.pushad64(); + a.mov(rcx, rax); + a.call_aligned(get_image_file_handle); + a.mov(qword_ptr(rsp, 0x80), rax); + a.popad64(); + a.pop(rax); + a.mov(r12, rax); + a.pop(rax); + + a.cmp(r12, r13); + a.jnz(handle_is_open); + a.jmp(SELECT_VALUE(0x1FAD49_b, 0x3A0CA5_b)); + + a.bind(handle_is_open); + a.jmp(SELECT_VALUE(0x1FAD99_b, 0x3A0CF5_b)); + } + + void* pakfile_open_stub(void* /*handles*/, unsigned int count, int is_imagefile, + unsigned int index, int is_localized) + { + if (index != CUSTOM_IMAGE_FILE_INDEX) + { + return utils::hook::invoke( + SELECT_VALUE(0x42BC00_b, 0x5B2030_b), + SELECT_VALUE(0x4801D80_b, 0x6306180_b), + count, is_imagefile, index, is_localized + ); + } + + const auto name = get_image_file_name(); + const auto db_fs = *game::db_fs; + const auto handle = db_fs->vftbl->OpenFile(db_fs, + game::SF_PAKFILE, utils::string::va("%s.pak", name.data())); + if (handle != nullptr) + { + image_file_handles[name] = handle; + } + return handle; + } + + int com_sprintf_stub(char* buffer, const int len, const char* fmt, unsigned int index) + { + if (index != CUSTOM_IMAGE_FILE_INDEX) + { + return game::Com_sprintf(buffer, len, fmt, index); + } + + const auto name = get_image_file_name(); + return game::Com_sprintf(buffer, len, "%s.pak", name.data()); + } + } + + void close_custom_handles() + { + const auto db_fs = *game::db_fs; + for (const auto& handle : image_file_handles) + { + if (handle.second != nullptr) + { + db_fs->vftbl->Close(db_fs, handle.second); + } + } + + image_file_handles.clear(); + sp::image_file_unk_map.clear(); + mp::image_file_unk_map.clear(); + image_file_allocator.clear(); + } + + void close_handle(const std::string& fastfile) + { + if (!image_file_handles.contains(fastfile)) + { + return; + } + + const auto db_fs = *game::db_fs; + const auto handle = image_file_handles[fastfile]; + if (handle != nullptr) + { + db_fs->vftbl->Close(db_fs, handle); + } + + image_file_handles.erase(fastfile); + if (game::environment::is_sp()) + { + image_file_allocator.free(sp::image_file_unk_map[fastfile]); + sp::image_file_unk_map.erase(fastfile); + } + else + { + image_file_allocator.free(mp::image_file_unk_map[fastfile]); + mp::image_file_unk_map.erase(fastfile); + } + } + + class component final : public component_interface + { + public: + void post_unpack() override + { + utils::hook::jump(SELECT_VALUE(0x1FAD35_b, 0x3A0C95_b), + utils::hook::assemble(db_create_gfx_image_stream_stub), true); + utils::hook::call(SELECT_VALUE(0x1FAD7B_b, 0x3A0CD7_b), pakfile_open_stub); + utils::hook::call(SELECT_VALUE(0x1FAD5D_b, 0x3A0CB9_b), com_sprintf_stub); + } + }; +} + +REGISTER_COMPONENT(imagefiles::component) diff --git a/src/client/component/imagefiles.hpp b/src/client/component/imagefiles.hpp new file mode 100644 index 00000000..ff309625 --- /dev/null +++ b/src/client/component/imagefiles.hpp @@ -0,0 +1,7 @@ +#pragma once + +namespace imagefiles +{ + void close_custom_handles(); + void close_handle(const std::string& fastfile); +} diff --git a/src/client/component/input.cpp b/src/client/component/input.cpp index ad14ae43..36fbb49e 100644 --- a/src/client/component/input.cpp +++ b/src/client/component/input.cpp @@ -17,15 +17,6 @@ namespace input void cl_char_event_stub(const int local_client_num, const int key) { - if (game::environment::is_sp() && ui_scripting::lui_running()) - { - ui_scripting::notify("keypress", - { - {"keynum", key}, - {"key", game::Key_KeynumToString(key, 0, 1)}, - }); - } - if (!game_console::console_char_event(local_client_num, key)) { return; @@ -36,15 +27,6 @@ namespace input void cl_key_event_stub(const int local_client_num, const int key, const int down) { - if (game::environment::is_sp() && ui_scripting::lui_running()) - { - ui_scripting::notify(down ? "keydown" : "keyup", - { - {"keynum", key}, - {"key", game::Key_KeynumToString(key, 0, 1)}, - }); - } - if (!game_console::console_key_event(local_client_num, key, down)) { return; diff --git a/src/client/component/logger.cpp b/src/client/component/logger.cpp index 80c59025..4db069f5 100644 --- a/src/client/component/logger.cpp +++ b/src/client/component/logger.cpp @@ -96,6 +96,12 @@ namespace logger console::info("%s", buffer); } + + void r_warn_once_per_frame_vsnprintf_stub(char* buffer, size_t buffer_length, char* msg, va_list va) + { + vsnprintf(buffer, buffer_length, msg, va); + console::warn(buffer); + } } class component final : public component_interface @@ -108,7 +114,15 @@ namespace logger // lua stuff utils::hook::jump(SELECT_VALUE(0x106010_b, 0x27CBB0_b), print_dev); // debug utils::hook::jump(SELECT_VALUE(0x107680_b, 0x27E210_b), print_error); // error - utils::hook::jump(SELECT_VALUE(0x0E6E30_b, 0x1F6140_b), printf); // print + utils::hook::jump(SELECT_VALUE(0x0E6E30_b, 0x1F6140_b), print); // print + + if (game::environment::is_mp()) + { + utils::hook::call(0x6BBB81_b, r_warn_once_per_frame_vsnprintf_stub); + + utils::hook::jump(0x498BD0_b, print_warning); // dmWarn + utils::hook::jump(0x498AD0_b, print); // dmLog + } } com_error_hook.create(game::Com_Error, com_error_stub); diff --git a/src/client/component/map_patches.cpp b/src/client/component/map_patches.cpp index 7c59b80c..cac9de29 100644 --- a/src/client/component/map_patches.cpp +++ b/src/client/component/map_patches.cpp @@ -7,6 +7,49 @@ namespace map_patches { + struct GfxLightGridTree + { + unsigned char index; + unsigned char maxDepth; + char unused[2]; + int nodeCount; + int leafCount; + int coordMinGridSpace[3]; + int coordMaxGridSpace[3]; + int coordHalfSizeGridSpace[3]; + int defaultColorIndexBitCount; + int defaultLightIndexBitCount; + unsigned int* p_nodeTable; + int leafTableSize; + unsigned char* p_leafTable; + }; + + enum leaf_table_version : std::int8_t + { + h2 = 0i8, + h1 = 0i8, + s1 = 0i8, + iw6 = 1i8, + }; + + utils::hook::detour r_decode_light_grid_block_hook; + void r_decode_light_grid_block_stub(GfxLightGridTree* p_tree, int child_mask, + char child_index, int encoded_node_address, char* p_node_raw, char* p_leaf_raw) + { + static const auto p_address = 0x6A032E_b + 1; + if (p_tree->unused[0] == leaf_table_version::iw6) + { + utils::hook::set(p_address, 6); // iw6 + } + else + { + utils::hook::set(p_address, 7); // s1,h1,h2 + } + + r_decode_light_grid_block_hook.invoke(p_tree, child_mask, + child_index, encoded_node_address, p_node_raw, p_leaf_raw); + } + class component final : public component_interface { public: @@ -20,6 +63,9 @@ namespace map_patches // skip fx name prefix checks utils::hook::set(0x2F377D_b, 0xEB); // createfx parse utils::hook::set(0x4444E0_b, 0xEB); // scr_loadfx + + // patch iw6 leafTable decoding + r_decode_light_grid_block_hook.create(0x69E7D0_b, r_decode_light_grid_block_stub); } }; } diff --git a/src/client/component/map_rotation.cpp b/src/client/component/map_rotation.cpp index 7b80bebf..1a39c0cb 100644 --- a/src/client/component/map_rotation.cpp +++ b/src/client/component/map_rotation.cpp @@ -3,6 +3,7 @@ #include "command.hpp" #include "console.hpp" +#include "map_rotation.hpp" #include "scheduler.hpp" #include "game/game.hpp" @@ -15,20 +16,23 @@ namespace map_rotation { namespace { - DWORD previous_priority{}; + rotation_data dedicated_rotation; - void set_dvar(const std::string& dvar, const std::string& value) - { - command::execute(utils::string::va("%s \"%s\"", dvar.data(), value.data()), true); - } + const game::dvar_t* sv_map_rotation; + const game::dvar_t* sv_map_rotation_current; + const game::dvar_t* sv_random_map_rotation; void set_gametype(const std::string& gametype) { - set_dvar("g_gametype", gametype); + assert(!gametype.empty()); + + game::Dvar_SetFromStringByNameFromSource("g_gametype", gametype.data(), game::DVAR_SOURCE_INTERNAL); } void launch_map(const std::string& mapname) { + assert(!mapname.empty()); + command::execute(utils::string::va("map %s", mapname.data()), false); } @@ -46,53 +50,103 @@ namespace map_rotation } } - std::string load_current_map_rotation() + void apply_rotation(rotation_data& rotation) { - auto* rotation = game::Dvar_FindVar("sv_mapRotationCurrent"); - if (!strlen(rotation->current.string)) + assert(!rotation.empty()); + + std::size_t i = 0; + while (i < rotation.get_entries_size()) { - rotation = game::Dvar_FindVar("sv_mapRotation"); - set_dvar("sv_mapRotationCurrent", rotation->current.string); - } - - return rotation->current.string; - } - - std::vector parse_current_map_rotation() - { - const auto rotation = load_current_map_rotation(); - return utils::string::split(rotation, ' '); - } - - void store_new_rotation(const std::vector& elements, const size_t index) - { - std::string value{}; - - for (auto i = index; i < elements.size(); ++i) - { - if (i != index) + const auto& entry = rotation.get_next_entry(); + if (entry.first == "map"s) { - value.push_back(' '); + console::info("Loading new map: '%s'\n", entry.second.data()); + if (!game::SV_MapExists(entry.second.data())) + { + console::info("map_rotation: '%s' map doesn't exist!\n", entry.second.data()); + launch_default_map(); + return; + } + + launch_map(entry.second); + + break; } - value.append(elements[i]); - } + if (entry.first == "gametype"s) + { + console::info("Applying new gametype: '%s'\n", entry.second.data()); + set_gametype(entry.second); + } - set_dvar("sv_mapRotationCurrent", value); + ++i; + } } - void change_process_priority() + void load_rotation(const std::string& data) { - auto* const dvar = game::Dvar_FindVar("sv_autoPriority"); - if (dvar && dvar->current.enabled) + static auto loaded = false; + if (loaded) { - scheduler::on_game_initialized([] - { - SetPriorityClass(GetCurrentProcess(), previous_priority); - }, scheduler::pipeline::main, 1s); + return; + } - previous_priority = GetPriorityClass(GetCurrentProcess()); - SetPriorityClass(GetCurrentProcess(), NORMAL_PRIORITY_CLASS); + loaded = true; + try + { + dedicated_rotation.parse(data); + } + catch (const std::exception& ex) + { + console::error("%s: sv_map_rotation contains invalid data!\n", ex.what()); + } + + console::debug("dedicated_rotation size after parsing is '%llu'", dedicated_rotation.get_entries_size()); + } + + void load_map_rotation() + { + const std::string map_rotation = sv_map_rotation->current.string; + if (!map_rotation.empty()) + { + console::debug("sv_map_rotation is not empty. Parsing...\n"); + load_rotation(map_rotation); + } + } + + void apply_map_rotation_current(const std::string& data) + { + assert(!data.empty()); + + rotation_data rotation_current; + + try + { + rotation_current.parse(data); + } + catch (const std::exception& ex) + { + console::error("%s: sv_map_rotation_current contains invalid data!\n", ex.what()); + } + + game::Dvar_SetFromStringByNameFromSource("sv_map_rotation_current", "", game::DVAR_SOURCE_INTERNAL); + + if (rotation_current.empty()) + { + console::warn("sv_map_rotation_current is empty or contains invalid data\n"); + launch_default_map(); + return; + } + + apply_rotation(rotation_current); + } + + void randomize_map_rotation() + { + if (sv_random_map_rotation->current.enabled) + { + console::info("Randomizing map rotation\n"); + dedicated_rotation.randomize(); } } @@ -104,37 +158,28 @@ namespace map_rotation return; } - const auto rotation = parse_current_map_rotation(); + console::info("Rotating map...\n"); - for (size_t i = 0; !rotation.empty() && i < (rotation.size() - 1); i += 2) + // This takes priority because of backwards compatibility + const std::string map_rotation_current = sv_map_rotation_current->current.string; + if (!map_rotation_current.empty()) { - const auto& key = rotation[i]; - const auto& value = rotation[i + 1]; - - if (key == "gametype") - { - set_gametype(value); - } - else if (key == "map") - { - store_new_rotation(rotation, i + 2); - change_process_priority(); - if (!game::SV_MapExists(value.data())) - { - console::info("map_rotation: '%s' map doesn't exist!\n", value.data()); - launch_default_map(); - return; - } - launch_map(value); - return; - } - else - { - console::info("Invalid map rotation key: %s\n", key.data()); - } + console::debug("Applying sv_map_rotation_current\n"); + apply_map_rotation_current(map_rotation_current); + return; } - launch_default_map(); + load_map_rotation(); + if (dedicated_rotation.empty()) + { + console::warn("sv_map_rotation is empty or contains invalid data. Restarting map\n"); + launch_default_map(); + return; + } + + randomize_map_rotation(); + + apply_rotation(dedicated_rotation); } void trigger_map_rotation() @@ -152,6 +197,68 @@ namespace map_rotation } } + rotation_data::rotation_data() + : index_(0) + { + } + + void rotation_data::randomize() + { + std::random_device rd; + std::mt19937 gen(rd()); + + std::ranges::shuffle(this->rotation_entries_, gen); + } + + void rotation_data::add_entry(const std::string& key, const std::string& value) + { + this->rotation_entries_.emplace_back(std::make_pair(key, value)); + } + + bool rotation_data::contains(const std::string& key, const std::string& value) const + { + return std::ranges::any_of(this->rotation_entries_, [&](const auto& entry) + { + return entry.first == key && entry.second == value; + }); + } + + bool rotation_data::empty() const noexcept + { + return this->rotation_entries_.empty(); + } + + std::size_t rotation_data::get_entries_size() const noexcept + { + return this->rotation_entries_.size(); + } + + rotation_data::rotation_entry& rotation_data::get_next_entry() + { + const auto index = this->index_; + ++this->index_ %= this->rotation_entries_.size(); + return this->rotation_entries_.at(index); + } + + void rotation_data::parse(const std::string& data) + { + const auto tokens = utils::string::split(data, ' '); + for (std::size_t i = 0; !tokens.empty() && i < (tokens.size() - 1); i += 2) + { + const auto& key = tokens[i]; + const auto& value = tokens[i + 1]; + + if (key == "map"s || key == "gametype"s) + { + this->add_entry(key, value); + } + else + { + throw parse_rotation_error(); + } + } + } + class component final : public component_interface { public: @@ -164,17 +271,16 @@ namespace map_rotation scheduler::once([] { - dvars::register_string("sv_mapRotation", "", game::DVAR_FLAG_NONE, ""); - dvars::register_string("sv_mapRotationCurrent", "", game::DVAR_FLAG_NONE, ""); - dvars::register_string("sv_autoPriority", "", game::DVAR_FLAG_NONE, "Lowers the process priority during map changes to not cause lags on other servers."); + sv_map_rotation = dvars::register_string("sv_mapRotation", "", game::DVAR_FLAG_NONE, ""); + sv_map_rotation_current = dvars::register_string("sv_mapRotationCurrent", "", game::DVAR_FLAG_NONE, ""); }, scheduler::pipeline::main); + sv_random_map_rotation = dvars::register_bool("sv_randomMapRotation", false, game::DVAR_FLAG_NONE, "Randomize map rotation"); + command::add("map_rotate", &perform_map_rotation); // Hook GScr_ExitLevel utils::hook::jump(0xE2670_b, &trigger_map_rotation, true); // not sure if working - - previous_priority = GetPriorityClass(GetCurrentProcess()); } }; } diff --git a/src/client/component/map_rotation.hpp b/src/client/component/map_rotation.hpp new file mode 100644 index 00000000..ab5d6605 --- /dev/null +++ b/src/client/component/map_rotation.hpp @@ -0,0 +1,34 @@ +#pragma once + +namespace map_rotation +{ + struct parse_rotation_error : public std::exception + { + const char* what() const noexcept override { return "Rotation parse error"; } + }; + + class rotation_data + { + public: + using rotation_entry = std::pair; + + rotation_data(); + + void randomize(); + + // In case a new way to enrich the map rotation is added (other than sv_mapRotation) + // this method should be called to add a new entry (gamemode/map & value) + void add_entry(const std::string& key, const std::string& value); + + [[nodiscard]] bool contains(const std::string& key, const std::string& value) const; + [[nodiscard]] bool empty() const noexcept; + [[nodiscard]] std::size_t get_entries_size() const noexcept; + [[nodiscard]] rotation_entry& get_next_entry(); + + void parse(const std::string& data); + + private: + std::vector rotation_entries_; + std::size_t index_; + }; +} diff --git a/src/client/component/mapents.cpp b/src/client/component/mapents.cpp index eee4297f..f38016e4 100644 --- a/src/client/component/mapents.cpp +++ b/src/client/component/mapents.cpp @@ -3,17 +3,12 @@ #include "game/game.hpp" #include "console.hpp" +#include "filesystem.hpp" #include #include -#include -#include -#include -#include -#include -#include -#include +#include "gsc/script_loading.hpp" namespace mapents { @@ -30,6 +25,7 @@ namespace mapents for (auto i = 0; i < lines.size(); i++) { + auto line_num = i+1; auto line = lines[i]; if (line.ends_with('\r')) { @@ -67,7 +63,7 @@ namespace mapents if (line[0] == '{' && in_map_ent) { - console::error("[map_ents parser] Unexpected '{' on line %i\n", i); + console::error("[map_ents parser] Unexpected '{' on line %i\n", line_num); return {}; } @@ -92,15 +88,15 @@ namespace mapents if (line[0] == '}' && !in_map_ent) { - console::error("[map_ents parser] Unexpected '}' on line %i\n", i); + console::error("[map_ents parser] Unexpected '}' on line %i\n", line_num); return {}; } std::regex expr(R"~((.+) "(.*)")~"); std::smatch match{}; - if (!std::regex_search(line, match, expr)) + if (!std::regex_search(line, match, expr) && !line.empty()) { - console::warn("[map_ents parser] Failed to parse line %i (%s)\n", i, line.data()); + console::warn("[map_ents parser] Failed to parse line %i (%s)\n", line_num, line.data()); continue; } @@ -109,7 +105,7 @@ namespace mapents if (key.size() <= 0) { - console::warn("[map_ents parser] Invalid key ('%s') on line %i (%s)\n", key.data(), i, line.data()); + console::warn("[map_ents parser] Invalid key ('%s') on line %i (%s)\n", key.data(), line_num, line.data()); continue; } @@ -128,10 +124,10 @@ namespace mapents } const auto key_ = key.substr(1, key.size() - 2); - const auto id = xsk::gsc::h1::resolver::token_id(key_); + const auto id = gsc::gsc_ctx->token_id(key_); if (id == 0) { - console::warn("[map_ents parser] Key '%s' not found, on line %i (%s)\n", key_.data(), i, line.data()); + console::warn("[map_ents parser] Key '%s' not found, on line %i (%s)\n", key_.data(), line_num, line.data()); continue; } @@ -141,16 +137,48 @@ namespace mapents return {out_buffer}; } + std::string raw_ents; + bool load_raw_mapents() + { + auto mapents_name = utils::string::va("%s.ents", **reinterpret_cast(SELECT_VALUE(0xB489D40_b, 0xA975F40_b))); + if (filesystem::exists(mapents_name)) + { + try + { + console::debug("Reading raw ents file \"%s\"\n", mapents_name); + raw_ents = filesystem::read_file(mapents_name); + if (!raw_ents.empty()) + { + return true; + } + } + catch (const std::exception& ex) + { + console::error("Failed to read raw ents file \"%s\"\n%s\n", mapents_name, ex.what()); + } + } + return false; + } + std::string entity_string; const char* cm_entity_string_stub() { - if (!entity_string.empty()) + const char* ents = nullptr; + if (load_raw_mapents()) { - return entity_string.data(); + ents = raw_ents.data(); + } + else + { + if (!entity_string.empty()) + { + return entity_string.data(); + } + + ents = utils::hook::invoke(SELECT_VALUE(0x3685C0_b, 0x4CD140_b)); } - const auto original = utils::hook::invoke(SELECT_VALUE(0x3685C0_b, 0x4CD140_b)); - const auto parsed = parse_mapents(original); + const auto parsed = parse_mapents(ents); if (parsed.has_value()) { entity_string = parsed.value(); @@ -158,13 +186,14 @@ namespace mapents } else { - return original; + return ents; } } void cm_unload_stub(void* clip_map) { entity_string.clear(); + raw_ents.clear(); utils::hook::invoke(SELECT_VALUE(0x368560_b, 0x4CD0E0_b), clip_map); } } diff --git a/src/client/component/materials.cpp b/src/client/component/materials.cpp index 371bea53..ee8cce39 100644 --- a/src/client/component/materials.cpp +++ b/src/client/component/materials.cpp @@ -4,6 +4,7 @@ #include "materials.hpp" #include "console.hpp" #include "filesystem.hpp" +#include "scheduler.hpp" #include "game/game.hpp" #include "game/dvars.hpp" @@ -20,123 +21,17 @@ namespace materials namespace { utils::hook::detour db_material_streaming_fail_hook; - utils::hook::detour material_register_handle_hook; utils::hook::detour db_get_material_index_hook; - struct material_data_t - { - std::unordered_map materials; - std::unordered_map images; - }; +#ifdef DEBUG + utils::hook::detour material_compare_hook; + utils::hook::detour set_pixel_texture_hook; + + const game::dvar_t* debug_materials = nullptr; +#endif char constant_table[0x20] = {}; - - utils::concurrency::container material_data; - - game::GfxImage* setup_image(game::GfxImage* image, const utils::image& raw_image) - { - image->imageFormat = 0x1000003; - image->resourceSize = -1; - - D3D11_SUBRESOURCE_DATA data{}; - data.SysMemPitch = raw_image.get_width() * 4; - data.SysMemSlicePitch = data.SysMemPitch * raw_image.get_height(); - data.pSysMem = raw_image.get_buffer(); - - game::Image_Setup(image, raw_image.get_width(), raw_image.get_height(), image->depth, image->numElements, - image->imageFormat, DXGI_FORMAT_R8G8B8A8_UNORM, image->name, &data); - - return image; - } - - game::Material* create_material(const std::string& name, const std::string& data) - { - const auto white = material_register_handle_hook.invoke("white"); - const auto material = utils::memory::get_allocator()->allocate(); - const auto texture_table = utils::memory::get_allocator()->allocate(); - const auto image = utils::memory::get_allocator()->allocate(); - - std::memcpy(material, white, sizeof(game::Material)); - std::memcpy(texture_table, white->textureTable, sizeof(game::MaterialTextureDef)); - std::memcpy(image, white->textureTable->u.image, sizeof(game::GfxImage)); - - material->constantTable = &constant_table; - material->name = utils::memory::get_allocator()->duplicate_string(name); - image->name = material->name; - - material->textureTable = texture_table; - material->textureTable->u.image = setup_image(image, data); - - return material; - } - - void free_material(game::Material* material) - { - material->textureTable->u.image->textures.___u0.map->Release(); - material->textureTable->u.image->textures.shaderView->Release(); - utils::memory::get_allocator()->free(material->textureTable->u.image); - utils::memory::get_allocator()->free(material->textureTable); - utils::memory::get_allocator()->free(material->name); - utils::memory::get_allocator()->free(material); - } - - game::Material* load_material(const std::string& name) - { - return material_data.access([&](material_data_t& data_) -> game::Material* - { - if (const auto i = data_.materials.find(name); i != data_.materials.end()) - { - return i->second; - } - - std::string data{}; - if (const auto i = data_.images.find(name); i != data_.images.end()) - { - data = i->second; - } - - if (data.empty() && !filesystem::read_file(utils::string::va("materials/%s.png", name.data()), &data)) - { - data_.materials[name] = nullptr; - return nullptr; - } - - const auto material = create_material(name, data); - data_.materials[name] = material; - - return material; - }); - } - - game::Material* try_load_material(const std::string& name) - { - if (name == "white") - { - return nullptr; - } - - try - { - return load_material(name); - } - catch (const std::exception& e) - { - console::error("Failed to load material %s: %s\n", name.data(), e.what()); - } - - return nullptr; - } - - game::Material* material_register_handle_stub(const char* name) - { - auto result = try_load_material(name); - if (result == nullptr) - { - result = material_register_handle_hook.invoke(name); - } - return result; - } - + int db_material_streaming_fail_stub(game::Material* material) { if (material->constantTable == &constant_table) @@ -156,40 +51,148 @@ namespace materials return db_get_material_index_hook.invoke(material); } - } - void add(const std::string& name, const std::string& data) - { - material_data.access([&](material_data_t& data_) +#ifdef DEBUG + char material_compare_stub(unsigned int index_a, unsigned int index_b) { - data_.images[name] = data; - }); - } + char result = 0; - bool exists(const std::string& name) - { - return material_data.access([&](material_data_t& data_) - { - return data_.images.find(name) != data_.images.end(); - }); - } - - void clear() - { - material_data.access([&](material_data_t& data_) - { - for (auto& material : data_.materials) + __try { - if (material.second == nullptr) - { - continue; - } - - free_material(material.second); + result = material_compare_hook.invoke(index_a, index_b); + } + __except (EXCEPTION_EXECUTE_HANDLER) + { + const auto* material_a = utils::hook::invoke(0x395FE0_b, index_a); + const auto* material_b = utils::hook::invoke(0x395FE0_b, index_b); + console::error("Material_Compare: %s - %s (%d - %d)", + material_a->name, material_b->name, material_a->info.sortKey, material_b->info.sortKey); } - data_.materials.clear(); - }); + return result; + } + + void print_material(const game::Material* material) + { + if (!debug_materials || !debug_materials->current.enabled) + { + return; + } + + console::debug("current material is \"%s\"\n", material->name); + } + + void print_current_material_stub(utils::hook::assembler& a) + { + const auto loc_6AD59B = a.newLabel(); + + a.pushad64(); + a.mov(rcx, r15); + a.call_aligned(print_material); + a.popad64(); + + a.cmp(byte_ptr(rbx), 5); + a.mov(rax, ptr(r15, 0x130)); + + a.jnz(loc_6AD59B); + a.nop(dword_ptr(rax, rax, 0x00000000)); + + a.jmp(0x6AD570_b); + + a.bind(loc_6AD59B); + a.jmp(0x6AD59B_b); + } + + void set_pixel_texture_stub(void* cmd_buf_state, unsigned int a2, const game::GfxImage* image) + { + if (!debug_materials || !debug_materials->current.enabled) + { + set_pixel_texture_hook.invoke(cmd_buf_state, a2, image); + return; + } + + if (image && image->name) + { + console::debug("set_pixel_texture_stub: \"%s\"\n", image->name); + } + else + { + console::error("set_pixel_texture_stub: texture has no name or is nullptr\n"); + } + + set_pixel_texture_hook.invoke(cmd_buf_state, a2, image); + } +#endif + } + + bool setup_material_image(game::Material* material, const std::string& data) + { + if (*game::d3d11_device == nullptr) + { + console::error("Tried to create texture while d3d11 device isn't initialized\n"); + return false; + } + + const auto image = material->textureTable->u.image; + image->imageFormat = 0x1000003; + image->resourceSize = -1; + + auto raw_image = utils::image{data}; + + D3D11_SUBRESOURCE_DATA resource_data{}; + resource_data.SysMemPitch = raw_image.get_width() * 4; + resource_data.SysMemSlicePitch = resource_data.SysMemPitch * raw_image.get_height(); + resource_data.pSysMem = raw_image.get_buffer(); + + game::Image_Setup(image, raw_image.get_width(), raw_image.get_height(), image->depth, image->numElements, + image->imageFormat, DXGI_FORMAT_R8G8B8A8_UNORM, image->name, &resource_data); + return true; + } + + game::Material* create_material(const std::string& name) + { + const auto white = game::Material_RegisterHandle("$white"); + const auto material = utils::memory::allocate(); + const auto texture_table = utils::memory::allocate(); + const auto image = utils::memory::allocate(); + + std::memcpy(material, white, sizeof(game::Material)); + std::memcpy(texture_table, white->textureTable, sizeof(game::MaterialTextureDef)); + std::memcpy(image, white->textureTable->u.image, sizeof(game::GfxImage)); + + material->constantTable = &constant_table; + material->name = utils::memory::duplicate_string(name); + image->name = material->name; + + image->textures.map = nullptr; + image->textures.shaderView = nullptr; + image->textures.shaderViewAlternate = nullptr; + texture_table->u.image = image; + + material->textureTable = texture_table; + + return material; + } + + void free_material(game::Material* material) + { + const auto try_release = [](T** resource) + { + if (*resource != nullptr) + { + (*resource)->Release(); + *resource = nullptr; + } + }; + + try_release(&material->textureTable->u.image->textures.map); + try_release(&material->textureTable->u.image->textures.shaderView); + try_release(&material->textureTable->u.image->textures.shaderViewAlternate); + + utils::memory::free(material->textureTable->u.image); + utils::memory::free(material->textureTable); + utils::memory::free(material->name); + utils::memory::free(material); } class component final : public component_interface @@ -202,9 +205,23 @@ namespace materials return; } - material_register_handle_hook.create(game::Material_RegisterHandle, material_register_handle_stub); db_material_streaming_fail_hook.create(SELECT_VALUE(0x1FB400_b, 0x3A1600_b), db_material_streaming_fail_stub); db_get_material_index_hook.create(SELECT_VALUE(0x1F1D80_b, 0x396000_b), db_get_material_index_stub); + +#ifdef DEBUG + if (!game::environment::is_sp()) + { + material_compare_hook.create(0x693B90_b, material_compare_stub); + set_pixel_texture_hook.create(0x6B33E0_b, set_pixel_texture_stub); + + utils::hook::jump(0x6AD55C_b, utils::hook::assemble(print_current_material_stub), true); + + scheduler::once([] + { + debug_materials = dvars::register_bool("debug_materials", false, game::DVAR_FLAG_NONE, "Print current material and images"); + }, scheduler::main); + } +#endif } }; } diff --git a/src/client/component/materials.hpp b/src/client/component/materials.hpp index 0c02227f..7ca0ddfa 100644 --- a/src/client/component/materials.hpp +++ b/src/client/component/materials.hpp @@ -1,8 +1,10 @@ #pragma once +#include "game/game.hpp" + namespace materials { - void add(const std::string& name, const std::string& data); - bool exists(const std::string& name); - void clear(); + bool setup_material_image(game::Material* material, const std::string& data); + game::Material* create_material(const std::string& name); + void free_material(game::Material* material); } diff --git a/src/client/component/memory.cpp b/src/client/component/memory.cpp new file mode 100644 index 00000000..e8fdaf8e --- /dev/null +++ b/src/client/component/memory.cpp @@ -0,0 +1,143 @@ +#include +#include "loader/component_loader.hpp" + +#include "memory.hpp" + +#include "game/game.hpp" + +#include +#include + +namespace memory +{ + namespace + { + constexpr auto mem_low_size = 0x80000000ui64 * 2; // default: 0x80000000 + constexpr auto mem_high_size = 0x80000000ui64 * 2; // default: 0x80000000 + + constexpr auto script_mem_low_size = 0x100000ui64; // default: 0x100000 + constexpr auto script_mem_high_size = 0x100000ui64 + custom_script_mem_size; // default: 0x100000 + + constexpr auto phys_mem_low_size = 0x700000000ui64; // default: 0x700000000 + constexpr auto phys_mem_high_size = 0x300000000i64; // default: 0x300000000 + + constexpr auto pmem_alloc_size = + mem_low_size + + mem_high_size + + script_mem_low_size + + script_mem_high_size + + phys_mem_low_size + + phys_mem_high_size; + + constexpr auto mem_low_buf = 0; + constexpr auto mem_high_buf = mem_low_buf + mem_low_size; + + constexpr auto script_mem_low_buf = mem_high_buf + mem_high_size; + constexpr auto script_mem_high_buf = script_mem_low_buf + script_mem_low_size; + + constexpr auto phys_mem_low_buf = script_mem_high_buf + script_mem_high_size; + constexpr auto phys_mem_high_buf = phys_mem_low_buf + phys_mem_low_size; + + constexpr auto stream_mem_size = 0x2000000ui64; + + void pmem_init() + { + const auto size = pmem_alloc_size; + const auto allocated_buffer = VirtualAlloc(NULL, size, MEM_RESERVE, PAGE_READWRITE); + auto buffer = reinterpret_cast(allocated_buffer); + *game::pmem_size = size; + *game::pmem_buffer = buffer; + + memset(game::g_mem, 0, sizeof(*game::g_mem)); + + game::g_mem->prim[game::PHYS_ALLOC_LOW].buf = buffer + mem_low_buf; + game::g_mem->prim[game::PHYS_ALLOC_LOW].pos = mem_low_size; + game::g_mem->prim[game::PHYS_ALLOC_HIGH].buf = buffer + mem_high_buf; + game::g_mem->prim[game::PHYS_ALLOC_HIGH].pos = mem_high_size; + + game::g_mem->prim[game::PHYS_ALLOC_LOW].unk1 = 0; + game::g_mem->prim[game::PHYS_ALLOC_HIGH].unk1 = 0; + + memset(game::g_scriptmem, 0, sizeof(*game::g_scriptmem)); + + game::g_scriptmem->prim[game::PHYS_ALLOC_LOW].buf = buffer + script_mem_low_buf; + game::g_scriptmem->prim[game::PHYS_ALLOC_LOW].pos = script_mem_low_size; + game::g_scriptmem->prim[game::PHYS_ALLOC_HIGH].buf = buffer + script_mem_high_buf; + game::g_scriptmem->prim[game::PHYS_ALLOC_HIGH].pos = script_mem_high_size; + + game::g_scriptmem->prim[game::PHYS_ALLOC_LOW].unk1 = 0; + game::g_scriptmem->prim[game::PHYS_ALLOC_HIGH].unk1 = 0; + + memset(game::g_physmem, 0, sizeof(*game::g_physmem)); + + game::g_physmem->prim[game::PHYS_ALLOC_LOW].buf = buffer + phys_mem_low_buf; + game::g_physmem->prim[game::PHYS_ALLOC_LOW].pos = phys_mem_low_size; + game::g_physmem->prim[game::PHYS_ALLOC_HIGH].buf = buffer + phys_mem_high_buf; + game::g_physmem->prim[game::PHYS_ALLOC_HIGH].pos = phys_mem_high_size; + + game::g_physmem->prim[game::PHYS_ALLOC_LOW].unk1 = 2; + game::g_physmem->prim[game::PHYS_ALLOC_HIGH].unk1 = 2; + + *game::stream_size = stream_mem_size; + *game::stream_buffer = reinterpret_cast(VirtualAlloc(NULL, *game::stream_size, MEM_COMMIT, PAGE_READWRITE)); + } + + void pmem_init_stub() + { + // call our own init + pmem_init(); + + const auto script_mem_size = script_mem_low_size + script_mem_high_size; + utils::hook::set(SELECT_VALUE(0x420252_b, 0x5A5582_b), static_cast(script_mem_size)); + } + } + + namespace + { + int out_of_memory_text_stub(char* dest, int size, const char* fmt, ...) + { + fmt = "%s (%d)\n\n" + "Disable shader caching, lower graphic settings, free up RAM, or update your GPU drivers.\n\n" + "If this still occurs, try using the '-memoryfix' parameter to generate the 'players2' folder."; + + char buffer[2048]; + + { + va_list ap; + va_start(ap, fmt); + + vsnprintf_s(buffer, sizeof(buffer), _TRUNCATE, fmt, ap); + + va_end(ap); + } + + return utils::hook::invoke(SELECT_VALUE(0x429200_b, 0x5AF0F0_b), dest, size, "%s", buffer); + } + } + + class component final : public component_interface + { + public: + void post_unpack() override + { + // patch PMem_Init, so we can use whatever memory size we want + utils::hook::call(SELECT_VALUE(0x38639C_b, 0x15C4D6_b), pmem_init_stub); + + // Com_sprintf for "Out of memory. You are probably low on disk space." + utils::hook::call(SELECT_VALUE(0x457BC9_b, 0x1D8E09_b), out_of_memory_text_stub); + + // "fix" for rare 'Out of memory error' error + // this will *at least* generate the configs for mp/sp, which is the #1 issue + if (utils::flags::has_flag("memoryfix")) + { + utils::hook::jump(SELECT_VALUE(0x5110D0_b, 0x6200C0_b), malloc); + utils::hook::jump(SELECT_VALUE(0x510FF0_b, 0x61FFE0_b), _aligned_malloc); + utils::hook::jump(SELECT_VALUE(0x511130_b, 0x620120_b), free); + utils::hook::jump(SELECT_VALUE(0x511220_b, 0x620210_b), realloc); + utils::hook::jump(SELECT_VALUE(0x511050_b, 0x620040_b), _aligned_realloc); + } + } + }; +} + +REGISTER_COMPONENT(memory::component) \ No newline at end of file diff --git a/src/client/component/memory.hpp b/src/client/component/memory.hpp new file mode 100644 index 00000000..2d430224 --- /dev/null +++ b/src/client/component/memory.hpp @@ -0,0 +1,6 @@ +#pragma once + +namespace memory +{ + constexpr auto custom_script_mem_size = 0x1000000ui64; +} diff --git a/src/client/component/mods.cpp b/src/client/component/mods.cpp index aad8c6ac..cfabb84d 100644 --- a/src/client/component/mods.cpp +++ b/src/client/component/mods.cpp @@ -5,12 +5,14 @@ #include "command.hpp" #include "console.hpp" +#include "dvars.hpp" #include "filesystem.hpp" #include "fonts.hpp" #include "localized_strings.hpp" #include "materials.hpp" #include "mods.hpp" #include "scheduler.hpp" +#include "game/demonware/services.hpp" #include #include @@ -28,7 +30,6 @@ namespace mods { if (release_assets) { - materials::clear(); fonts::clear(); } @@ -53,8 +54,11 @@ namespace mods { if (game::environment::is_mp()) { - // vid_restart works on multiplayer, but not on singleplayer command::execute("vid_restart"); + scheduler::once([] + { + mods::read_stats(); + }, scheduler::main); return; } @@ -69,32 +73,37 @@ namespace mods return utils::io::file_exists(path + "/mod.ff") || utils::io::file_exists(path + "/zone/mod.ff"); } - void set_filesystem_data(const std::string& path) + void set_filesystem_data(const std::string& path, bool change_fs_game) { if (mod_path.has_value()) { filesystem::unregister_path(mod_path.value()); } - if (!game::environment::is_sp()) + if (change_fs_game) { - // modify fs_game on mp/dedi because its not set when we obviously vid_restart (sp does a full relaunch with command line arguments) - game::Dvar_SetFromStringByNameFromSource("fs_game", path.data(), - game::DVAR_SOURCE_INTERNAL); + game::Dvar_SetFromStringByNameFromSource("fs_game", path.data(), game::DVAR_SOURCE_INTERNAL); + } + + if (path != "") + { + filesystem::register_path(path); } } } - void set_mod(const std::string& path) + void set_mod(const std::string& path, bool change_fs_game) { - set_filesystem_data(path); - mod_path = path; - } + set_filesystem_data(path, change_fs_game); - void clear_mod() - { - set_filesystem_data(""); - mod_path.reset(); + if (path != "") + { + mod_path = path; + } + else + { + mod_path.reset(); + } } std::optional get_mod() @@ -102,6 +111,12 @@ namespace mods return mod_path; } + void read_stats() + { + demonware::set_storage_path(mod_path.value_or("")); + utils::hook::invoke(0x4E6B60_b, 0); // read stats + } + class component final : public component_interface { public: @@ -114,6 +129,12 @@ namespace mods db_release_xassets_hook.create(SELECT_VALUE(0x1F4DB0_b, 0x399740_b), db_release_xassets_stub); + dvars::callback::on_new_value("fs_game", [](game::dvar_value* value) + { + console::warn("fs_game value changed to '%s'\n", value->string); + set_mod(value->string, false); + }); + command::add("loadmod", [](const command::params& params) { if (params.size() < 2) @@ -143,7 +164,7 @@ namespace mods mod_requires_restart(path)) { console::info("Restarting...\n"); - full_restart("+set fs_game \""s + path + "\""); + full_restart("-mod \""s + path + "\""); } else { @@ -171,12 +192,12 @@ namespace mods if (mod_requires_restart(mod_path.value())) { console::info("Restarting...\n"); - clear_mod(); + set_mod(""); full_restart(""); } else { - clear_mod(); + set_mod(""); restart(); } }); diff --git a/src/client/component/mods.hpp b/src/client/component/mods.hpp index 6359a7ab..950e2c23 100644 --- a/src/client/component/mods.hpp +++ b/src/client/component/mods.hpp @@ -2,7 +2,7 @@ namespace mods { - void set_mod(const std::string& path); - void clear_mod(); + void set_mod(const std::string& path, bool change_fs_game = true); std::optional get_mod(); -} \ No newline at end of file + void read_stats(); +} diff --git a/src/client/component/motd.cpp b/src/client/component/motd.cpp new file mode 100644 index 00000000..0b8400c1 --- /dev/null +++ b/src/client/component/motd.cpp @@ -0,0 +1,37 @@ +#include +#include "loader/component_loader.hpp" + +#include "materials.hpp" + +#include "game/game.hpp" + +#include + +namespace motd +{ + class component final : public component_interface + { + public: + void post_load() override + { + std::thread([] + { + //auto data = utils::http::get_data("https://h1.gg/data/motd.png"); + //if (data.has_value()) + //{ + // materials::add("motd_image", data.value().buffer); + //} + }).detach(); + } + + void post_unpack() override + { + if (game::environment::is_sp()) + { + return; + } + } + }; +} + +REGISTER_COMPONENT(motd::component) \ No newline at end of file diff --git a/src/client/component/movement.cpp b/src/client/component/movement.cpp index e1b52750..85177539 100644 --- a/src/client/component/movement.cpp +++ b/src/client/component/movement.cpp @@ -112,7 +112,7 @@ namespace movement const auto ps = pm->ps; - ps->sprintButtonUpRequired = 1; + ps->sprintState.sprintButtonUpRequired = 1; float fmove{}, smove{}, wishspeed{}; game::vec3_t wishvel{}, wishdir{}; diff --git a/src/client/component/party.cpp b/src/client/component/party.cpp index a136f8b6..0ccf100b 100644 --- a/src/client/component/party.cpp +++ b/src/client/component/party.cpp @@ -11,11 +11,14 @@ #include "fastfiles.hpp" #include "mods.hpp" +#include "game/dvars.hpp" #include "game/game.hpp" #include "game/ui_scripting/execution.hpp" #include "steam/steam.hpp" +#include "utils/hash.hpp" + #include #include #include @@ -23,19 +26,14 @@ #include #include +#include + namespace party { namespace { - struct - { - game::netadr_s host{}; - std::string challenge{}; - bool hostDefined{false}; - } connect_state; - - std::string sv_motd; - int sv_maxclients; + connection_state server_connection_state{}; + std::optional server_discord_info{}; struct usermap_file { @@ -44,11 +42,20 @@ namespace party bool optional; }; + // snake case these names before release std::vector usermap_files = { - {".ff", "usermaphash", false}, - {"_load.ff", "usermaploadhash", true}, - {".arena", "usermaparenahash", true}, + {".ff", "usermap_hash", false}, + {"_load.ff", "usermap_load_hash", true}, + {".arena", "usermap_arena_hash", true}, + {".pak", "usermap_pak_hash", true}, + }; + + std::vector mod_files = + { + {".ff", "mod_hash", false}, + {"_pre_gfx.ff", "mod_pre_gfx_hash", true}, + {".pak", "mod_pak_hash", true}, }; struct @@ -57,6 +64,8 @@ namespace party utils::info_string info_string{}; } saved_info_response; + const game::dvar_t* sv_say_name = nullptr; + void perform_game_initialization() { command::execute("onlinegame 1", true); @@ -149,11 +158,11 @@ namespace party const char* get_didyouknow_stub(void* table, int row, int column) { - if (party::sv_motd.empty()) + if (server_connection_state.motd.empty()) { return utils::hook::invoke(0x5A0AC0_b, table, row, column); } - return utils::string::va("%s", party::sv_motd.data()); + return utils::string::va("%s", server_connection_state.motd.data()); } void disconnect() @@ -188,26 +197,46 @@ namespace party std::string get_file_hash(const std::string& file) { - if (!utils::io::file_exists(file)) - { - return {}; - } - const auto iter = hash_cache.find(file); if (iter != hash_cache.end()) { return iter->second; } - const auto data = utils::io::read_file(file); - const auto sha = utils::cryptography::sha1::compute(data, true); - hash_cache[file] = sha; - return sha; + const auto hash = utils::hash::get_file_hash(file); + if (!hash.empty()) + { + hash_cache.insert(std::make_pair(file, hash)); + } + + return hash; } std::string get_usermap_file_path(const std::string& mapname, const std::string& extension) { - return utils::string::va("usermaps\\%s\\%s%s", mapname.data(), mapname.data(), extension.data()); + return std::format("usermaps\\{}\\{}{}", mapname, mapname, extension); + } + + // generate hashes so they are cached + void generate_hashes(const std::string& mapname) + { + // usermap + for (const auto& file : usermap_files) + { + const auto path = get_usermap_file_path(mapname, file.extension); + get_file_hash(path); + } + + // mod + const auto fs_game = get_dvar_string("fs_game"); + if (!fs_game.empty()) + { + for (const auto& file : mod_files) + { + const auto path = std::format("{}\\mod{}", fs_game, file.extension); + get_file_hash(path); + } + } } void check_download_map(const utils::info_string& info, std::vector& files) @@ -223,21 +252,23 @@ namespace party throw std::runtime_error(utils::string::va("Invalid server mapname value %s\n", mapname.data())); } - const auto check_file = [&](const std::string& ext, const std::string& name, bool optional) + const auto check_file = [&](const usermap_file& file) { - const std::string filename = utils::string::va("usermaps/%s/%s%s", mapname.data(), mapname.data(), ext.data()); - const auto source_hash = info.get(name); + const std::string filename = utils::string::va("usermaps/%s/%s%s", + mapname.data(), mapname.data(), file.extension.data()); + const auto source_hash = info.get(file.name); if (source_hash.empty()) { - if (!optional) + if (!file.optional) { - throw std::runtime_error(utils::string::va("Server %s is empty", name.data())); + throw std::runtime_error(utils::string::va("Server %s is empty", file.name.data())); } return; } const auto hash = get_file_hash(filename); + console::debug("hash != source_hash => %s != %s\n", source_hash.data(), hash.data()); if (hash != source_hash) { files.emplace_back(filename, source_hash); @@ -245,9 +276,9 @@ namespace party } }; - for (const auto& [ext, name, opt] : usermap_files) + for (const auto& file : usermap_files) { - check_file(ext, name, opt); + check_file(file); } } @@ -264,7 +295,7 @@ namespace party if (server_fs_game.empty() && !client_fs_game.empty()) { - mods::clear_mod(); + mods::set_mod(""); return true; } @@ -273,35 +304,52 @@ namespace party throw std::runtime_error(utils::string::va("Invalid server fs_game value %s\n", server_fs_game.data())); } - const auto source_hash = info.get("modHash"); - if (source_hash.empty()) + auto needs_restart = false; + for (const auto& file : mod_files) { - throw std::runtime_error("Connection failed: Server mod hash is empty."); + const auto source_hash = info.get(file.name); + if (source_hash.empty()) + { + if (file.optional) + { + continue; + } + + throw std::runtime_error( + utils::string::va("Connection failed: Server %s is empty.", file.name.data())); + } + + const auto file_path = server_fs_game + "/mod" + file.extension; + auto has_to_download = !utils::io::file_exists(file_path); + + if (!has_to_download) + { + const auto hash = get_file_hash(file_path); + console::debug("has_to_download = %s != %s\n", source_hash.data(), hash.data()); + has_to_download = source_hash != hash; + } + + if (has_to_download) + { + // unload mod before downloading it + if (client_fs_game == server_fs_game) + { + mods::set_mod("", true); + return true; + } + else + { + files.emplace_back(file_path, source_hash); + } + } + else if (client_fs_game != server_fs_game) + { + mods::set_mod(server_fs_game); + needs_restart = true; + } } - const auto mod_path = server_fs_game + "/mod.ff"; - auto has_to_download = !utils::io::file_exists(mod_path); - - if (!has_to_download) - { - const auto data = utils::io::read_file(mod_path); - const auto hash = utils::cryptography::sha1::compute(data, true); - - has_to_download = source_hash != hash; - } - - if (has_to_download) - { - files.emplace_back(mod_path, source_hash); - return false; - } - else if (client_fs_game != server_fs_game) - { - mods::set_mod(server_fs_game); - return true; - } - - return false; + return needs_restart; } void close_joining_popups() @@ -404,6 +452,7 @@ namespace party needs_vid_restart = false; scheduler::once([=]() { + mods::read_stats(); connect(target); }, scheduler::pipeline::main); return true; @@ -428,21 +477,21 @@ namespace party fastfiles::set_usermap(mapname); - for (const auto& [ext, key, opt] : usermap_files) + for (const auto& file : usermap_files) { char buffer[0x100] = {0}; const std::string source_hash = game::MSG_ReadStringLine(msg, buffer, static_cast(sizeof(buffer))); - const auto path = get_usermap_file_path(mapname, ext); + const auto path = get_usermap_file_path(mapname, file.extension); const auto hash = get_file_hash(path); - if ((!source_hash.empty() && hash != source_hash) || (source_hash.empty() && !opt)) + if ((!source_hash.empty() && hash != source_hash) || (source_hash.empty() && !file.optional)) { command::execute("disconnect"); scheduler::once([] { - connect(connect_state.host); + connect(server_connection_state.host); }, scheduler::pipeline::main); return; } @@ -473,6 +522,12 @@ namespace party hash_cache.clear(); current_sv_mapname = map; + + if (game::environment::is_dedi()) + { + generate_hashes(map); + } + utils::hook::invoke(0x54BBB0_b, map, a2, a3, a4, a5); } @@ -496,19 +551,14 @@ namespace party line(current_sv_mapname); line(sv_gametype->current.string); - const auto add_hash = [&](const std::string extension) - { - const auto filename = get_usermap_file_path(current_sv_mapname, extension); - const auto hash = get_file_hash(filename); - line(hash); - }; - const auto is_usermap = fastfiles::usermap_exists(current_sv_mapname); - for (const auto& [ext, key, opt] : usermap_files) + for (const auto& file : usermap_files) { if (is_usermap) { - add_hash(ext); + const auto filename = get_usermap_file_path(current_sv_mapname, file.extension); + const auto hash = get_file_hash(filename); + line(hash); } else { @@ -557,7 +607,7 @@ namespace party void clear_sv_motd() { - party::sv_motd.clear(); + server_connection_state.motd.clear(); } int get_client_num_by_name(const std::string& name) @@ -579,9 +629,9 @@ namespace party return -1; } - void reset_connect_state() + void reset_server_connection_state() { - connect_state = {}; + server_connection_state = {}; } int get_client_count() @@ -634,16 +684,11 @@ namespace party command::execute("lui_open_popup popup_acceptinginvite", false); - connect_state.host = target; - connect_state.challenge = utils::cryptography::random::get_challenge(); - connect_state.hostDefined = true; + server_connection_state.host = target; + server_connection_state.challenge = utils::cryptography::random::get_challenge(); + server_connection_state.hostDefined = true; - network::send(target, "getInfo", connect_state.challenge); - } - - game::netadr_s get_state_host() - { - return connect_state.host; + network::send(target, "getInfo", server_connection_state.challenge); } void start_map(const std::string& mapname, bool dev) @@ -705,9 +750,14 @@ namespace party } } - int server_client_count() + connection_state get_server_connection_state() { - return party::sv_maxclients; + return server_connection_state; + } + + std::optional get_server_discord_info() + { + return server_discord_info; } class component final : public component_interface @@ -720,7 +770,7 @@ namespace party return; } - // detour CL_Disconnect to clear motd + // clear motd & usermap cl_disconnect_hook.create(0x12F080_b, cl_disconnect_stub); if (game::environment::is_mp()) @@ -792,7 +842,7 @@ namespace party command::add("reconnect", [](const command::params& argument) { - if (!connect_state.hostDefined) + if (!server_connection_state.hostDefined) { console::info("Cannot connect to server.\n"); return; @@ -805,7 +855,7 @@ namespace party } else { - connect(connect_state.host); + connect(server_connection_state.host); } }); @@ -908,8 +958,7 @@ namespace party scheduler::once([]() { - const auto hash = game::generateHashValue("sv_sayName"); - game::Dvar_RegisterString(hash, "sv_sayName", "console", game::DvarFlags::DVAR_FLAG_NONE); + sv_say_name = dvars::register_string("sv_sayName", "console", game::DvarFlags::DVAR_FLAG_NONE, "Custom name for RCON console"); }, scheduler::pipeline::main); command::add("tell", [](const command::params& params) @@ -921,7 +970,7 @@ namespace party const auto client_num = atoi(params.get(1)); const auto message = params.join(2); - const auto* const name = game::Dvar_FindVar("sv_sayName")->current.string; + const auto* const name = sv_say_name->current.string; game::SV_GameSendServerCommand(client_num, game::SV_CMD_CAN_IGNORE, utils::string::va("%c \"%s: %s\"", 84, name, message.data())); @@ -951,7 +1000,7 @@ namespace party } const auto message = params.join(1); - const auto* const name = game::Dvar_FindVar("sv_sayName")->current.string; + const auto* const name = sv_say_name->current.string; game::SV_GameSendServerCommand( -1, game::SV_CMD_CAN_IGNORE, utils::string::va("%c \"%s: %s\"", 84, name, message.data())); @@ -972,6 +1021,17 @@ namespace party console::info("%s\n", message.data()); }); + command::add("hash", [](const command::params& params) + { + if (params.size() < 2) + { + return; + } + + const auto hash = get_file_hash(params.get(1)); + console::info("hash output: %s\n", hash.data()); + }); + network::on("getInfo", [](const game::netadr_s& target, const std::string& data) { const auto mapname = get_dvar_string("mapname"); @@ -993,19 +1053,16 @@ namespace party info.set("sv_running", utils::string::va("%i", get_dvar_bool("sv_running") && !game::VirtualLobby_Loaded())); info.set("dedicated", utils::string::va("%i", get_dvar_bool("dedicated"))); info.set("sv_wwwBaseUrl", get_dvar_string("sv_wwwBaseUrl")); + info.set("sv_discordImageUrl", get_dvar_string("sv_discordImageUrl")); + info.set("sv_discordImageText", get_dvar_string("sv_discordImageText")); if (!fastfiles::is_stock_map(mapname)) { - const auto add_hash = [&](const std::string& extension, const std::string& name) + for (const auto& file : usermap_files) { - const auto path = get_usermap_file_path(mapname, extension); + const auto path = get_usermap_file_path(mapname, file.extension); const auto hash = get_file_hash(path); - info.set(name, hash); - }; - - for (const auto& [ext, name, opt] : usermap_files) - { - add_hash(ext, name); + info.set(file.name, hash); } } @@ -1014,8 +1071,12 @@ namespace party if (!fs_game.empty()) { - const auto hash = get_file_hash(utils::string::va("%s/mod.ff", fs_game.data())); - info.set("modHash", hash); + for (const auto& file : mod_files) + { + const auto hash = get_file_hash(utils::string::va("%s/mod%s", + fs_game.data(), file.extension.data())); + info.set(file.name, hash); + } } network::send(target, "infoResponse", info.build(), '\n'); @@ -1026,7 +1087,7 @@ namespace party const utils::info_string info(data); server_list::handle_info_response(target, info); - if (connect_state.host != target) + if (server_connection_state.host != target) { return; } @@ -1035,7 +1096,14 @@ namespace party saved_info_response.host = target; saved_info_response.info_string = info; - if (info.get("challenge") != connect_state.challenge) + const auto protocol = info.get("protocol"); + if (std::atoi(protocol.data()) != PROTOCOL) + { + menu_error("Connection failed: Invalid protocol."); + return; + } + + if (info.get("challenge") != server_connection_state.challenge) { menu_error("Connection failed: Invalid challenge."); return; @@ -1081,8 +1149,17 @@ namespace party return; } - party::sv_motd = info.get("sv_motd"); - party::sv_maxclients = std::stoi(info.get("sv_maxclients")); + server_connection_state.motd = info.get("sv_motd"); + server_connection_state.max_clients = std::stoi(info.get("sv_maxclients")); + server_connection_state.base_url = info.get("sv_wwwBaseUrl"); + + discord_information discord_info{}; + discord_info.image = info.get("sv_discordImageUrl"); + discord_info.image_text = info.get("sv_discordImageText"); + if (!discord_info.image.empty() || !discord_info.image_text.empty()) + { + server_discord_info.emplace(discord_info); + } connect_to_party(target, mapname, gametype); }); @@ -1090,4 +1167,4 @@ namespace party }; } -REGISTER_COMPONENT(party::component) \ No newline at end of file +REGISTER_COMPONENT(party::component) diff --git a/src/client/component/party.hpp b/src/client/component/party.hpp index ca4852b2..a9c387bf 100644 --- a/src/client/component/party.hpp +++ b/src/client/component/party.hpp @@ -3,19 +3,34 @@ namespace party { - std::string get_www_url(); + struct connection_state + { + game::netadr_s host; + std::string challenge; + bool hostDefined; + std::string motd; + int max_clients; + std::string base_url; + }; + + struct discord_information + { + std::string image; + std::string image_text; + }; + void user_download_response(bool response); void menu_error(const std::string& error); - void reset_connect_state(); + void reset_server_connection_state(); void connect(const game::netadr_s& target); void start_map(const std::string& mapname, bool dev = false); void clear_sv_motd(); - game::netadr_s get_state_host(); - int server_client_count(); + connection_state get_server_connection_state(); + std::optional get_server_discord_info(); int get_client_num_by_name(const std::string& name); diff --git a/src/client/component/patches.cpp b/src/client/component/patches.cpp index 1dfd1b8f..c1dce5eb 100644 --- a/src/client/component/patches.cpp +++ b/src/client/component/patches.cpp @@ -68,7 +68,7 @@ namespace patches void cg_set_client_dvar_from_server_stub(void* clientNum, void* cgameGlob, const char* dvar_hash, const char* value) { - int hash = atoi(dvar_hash); + const auto hash = std::atoi(dvar_hash); auto* dvar = game::Dvar_FindMalleableVar(hash); if (hash == game::generateHashValue("cg_fov") || @@ -192,7 +192,7 @@ namespace patches void sv_execute_client_message_stub(game::mp::client_t* client, game::msg_t* msg) { - if (client->reliableAcknowledge < 0) + if ((client->reliableSequence - client->reliableAcknowledge) < 0) { client->reliableAcknowledge = client->reliableSequence; console::info("Negative reliableAcknowledge from %s - cl->reliableSequence is %i, reliableAcknowledge is %i\n", @@ -223,7 +223,7 @@ namespace patches utils::hook::detour init_network_dvars_hook; void init_network_dvars_stub(game::dvar_t* dvar) { - static const auto hash = game::generateHashValue("r_tonemapHighlightRange"); + constexpr auto hash = dvars::generate_hash("r_tonemapHighlightRange"); if (dvar->hash == hash) { init_network_dvars_hook.invoke(dvar); @@ -244,26 +244,6 @@ namespace patches } } - int out_of_memory_text_stub(char* dest, int size, const char* fmt, ...) - { - fmt = "%s (%d)\n\n" - "Disable shader caching, lower graphic settings, free up RAM, or update your GPU drivers.\n\n" - "If this still occurs, try using the '-memoryfix' parameter to generate the 'players2' folder."; - - char buffer[2048]; - - { - va_list ap; - va_start(ap, fmt); - - vsnprintf_s(buffer, sizeof(buffer), _TRUNCATE, fmt, ap); - - va_end(ap); - } - - return utils::hook::invoke(SELECT_VALUE(0x429200_b, 0x5AF0F0_b), dest, size, "%s", buffer); - } - void create_2d_texture_stub_1(const char* fmt, ...) { fmt = "Create2DTexture( %s, %i, %i, %i, %i ) failed\n\n" @@ -351,9 +331,18 @@ namespace patches dvars::override::register_float("cg_fovScale", 1.f, 0.1f, 2.f, game::DvarFlags::DVAR_FLAG_SAVED); dvars::override::register_float("cg_fovMin", 1.f, 1.0f, 90.f, game::DvarFlags::DVAR_FLAG_SAVED); + // Enable Marketing Comms + dvars::override::register_int("marketing_active", 1, 1, 1, game::DVAR_FLAG_WRITE); + + // Makes com_maxfps saved dvar + dvars::override::register_int("com_maxfps", 0, 0, 1000, game::DVAR_FLAG_SAVED); + // Makes mis_cheat saved dvar dvars::override::register_bool("mis_cheat", 0, game::DVAR_FLAG_SAVED); + // Fix speaker config bug + dvars::override::register_int("snd_detectedSpeakerConfig", 0, 0, 100, 0); + // Allow kbam input when gamepad is enabled utils::hook::nop(SELECT_VALUE(0x1AC0CE_b, 0x135EFB_b), 2); utils::hook::nop(SELECT_VALUE(0x1A9DDC_b, 0x13388F_b), 6); @@ -368,18 +357,6 @@ namespace patches utils::hook::call(SELECT_VALUE(0x55E919_b, 0x681A69_b), create_2d_texture_stub_1); // Sys_Error for "Create2DTexture( %s, %i, %i, %i, %i ) failed" utils::hook::call(SELECT_VALUE(0x55EACB_b, 0x681C1B_b), create_2d_texture_stub_2); // Com_Error for ^ utils::hook::call(SELECT_VALUE(0x5B35BA_b, 0x6CB1BC_b), swap_chain_stub); // Com_Error for "IDXGISwapChain::Present failed: %s" - utils::hook::call(SELECT_VALUE(0x457BC9_b, 0x1D8E09_b), out_of_memory_text_stub); // Com_sprintf for "Out of memory. You are probably low on disk space." - - // "fix" for rare 'Out of memory error' error - // this will *at least* generate the configs for mp/sp, which is the #1 issue - if (utils::flags::has_flag("memoryfix")) - { - utils::hook::jump(SELECT_VALUE(0x5110D0_b, 0x6200C0_b), malloc); - utils::hook::jump(SELECT_VALUE(0x510FF0_b, 0x61FFE0_b), _aligned_malloc); - utils::hook::jump(SELECT_VALUE(0x511130_b, 0x620120_b), free); - utils::hook::jump(SELECT_VALUE(0x511220_b, 0x620210_b), realloc); - utils::hook::jump(SELECT_VALUE(0x511050_b, 0x620040_b), _aligned_realloc); - } // Uncheat protect gamepad-related dvars dvars::override::register_float("gpad_button_deadzone", 0.13f, 0, 1, game::DVAR_FLAG_SAVED); @@ -438,7 +415,7 @@ namespace patches // prevent servers overriding our fov utils::hook::nop(0x17DA96_b, 0x16); utils::hook::nop(0xE00BE_b, 0x17); - utils::hook::set(0x307F39_b, 0xEB); + utils::hook::nop(0x307F90_b, 0x5); // don't change cg_fov when toggling third person spectating // make setclientdvar behave like older games cg_set_client_dvar_from_server_hook.create(0x11AA90_b, cg_set_client_dvar_from_server_stub); @@ -474,8 +451,6 @@ namespace patches dvars::override::register_bool("ui_drawCrosshair", true, game::DVAR_FLAG_WRITE); utils::hook::jump(0x1E6010_b, ui_draw_crosshair); - dvars::override::register_int("com_maxfps", 0, 0, 1000, game::DVAR_FLAG_SAVED); - // Prevent clients from ending the game as non host by sending 'end_game' lui notification cmd_lui_notify_server_hook.create(0x412D50_b, cmd_lui_notify_server_stub); @@ -497,4 +472,4 @@ namespace patches }; } -REGISTER_COMPONENT(patches::component) \ No newline at end of file +REGISTER_COMPONENT(patches::component) diff --git a/src/client/component/redirect.cpp b/src/client/component/redirect.cpp index e44be742..1fa38f44 100644 --- a/src/client/component/redirect.cpp +++ b/src/client/component/redirect.cpp @@ -21,10 +21,10 @@ namespace redirect ZeroMemory(&process_info, sizeof(process_info)); startup_info.cb = sizeof(startup_info); - auto* arguments = const_cast(utils::string::va("%s%s%s", self.get_path().data(), + auto* arguments = const_cast(utils::string::va("%s%s%s", self.get_path().generic_string().data(), (singleplayer ? " -singleplayer" : " -multiplayer"), (mode.empty() ? "" : (" +"s + mode).data()))); - CreateProcessA(self.get_path().data(), arguments, nullptr, nullptr, false, NULL, nullptr, nullptr, + CreateProcessA(self.get_path().generic_string().data(), arguments, nullptr, nullptr, false, NULL, nullptr, nullptr, &startup_info, &process_info); if (process_info.hThread && process_info.hThread != INVALID_HANDLE_VALUE) diff --git a/src/client/component/renderer.cpp b/src/client/component/renderer.cpp index f2758790..3c0e8b0e 100644 --- a/src/client/component/renderer.cpp +++ b/src/client/component/renderer.cpp @@ -132,6 +132,21 @@ namespace renderer a.jmp(0x1C4136_b); }); } + + void r_preload_shaders_stub(utils::hook::assembler& a) + { + const auto is_zero = a.newLabel(); + + a.mov(rax, qword_ptr(SELECT_VALUE(0x123FFF30_b, 0x111DC230_b))); + a.test(rax, rax); + a.jz(is_zero); + + a.mov(rcx, qword_ptr(rax, 0x540C68)); + a.jmp(SELECT_VALUE(0x5CF1FF_b, 0x6E76FF_b)); + + a.bind(is_zero); + a.jmp(SELECT_VALUE(0x5CF20A_b, 0x6E7722_b)); + } } class component final : public component_interface @@ -166,6 +181,10 @@ namespace renderer r_use_custom_red_dot_brightness = dvars::register_bool("r_useCustomRedDotBrightness", true, game::DVAR_FLAG_SAVED, "Use custom red-dot brightness values"); } + + // patch r_preloadShaders crash at init + utils::hook::jump(SELECT_VALUE(0x5CF1F1_b, 0x6E76F1_b), utils::hook::assemble(r_preload_shaders_stub), true); + dvars::override::register_bool("r_preloadShaders", false, game::DVAR_FLAG_SAVED); } }; } diff --git a/src/client/component/scripting.cpp b/src/client/component/scripting.cpp index 3f9e9ad9..6b9e1b35 100644 --- a/src/client/component/scripting.cpp +++ b/src/client/component/scripting.cpp @@ -28,6 +28,7 @@ namespace scripting utils::concurrency::container shared_table; std::string current_file; + unsigned int current_file_id{}; namespace { @@ -46,8 +47,7 @@ namespace scripting utils::hook::detour db_find_xasset_header_hook; - std::string current_script_file; - unsigned int current_file_id{}; + const char* current_script_file_name; game::dvar_t* g_dump_scripts; @@ -63,7 +63,7 @@ namespace scripting const auto* string = game::SL_ConvertToString(string_value); if (string) { - event e; + event e{}; e.name = string; e.entity = notify_list_owner_id; @@ -97,20 +97,16 @@ namespace scripting game::G_LogPrintf("InitGame\n"); lua::engine::start(); - - gsc::load_main_handles(); } + gsc::load_main_handles(); + g_load_structs_hook.invoke(); } void scr_load_level_stub() { - if (!game::VirtualLobby_Loaded()) - { - gsc::load_init_handles(); - } - + gsc::load_init_handles(); scr_load_level_hook.invoke(); } @@ -157,12 +153,13 @@ namespace scripting void process_script_stub(const char* filename) { - current_script_file = filename; + current_script_file_name = filename; const auto file_id = atoi(filename); if (file_id) { current_file_id = static_cast(file_id); + current_file = scripting::get_token(current_file_id); } else { @@ -176,14 +173,10 @@ namespace scripting void add_function_sort(unsigned int id, const char* pos) { std::string filename = current_file; - if (current_file_id) - { - filename = scripting::get_token(current_file_id); - } if (!script_function_table_sort.contains(filename)) { - const auto script = gsc::find_script(game::ASSET_TYPE_SCRIPTFILE, current_script_file.data(), false); + const auto script = gsc::find_script(game::ASSET_TYPE_SCRIPTFILE, current_script_file_name, false); if (script) { const auto end = &script->bytecode[script->bytecodeLen]; @@ -207,15 +200,7 @@ namespace scripting { add_function_sort(thread_name, code_pos); - if (current_file_id) - { - const auto name = get_token(current_file_id); - add_function(name, thread_name, code_pos); - } - else - { - add_function(current_file, thread_name, code_pos); - } + add_function(current_file, thread_name, code_pos); scr_set_thread_position_hook.invoke(thread_name, code_pos); } diff --git a/src/client/component/scripting.hpp b/src/client/component/scripting.hpp index 12598b9c..14f2bd49 100644 --- a/src/client/component/scripting.hpp +++ b/src/client/component/scripting.hpp @@ -13,6 +13,7 @@ namespace scripting extern utils::concurrency::container shared_table; extern std::string current_file; + extern unsigned int current_file_id; void on_shutdown(const std::function& callback); std::optional get_canonical_string(const unsigned int id); diff --git a/src/client/component/server_list.cpp b/src/client/component/server_list.cpp index 7dc1bde8..e3577e2e 100644 --- a/src/client/component/server_list.cpp +++ b/src/client/component/server_list.cpp @@ -33,6 +33,7 @@ namespace server_list std::string host_name; std::string map_name; std::string game_type; + std::string mod_name; game::CodPlayMode play_mode; char in_game; game::netadr_s address; @@ -76,7 +77,7 @@ namespace server_list server_list_page = 0; } - party::reset_connect_state(); + party::reset_server_connection_state(); if (get_master_server(master_state.address)) { @@ -131,39 +132,29 @@ namespace server_list return ""; } - if (column == 0) + switch (column) { - return servers[i].host_name.empty() ? "" : utils::string::va("%s", servers[i].host_name.data()); - } - - if (column == 1) - { - return servers[i].map_name.empty() ? "Unknown" : utils::string::va("%s", servers[i].map_name.data()); - } - - if (column == 2) + case 0: + return servers[i].host_name.empty() ? "" : servers[i].host_name.data(); + case 1: + return servers[i].map_name.empty() ? "Unknown" : servers[i].map_name.data(); + case 2: { const auto client_count = servers[i].clients - servers[i].bots; return utils::string::va("%d/%d [%d]", client_count, servers[i].max_clients, servers[i].clients); } - - if (column == 3) - { - return servers[i].game_type.empty() ? "" : utils::string::va("%s", servers[i].game_type.data()); - } - - if (column == 4) - { - return servers[i].game_type.empty() ? "" : utils::string::va("%i", servers[i].ping); - } - - if (column == 5) - { + case 3: + return servers[i].game_type.empty() ? "" : servers[i].game_type.data(); + case 4: + return servers[i].ping ? utils::string::va("%i", servers[i].ping) : "999"; + case 5: return servers[i].is_private ? "1" : "0"; + case 6: + return servers[i].mod_name.empty() ? "" : servers[i].mod_name.data(); + default: + return ""; } - - return ""; } void sort_serverlist() @@ -325,6 +316,13 @@ namespace server_list void handle_info_response(const game::netadr_s& address, const utils::info_string& info) { + // Don't show servers that aren't using the same protocol! + const auto protocol = std::atoi(info.get("protocol").data()); + if (protocol != PROTOCOL) + { + return; + } + // Don't show servers that aren't dedicated! const auto dedicated = std::atoi(info.get("dedicated").data()); if (!dedicated) @@ -372,6 +370,7 @@ namespace server_list server.host_name = info.get("hostname"); server.map_name = game::UI_GetMapDisplayName(info.get("mapname").data()); server.game_type = game::UI_GetGameTypeDisplayName(info.get("gametype").data()); + server.mod_name = info.get("fs_game"); server.play_mode = playmode; server.clients = atoi(info.get("clients").data()); server.max_clients = atoi(info.get("sv_maxclients").data()); @@ -411,7 +410,7 @@ namespace server_list scheduler::once([]() { // add dvars to change destination master server ip/port - master_server_ip = dvars::register_string("masterServerIP", "h1.fed0001.xyz", game::DVAR_FLAG_NONE, + master_server_ip = dvars::register_string("masterServerIP", "h1.fed.cat", game::DVAR_FLAG_NONE, "IP of the destination master server to connect to"); master_server_port = dvars::register_string("masterServerPort", "20810", game::DVAR_FLAG_NONE, "Port of the destination master server to connect to"); diff --git a/src/client/component/thirdperson.cpp b/src/client/component/thirdperson.cpp new file mode 100644 index 00000000..2c6359ba --- /dev/null +++ b/src/client/component/thirdperson.cpp @@ -0,0 +1,84 @@ +#include +#include "loader/component_loader.hpp" + +#include "game/game.hpp" +#include "game/dvars.hpp" + +#include "scheduler.hpp" + +#include + +namespace thirdperson +{ + namespace + { + game::dvar_t* cg_thirdPerson = nullptr; + game::dvar_t* cg_thirdPersonRange = nullptr; + game::dvar_t* cg_thirdPersonAngle = nullptr; + + namespace mp + { + __int64 sub_1D5950_stub([[maybe_unused]] int local_client_num, game::mp::cg_s* a2) + { + auto next_snap = a2->nextSnap; + if (next_snap->ps.pm_type < 7u) + { + int link_flags = next_snap->ps.linkFlags; + if ((link_flags & 2) == 0 && (next_snap->ps.otherFlags & 4) == 0) + { + auto client_globals = a2; + if (!client_globals->unk_979676 || !client_globals->unk_979696) + { + if (cg_thirdPerson && cg_thirdPerson->current.enabled) + { + return 1; + } + + if (!(link_flags & (1 << 0xE)) || client_globals->unk_979696) + return (link_flags >> 27) & 1; + if (link_flags & (1 << 0x1D)) + return 0; + if (!(link_flags & (1 << 0x1C))) + return a2->unk_601088; + } + } + } + return 1; + } + + void sub_10C280_stub(int local_client_num, float angle, float range, int a4, int a5, int a6, int a7) + { + angle = cg_thirdPersonAngle->current.value; + range = cg_thirdPersonRange->current.value; + utils::hook::invoke(0x10C280_b, local_client_num, angle, range, a4, a5, a6, a7); + } + } + + } + + class component final : public component_interface + { + public: + void post_unpack() override + { + if (!game::environment::is_mp()) + { + return; + } + + scheduler::once([]() + { + cg_thirdPerson = dvars::register_bool("cg_thirdPerson", 0, 4, "Use third person view"); + cg_thirdPersonAngle = dvars::register_float("cg_thirdPersonAngle", 356.0f, -180.0f, 360.0f, 4, + "The angle of the camera from the player in third person view"); + cg_thirdPersonRange = dvars::register_float("cg_thirdPersonRange", 120.0f, 0.0f, 1024, 4, + "The range of the camera from the player in third person view"); + }, scheduler::main); + + utils::hook::jump(0x1D5950_b, mp::sub_1D5950_stub); + utils::hook::call(0x10C26B_b, mp::sub_10C280_stub); + } + }; +} + +REGISTER_COMPONENT(thirdperson::component) diff --git a/src/client/component/ui_scripting.cpp b/src/client/component/ui_scripting.cpp index 1fcfb1d1..bfb8730d 100644 --- a/src/client/component/ui_scripting.cpp +++ b/src/client/component/ui_scripting.cpp @@ -9,6 +9,7 @@ #include "localized_strings.hpp" #include "console.hpp" +#include "discord.hpp" #include "download.hpp" #include "game_module.hpp" #include "fps.hpp" @@ -33,6 +34,8 @@ #include "steam/steam.hpp" +#include + namespace ui_scripting { namespace @@ -318,7 +321,7 @@ namespace ui_scripting game_type["virtuallobbypresentable"] = [](const game&) { - ::game::Dvar_SetFromStringByNameFromSource("virtualLobbyPresentable", "1", ::game::DvarSetSource::DVAR_SOURCE_INTERNAL); + ::game::Dvar_SetFromStringByNameFromSource("virtualLobbyPresentable", "1", ::game::DVAR_SOURCE_INTERNAL); }; game_type["getcurrentgamelanguage"] = [](const game&) @@ -332,6 +335,29 @@ namespace ui_scripting material.data())); }; + game_type["getcommandbind"] = [](const game&, const std::string& cmd) + { + const auto binding = ::game::Key_GetBindingForCmd(cmd.data()); + auto key = -1; + for (auto i = 0; i < 256; i++) + { + if (::game::playerKeys[0].keys[i].binding == binding) + { + key = i; + } + } + + if (key == -1) + { + return ::game::UI_SafeTranslateString("KEY_UNBOUND"); + } + else + { + const auto loc_string = ::game::Key_KeynumToString(key, 1, 0); + return ::game::UI_SafeTranslateString(loc_string); + } + }; + auto server_list_table = table(); lua["serverlist"] = server_list_table; @@ -367,7 +393,29 @@ namespace ui_scripting download_table["abort"] = download::stop_download; download_table["userdownloadresponse"] = party::user_download_response; - download_table["getwwwurl"] = party::get_www_url; + download_table["getwwwurl"] = party::get_server_connection_state().base_url; + + auto discord_table = table(); + lua["discord"] = discord_table; + + discord_table["respond"] = discord::respond; + + discord_table["getavatarmaterial"] = [](const std::string& id) + -> script_value + { + const auto material = discord::get_avatar_material(id); + if (material == nullptr) + { + return {}; + } + + return lightuserdata(material); + }; + + discord_table["reply"] = table(); + discord_table["reply"]["yes"] = DISCORD_REPLY_YES; + discord_table["reply"]["ignore"] = DISCORD_REPLY_IGNORE; + discord_table["reply"]["no"] = DISCORD_REPLY_NO; } void start() @@ -379,7 +427,26 @@ namespace ui_scripting setup_functions(); - lua["print"] = function(reinterpret_cast(SELECT_VALUE(0x93490_b, 0x209EB0_b))); + lua["print"] = [](const variadic_args& va) + { + std::string buffer{}; + const auto to_string = get_globals()["tostring"]; + + for (auto i = 0; i < va.size(); i++) + { + const auto& arg = va[i]; + const auto str = to_string(arg)[0].as(); + buffer.append(str); + + if (i < va.size() - 1) + { + buffer.append("\t"); + } + } + + console::info("%s\n", buffer.data()); + }; + lua["table"]["unpack"] = lua["unpack"]; lua["luiglobals"] = lua; @@ -553,6 +620,8 @@ namespace ui_scripting return; } + dvars::register_bool("r_preloadShadersFrontendAllow", true, game::DVAR_FLAG_SAVED, "Allow shader popup on startup"); + utils::hook::call(SELECT_VALUE(0xE7419_b, 0x25E809_b), db_find_x_asset_header_stub); utils::hook::call(SELECT_VALUE(0xE72CB_b, 0x25E6BB_b), db_find_x_asset_header_stub); diff --git a/src/client/component/updater.cpp b/src/client/component/updater.cpp index dcb44584..7d56e769 100644 --- a/src/client/component/updater.cpp +++ b/src/client/component/updater.cpp @@ -14,13 +14,12 @@ #include #include -#include #include #include #include #include -#define MASTER "https://master.fed0001.xyz/h1-mod/" +#define MASTER "https://h1-mod.fed.cat/" #define FILES_PATH "files.json" #define FILES_PATH_DEV "files-dev.json" @@ -161,10 +160,17 @@ namespace updater { return utils::string::va("%i", uint32_t(time(nullptr))); } - - std::optional download_file(const std::string& name) + + std::optional download_data_file(const std::string& name) { - return utils::http::get_data(MASTER + select(DATA_PATH, DATA_PATH_DEV) + name + "?" + get_time_str()); + const auto file = std::format("{}{}?{}", select(DATA_PATH, DATA_PATH_DEV), name, get_time_str()); + return updater::get_server_file(file); + } + + std::optional download_file_list() + { + const auto file = std::format("{}?{}", select(FILES_PATH, FILES_PATH_DEV), get_time_str()); + return updater::get_server_file(file); } bool has_old_data_files() @@ -301,6 +307,34 @@ namespace updater } } + std::optional get_server_file(const std::string& endpoint) + { + static std::vector server_urls = + { + {"https://h1-mod.fed.cat/"}, + {"https://master.fed0001.xyz/h1-mod/"}, // remove this at some point + }; + + const auto try_url = [&](const std::string& base_url) + { + const auto url = base_url + endpoint; + console::debug("[HTTP] GET file \"%s\"\n", url.data()); + const auto result = utils::http::get_data(url); + return result; + }; + + for (const auto& url : server_urls) + { + const auto result = try_url(url); + if (result.has_value()) + { + return result; + } + } + + return {}; + } + void relaunch() { const auto mode = game::environment::is_mp() ? "-multiplayer" : "-singleplayer"; @@ -406,7 +440,7 @@ namespace updater scheduler::once([]() { - const auto files_data = utils::http::get_data(MASTER + select(FILES_PATH, FILES_PATH_DEV) + "?" + get_time_str()); + const auto files_data = download_file_list(); if (is_update_cancelled()) { @@ -546,7 +580,7 @@ namespace updater console::debug("[Updater] downloading file %s\n", file.data()); - const auto data = download_file(file); + const auto data = download_data_file(file); if (is_update_cancelled()) { diff --git a/src/client/component/updater.hpp b/src/client/component/updater.hpp index 0301a6c3..ba8039c2 100644 --- a/src/client/component/updater.hpp +++ b/src/client/component/updater.hpp @@ -2,8 +2,12 @@ #define CLIENT_DATA_FOLDER "cdata" +#include + namespace updater { + std::optional get_server_file(const std::string& endpoint); + void relaunch(); void set_has_tried_update(bool tried); diff --git a/src/client/component/weapon.cpp b/src/client/component/weapon.cpp index e7e2342f..3b2c1713 100644 --- a/src/client/component/weapon.cpp +++ b/src/client/component/weapon.cpp @@ -6,8 +6,10 @@ #include "fastfiles.hpp" #include "game/game.hpp" +#include "game/dvars.hpp" #include +#include namespace weapon { @@ -37,27 +39,38 @@ namespace weapon // precache items for (std::size_t i = 0; i < weapons.size(); i++) { - console::debug("precaching weapon \"%s\"\n", weapons[i]->name); + //console::debug("precaching weapon \"%s\"\n", weapons[i]->name); game::G_GetWeaponForName(weapons[i]->name); } } utils::hook::detour xmodel_get_bone_index_hook; - __int64 xmodel_get_bone_index_stub(game::XModel* model, game::scr_string_t name, unsigned int offset, char* index) + int xmodel_get_bone_index_stub(game::XModel* model, game::scr_string_t name, unsigned int offset, char* index) { - auto result = xmodel_get_bone_index_hook.invoke<__int64>(model, name, offset, index); - if (!result) + auto result = xmodel_get_bone_index_hook.invoke(model, name, offset, index); + if (result) { - if (name == game::SL_GetString("tag_weapon_right", 0) || - name == game::SL_GetString("tag_knife_attach", 0)) + return result; + } + + const auto original_index = *index; + const auto original_result = result; + + if (name == game::SL_FindString("tag_weapon_right") || + name == game::SL_FindString("tag_knife_attach")) + { + const auto tag_weapon = game::SL_FindString("tag_weapon"); + result = xmodel_get_bone_index_hook.invoke(model, tag_weapon, offset, index); + if (result) { - result = xmodel_get_bone_index_hook.invoke<__int64>(model, game::SL_GetString("tag_weapon", 0), offset, index); - if (result) - { - console::debug("using tag_weapon instead of %s (%s, %d)\n", game::SL_ConvertToString(name), model->name, offset); - } + console::debug("using tag_weapon instead of %s (%s, %d, %d)\n", game::SL_ConvertToString(name), model->name, offset, *index); + return result; } } + + *index = original_index; + result = original_result; + return result; } @@ -116,6 +129,131 @@ namespace weapon { set_weapon_field(weapon_name, field, value); } + + int compare_hash(const void* a, const void* b) + { + const auto hash_a = reinterpret_cast( + reinterpret_cast(a))->hash; + const auto hash_b = reinterpret_cast( + reinterpret_cast(b))->hash; + + if (hash_a < hash_b) + { + return -1; + } + else if (hash_a > hash_b) + { + return 1; + } + + return 0; + } + + utils::memory::allocator ddl_allocator; + std::unordered_set modified_enums; + + std::vector get_stringtable_entries(const std::string& name) + { + std::vector entries; + + const auto string_table = game::DB_FindXAssetHeader( + game::ASSET_TYPE_STRINGTABLE, name.data(), false).stringTable; + + if (string_table == nullptr) + { + return entries; + } + + for (auto row = 0; row < string_table->rowCount; row++) + { + if (string_table->columnCount <= 0) + { + continue; + } + + const auto index = (row * string_table->columnCount); + const auto weapon = string_table->values[index].string; + entries.push_back(ddl_allocator.duplicate_string(weapon)); + } + + return entries; + } + + void add_entries_to_enum(game::DDLEnum* enum_, const std::vector entries) + { + if (entries.size() <= 0) + { + return; + } + + const auto new_size = enum_->memberCount + entries.size(); + const auto members = ddl_allocator.allocate_array(new_size); + const auto hash_list = ddl_allocator.allocate_array(new_size); + + std::memcpy(members, enum_->members, 8 * enum_->memberCount); + std::memcpy(hash_list, enum_->hashTable.list, 8 * enum_->hashTable.count); + + for (auto i = 0; i < entries.size(); i++) + { + const auto hash = utils::hook::invoke(0x794FB0_b, entries[i], 0); + const auto index = enum_->memberCount + i; + hash_list[index].index = index; + hash_list[index].hash = hash; + members[index] = entries[i]; + } + + std::qsort(hash_list, new_size, sizeof(game::DDLHash), compare_hash); + + enum_->members = members; + enum_->hashTable.list = hash_list; + enum_->memberCount = static_cast(new_size); + enum_->hashTable.count = static_cast(new_size); + } + + void load_ddl_asset_stub(game::DDLRoot** asset) + { + const auto root = *asset; + if (!root->ddlDef) + { + return utils::hook::invoke(0x39BE20_b, root); + } + + auto ddl_def = root->ddlDef; + while (ddl_def) + { + for (auto i = 0; i < ddl_def->enumCount; i++) + { + const auto enum_ = &ddl_def->enumList[i]; + if (modified_enums.contains(enum_)) + { + continue; + } + + if ((enum_->name == "WeaponStats"s || enum_->name == "Weapon"s)) + { + const auto weapons = get_stringtable_entries("mp/customweapons.csv"); + add_entries_to_enum(enum_, weapons); + modified_enums.insert(enum_); + } + + if (enum_->name == "AttachmentBase"s) + { + const auto attachments = get_stringtable_entries("mp/customattachments.csv"); + add_entries_to_enum(enum_, attachments); + modified_enums.insert(enum_); + } + } + + ddl_def = ddl_def->next; + } + + utils::hook::invoke(0x39BE20_b, asset); + } + } + + void clear_modifed_enums() + { + modified_enums.clear(); } class component final : public component_interface @@ -137,6 +275,11 @@ namespace weapon // patch attachment configstring so it will create if not found utils::hook::call(0x41C595_b, g_find_config_string_index_stub); + + utils::hook::call(0x36B4D4_b, load_ddl_asset_stub); + + dvars::register_bool("sv_disableCustomClasses", + false, game::DVAR_FLAG_REPLICATED, "Disable custom classes on server"); } #ifdef DEBUG diff --git a/src/client/component/weapon.hpp b/src/client/component/weapon.hpp new file mode 100644 index 00000000..f0033b2e --- /dev/null +++ b/src/client/component/weapon.hpp @@ -0,0 +1,6 @@ +#pragma once + +namespace weapon +{ + void clear_modifed_enums(); +} \ No newline at end of file diff --git a/src/client/game/demonware/data_types.hpp b/src/client/game/demonware/data_types.hpp index 9b6a909b..8d1bb377 100644 --- a/src/client/game/demonware/data_types.hpp +++ b/src/client/game/demonware/data_types.hpp @@ -237,4 +237,29 @@ namespace demonware buffer->read_string(&this->filename); } }; + + class bdMarketingMessage final : public bdTaskResult + { + public: + uint64_t m_messageID; + std::string m_languageCode; + std::string m_content; + std::string m_metadata; + + void serialize(byte_buffer* buffer) override + { + buffer->write_uint64(this->m_messageID); + buffer->write_string(this->m_languageCode); + buffer->write_blob(this->m_content); + buffer->write_blob(this->m_metadata); + } + + void deserialize(byte_buffer* buffer) override + { + buffer->read_uint64(&this->m_messageID); + buffer->read_string(&this->m_languageCode); + buffer->read_blob(&this->m_content); + buffer->read_blob(&this->m_metadata); + } + }; } diff --git a/src/client/game/demonware/services/bdMarketingComms.cpp b/src/client/game/demonware/services/bdMarketingComms.cpp index 409b278a..aa6ac61c 100644 --- a/src/client/game/demonware/services/bdMarketingComms.cpp +++ b/src/client/game/demonware/services/bdMarketingComms.cpp @@ -1,4 +1,6 @@ #include + +#include #include "../services.hpp" namespace demonware @@ -7,11 +9,71 @@ namespace demonware { this->register_task(1, &bdMarketingComms::getMessages); this->register_task(4, &bdMarketingComms::reportFullMessagesViewed); + this->register_task(5, &bdMarketingComms::getMessages_); } void bdMarketingComms::getMessages(service_server* server, byte_buffer* /*buffer*/) const { - // TODO: + /*auto reply = server->create_reply(this->task_id()); + + const int timeout = 7; // seconds + + std::optional motd_content; + std::optional featured_content; + std::optional featured2_content; + + auto get_motd = [&motd_content]() + { + motd_content = utils::http::get_data("https://h1.gg/data/motd.json", {}, {}, {}, timeout); + }; + auto get_featured = [&featured_content]() + { + featured_content = utils::http::get_data("https://h1.gg/data/featured.json", {}, {}, {}, timeout); + }; + auto get_featured2 = [&featured2_content]() + { + featured2_content = utils::http::get_data("https://h1.gg/data/featured2.json", {}, {}, {}, timeout); + }; + + std::thread get_motd_thread(get_motd); + std::thread get_featured_thread(get_featured); + std::thread get_featured2_thread(get_featured2); + + get_motd_thread.join(); + get_featured_thread.join(); + get_featured2_thread.join(); + + if (motd_content.has_value()) + { + const auto motd = new bdMarketingMessage; + motd->m_messageID = 1; + motd->m_languageCode = "en-US"; + motd->m_content = motd_content.value().buffer.data(); + motd->m_metadata = "{}"; + reply->add(motd); + } + + if (featured_content.has_value()) + { + const auto featured = new bdMarketingMessage; + featured->m_messageID = 2; + featured->m_languageCode = "en-US"; + featured->m_content = featured_content.value().buffer.data(); + featured->m_metadata = "{}"; + reply->add(featured); + } + + if (featured2_content.has_value()) + { + const auto featured2 = new bdMarketingMessage; + featured2->m_messageID = 3; + featured2->m_languageCode = "en-US"; + featured2->m_content = featured2_content.value().buffer.data(); + featured2->m_metadata = "{}"; + reply->add(featured2); + } + + reply->send();*/ auto reply = server->create_reply(this->task_id()); reply->send(); } @@ -22,4 +84,11 @@ namespace demonware auto reply = server->create_reply(this->task_id()); reply->send(); } + + void bdMarketingComms::getMessages_(service_server* server, byte_buffer* /*buffer*/) const + { + // TODO: + auto reply = server->create_reply(this->task_id()); + reply->send(); + } } diff --git a/src/client/game/demonware/services/bdMarketingComms.hpp b/src/client/game/demonware/services/bdMarketingComms.hpp index c973d7ae..8a4aa007 100644 --- a/src/client/game/demonware/services/bdMarketingComms.hpp +++ b/src/client/game/demonware/services/bdMarketingComms.hpp @@ -10,5 +10,6 @@ namespace demonware private: void getMessages(service_server* server, byte_buffer* buffer) const; void reportFullMessagesViewed(service_server* server, byte_buffer* buffer) const; + void getMessages_(service_server* server, byte_buffer* buffer) const; }; } diff --git a/src/client/game/demonware/services/bdStorage.cpp b/src/client/game/demonware/services/bdStorage.cpp index b5a80925..047f8a64 100644 --- a/src/client/game/demonware/services/bdStorage.cpp +++ b/src/client/game/demonware/services/bdStorage.cpp @@ -7,8 +7,16 @@ #include "game/game.hpp" +#include "../../../component/mods.hpp" + namespace demonware { + std::string storage_path; + void set_storage_path(const std::string& path) + { + storage_path = path; + } + bdStorage::bdStorage() : service(10, "bdStorage") { this->register_task(20, &bdStorage::list_publisher_files); @@ -162,7 +170,31 @@ namespace demonware std::string bdStorage::get_user_file_path(const std::string& name) { - return "players2/user/" + name; + const auto regular_path = "players2/user/"; + if (storage_path.empty()) + { + return regular_path + name; + } + + const auto custom_path = regular_path + storage_path + "/"; + if (!utils::io::directory_exists(custom_path)) + { + utils::io::create_directory(custom_path); + const auto copy_file = [&](const std::string& name) + { + const auto file_path = regular_path + name; + if (utils::io::file_exists(file_path)) + { + const auto data = utils::io::read_file(file_path); + utils::io::write_file(custom_path + name, data, false); + } + }; + + copy_file("commondata"); + copy_file("mpdata"); + } + + return custom_path + name; } void bdStorage::upload_and_validate_files(service_server* server, byte_buffer* buffer) const diff --git a/src/client/game/demonware/services/bdStorage.hpp b/src/client/game/demonware/services/bdStorage.hpp index e34036fb..58dcbd80 100644 --- a/src/client/game/demonware/services/bdStorage.hpp +++ b/src/client/game/demonware/services/bdStorage.hpp @@ -2,6 +2,8 @@ namespace demonware { + void set_storage_path(const std::string& path); + class bdStorage final : public service { public: diff --git a/src/client/game/dvars.cpp b/src/client/game/dvars.cpp index 739a087a..035d1588 100644 --- a/src/client/game/dvars.cpp +++ b/src/client/game/dvars.cpp @@ -10,6 +10,8 @@ namespace dvars { + std::unordered_map dvar_map; + game::dvar_t* aimassist_enabled = nullptr; game::dvar_t* con_inputBoxColor = nullptr; @@ -176,11695 +178,54 @@ namespace dvars } } - constexpr int generate_hash(const char* string) + std::int32_t generate_hash(const std::string& string) { - const char* v1; - char v2, v6; - int v4, v5, v7; - char* end_ptr; - - v1 = string; - v2 = *string; - - if (v2 == 48 && v1[1] == 120) - { - return strtoul(v1 + 2, &end_ptr, 16); - } - - v4 = v2; - - if ((v2 - 65) <= 0x19u) - { - v4 = v2 + 32; - } - - v5 = 0xB3CB2E29 * static_cast(v4 ^ 0x319712C3); - - if (v2) - { - do - { - v6 = *++v1; - v7 = v6; - if ((v6 - 65) <= 0x19u) - { - v7 = v6 + 32; - } - - v5 = 0xB3CB2E29 * static_cast(v5 ^ v7); - } - while (v6); - } - - return v5; + return generate_hash(string.data()); } - std::vector dvar_list = + void insert_dvar_info(const std::int32_t hash, const std::string& name, const std::string& description) { - { - "ai_count", - "Sets AI count.", - generate_hash("ai_count") - }, - { - "accessToSubscriberContent", - "Whether to display the subscriber maps.", - generate_hash("accessToSubscriberContent") - }, - { - "aci", - "", - generate_hash("aci") - }, - { - "actionSlotsHide", - "Hide the actionslots.", - generate_hash("actionSlotsHide") - }, - { - "ai_grenadeReturn_approachMinDot", - "Minimal dot product between the approach and throw vectors to perform a grenade return", - generate_hash("ai_grenadeReturn_approachMinDot") - }, - { - "ai_grenadeReturn_debug", - "Turns on debug info for AI grenade returns", - generate_hash("ai_grenadeReturn_debug") - }, - { - "ai_grenadeReturn_extraFuseTime", - "The amount of time (in ms) to add to a grenade fuse when trying to return grenade that's below minFuseTime", - generate_hash("ai_grenadeReturn_extraFuseTime") - }, - { - "ai_grenadeReturn_minDistSqr", - "Minimal distance to a grenade to consider it for a return so that transition anims will play", - generate_hash("ai_grenadeReturn_minDistSqr") - }, - { - "ai_grenadeReturn_minFuseTime", - "If the fuse time drops below this value when an ally is attempting to return a grenade, add extra fuse time", - generate_hash("ai_grenadeReturn_minFuseTime") - }, - { - "ai_grenadeReturn_stationary", - "If set, AI will attempt to return grenades that they are within pickup distance - regardless of min dist", - generate_hash("ai_grenadeReturn_stationary") - }, - { - "ai_grenadeReturn_traceToGrenade", - "If set, AI will only attempt to return grenades when they have a clear sight trace to the grenade", - generate_hash("ai_grenadeReturn_traceToGrenade") - }, - { - "ai_threatUpdateInterval", - "AI target threat update interval in milliseconds", - generate_hash("ai_threatUpdateInterval") - }, - { - "aim_autoaim_enabled", - "", - generate_hash("aim_autoaim_enabled") - }, - { - "aim_target_sentient_radius", - "The radius used to calculate target bounds for a sentient(actor or player)", - generate_hash("aim_target_sentient_radius") - }, - { - "ammoCounterHide", - "Hide the Ammo Counter", - generate_hash("ammoCounterHide") - }, - { - "armory_contentpacks_enabled", - "Allowed armory content packs. 0: none , 1: first armory content pack enabled, 2: first and second armory content pack enabled", - generate_hash("armory_contentpacks_enabled") - }, - { - "badHost_detectMinServerTime", - "Time in MS before the bad host dection system kicks in after match start", - generate_hash("badHost_detectMinServerTime") - }, - { - "badhost_maxDoISuckFrames", - "Max lagged frames need to end match", - generate_hash("badhost_maxDoISuckFrames") - }, - { - "band_12players", - "12 player bandwidth req'd", - generate_hash("band_12players") - }, - { - "band_18players", - "18 player bandwidth req'd", - generate_hash("band_18players") - }, - { - "band_2players", - "2 player bandwidth req'd", - generate_hash("band_2players") - }, - { - "band_4players", - "4 player bandwidth req'd", - generate_hash("band_4players") - }, - { - "band_8players", - "8 player bandwidth req'd", - generate_hash("band_8players") - }, - { - "bg_allowScuffFootsteps", - "If true, scuff sounds will be played when the player rotates in place.", - generate_hash("bg_allowScuffFootsteps") - }, - { - "bg_bulletExplDmgFactor", - "Weapon damage multiplier that will be applied at the center of the slash damage area.", - generate_hash("bg_bulletExplDmgFactor") - }, - { - "bg_bulletExplRadius", - "The radius of the bullet splash damage, where the damage gradually falls off to 0.", - generate_hash("bg_bulletExplRadius") - }, - { - "bg_compassShowEnemies", - "Whether enemies are visible on the compass at all times", - generate_hash("bg_compassShowEnemies") - }, - { - "bg_idleSwingSpeed", - "The rate at which the player's legs swing around when idle (multi-player only)", - generate_hash("bg_idleSwingSpeed") - }, - { - "bg_shieldHitEncodeHeightVM", - "The decoding range, in height, of a client's viewmodel shield.", - generate_hash("bg_shieldHitEncodeHeightVM") - }, - { - "bg_shieldHitEncodeHeightWorld", - "The encoding range, in height, of a client's world shield. A hit in this range is encoded into one of 8 rows.", - generate_hash("bg_shieldHitEncodeHeightWorld") - }, - { - "bg_shieldHitEncodeWidthVM", - "The decoding range, in width, of a client's viewmodel shield.", - generate_hash("bg_shieldHitEncodeWidthVM") - }, - { - "bg_shieldHitEncodeWidthWorld", - "The encoding range, in width, of a client's world shield. A hit in this range is encoded into one of 16 collumns.", - generate_hash("bg_shieldHitEncodeWidthWorld") - }, - { - "bg_shock_fadeOverride", - "Override the time for the shellshock kick effect to fade in MP", - generate_hash("bg_shock_fadeOverride") - }, - { - "bg_shock_lookControl", - "Alter player control during shellshock", - generate_hash("bg_shock_lookControl") - }, - { - "bg_shock_lookControl_fadeTime", - "The time for the shellshock player control to fade in seconds", - generate_hash("bg_shock_lookControl_fadeTime") - }, - { - "bg_shock_lookControl_maxpitchspeed", - "Maximum pitch movement rate while shellshocked in degrees per second", - generate_hash("bg_shock_lookControl_maxpitchspeed") - }, - { - "bg_shock_lookControl_maxyawspeed", - "Maximum yaw movement rate while shell shocked in degrees per second", - generate_hash("bg_shock_lookControl_maxyawspeed") - }, - { - "bg_shock_lookControl_mousesensitivityscale", - "Sensitivity scale to apply to a shellshocked player", - generate_hash("bg_shock_lookControl_mousesensitivityscale") - }, - { - "bg_shock_movement", - "Affect player's movement speed duringi shellshock", - generate_hash("bg_shock_movement") - }, - { - "bg_shock_screenBlurBlendFadeTime", - "The amount of time in seconds for the shellshock effect to fade", - generate_hash("bg_shock_screenBlurBlendFadeTime") - }, - { - "bg_shock_screenBlurBlendTime", - "The amount of time in seconds for the shellshock effect to fade", - generate_hash("bg_shock_screenBlurBlendTime") - }, - { - "bg_shock_screenFlashShotFadeTime", - "In seconds, how soon from the end of the effect to start blending out the whiteout layer.", - generate_hash("bg_shock_screenFlashShotFadeTime") - }, - { - "bg_shock_screenFlashWhiteFadeTime", - "In seconds, how soon from the end of the effect to start blending out the whiteout layer.", - generate_hash("bg_shock_screenFlashWhiteFadeTime") - }, - { - "bg_shock_screenType", - "Shell shock screen effect type", - generate_hash("bg_shock_screenType") - }, - { - "bg_shock_sound", - "Play shell shock sound", - generate_hash("bg_shock_sound") - }, - { - "bg_shock_soundDryLevel", - "Shell shock sound dry level", - generate_hash("bg_shock_soundDryLevel") - }, - { - "bg_shock_soundEnd", - "Shellshock end sound alias", - generate_hash("bg_shock_soundEnd") - }, - { - "bg_shock_soundEndAbort", - "Shellshock aborted end sound alias", - generate_hash("bg_shock_soundEndAbort") - }, - { - "bg_shock_soundFadeInTime", - "Shell shock sound fade in time in seconds", - generate_hash("bg_shock_soundFadeInTime") - }, - { - "bg_shock_soundFadeOutTime", - "Shell shock sound fade out time in seconds", - generate_hash("bg_shock_soundFadeOutTime") - }, - { - "bg_shock_soundLoop", - "Shellshock loop alias", - generate_hash("bg_shock_soundLoop") - }, - { - "bg_shock_soundLoopEndDelay", - "Sound loop end offset time from the end of the shellshock in seconds", - generate_hash("bg_shock_soundLoopEndDelay") - }, - { - "bg_shock_soundLoopFadeTime", - "Shell shock sound loop fade time in seconds", - generate_hash("bg_shock_soundLoopFadeTime") - }, - { - "bg_shock_soundLoopSilent", - "The sound that gets blended with the shellshock loop alias", - generate_hash("bg_shock_soundLoopSilent") - }, - { - "bg_shock_soundModEndDelay", - "The delay from the end of the shell shock to the end of the sound modification", - generate_hash("bg_shock_soundModEndDelay") - }, - { - "bg_shock_soundRoomType", - "Shell shock sound reverb room type", - generate_hash("bg_shock_soundRoomType") - }, - { - "bg_shock_soundSubmix", - "Shell shock submix to apply", - generate_hash("bg_shock_soundSubmix") - }, - { - "bg_shock_soundWetLevel", - "Shell shock sound wet level", - generate_hash("bg_shock_soundWetLevel") - }, - { - "bg_shock_viewKickFadeTime", - "The time for the shellshock kick effect to fade", - generate_hash("bg_shock_viewKickFadeTime") - }, - { - "bg_shock_viewKickPeriod", - "The period of the shellshock view kick effect", - generate_hash("bg_shock_viewKickPeriod") - }, - { - "bg_shock_viewKickRadius", - "Shell shock kick radius", - generate_hash("bg_shock_viewKickRadius") - }, - { - "bg_swingSpeed", - "The rate at which the player's legs swing around when idle (multi-player only)", - generate_hash("bg_swingSpeed") - }, - { - "bg_torsoSwingSpeed", - "The rate at which the player's torso swings around when strafing (multi-player only)", - generate_hash("bg_torsoSwingSpeed") - }, - { - "boostcheatHeadshotsTotalCoef", - "", - generate_hash("boostcheatHeadshotsTotalCoef") - }, - { - "boostcheatHeadshotsTotalMean", - "", - generate_hash("boostcheatHeadshotsTotalMean") - }, - { - "boostcheatHeadshotsTotalStddev", - "", - generate_hash("boostcheatHeadshotsTotalStddev") - }, - { - "boostcheatIntercept", - "", - generate_hash("boostcheatIntercept") - }, - { - "boostcheatKillerXAnomalyCoef", - "", - generate_hash("boostcheatKillerXAnomalyCoef") - }, - { - "boostcheatKillerXAnomalyMean", - "", - generate_hash("boostcheatKillerXAnomalyMean") - }, - { - "boostcheatKillerXAnomalyStddev", - "", - generate_hash("boostcheatKillerXAnomalyStddev") - }, - { - "boostcheatKillerYAnomalyCoef", - "", - generate_hash("boostcheatKillerYAnomalyCoef") - }, - { - "boostcheatKillerYAnomalyMean", - "", - generate_hash("boostcheatKillerYAnomalyMean") - }, - { - "boostcheatKillerYAnomalyStddev", - "", - generate_hash("boostcheatKillerYAnomalyStddev") - }, - { - "boostcheatMeanDistanceMostKilledPlayerTraveledMean", - "", - generate_hash("boostcheatMeanDistanceMostKilledPlayerTraveledMean") - }, - { - "boostcheatMeanDistanceVictimTraveledCoef", - "", - generate_hash("boostcheatMeanDistanceVictimTraveledCoef") - }, - { - "boostcheatMeanDistanceVictimTraveledMean", - "", - generate_hash("boostcheatMeanDistanceVictimTraveledMean") - }, - { - "boostcheatMeanDistanceVictimTraveledStddev", - "", - generate_hash("boostcheatMeanDistanceVictimTraveledStddev") - }, - { - "boostcheatMeanMostKilledPlayerLifetimeMillisecondsMean", - "", - generate_hash("boostcheatMeanMostKilledPlayerLifetimeMillisecondsMean") - }, - { - "boostcheatMostKilledPlayerHKRatioCoef", - "", - generate_hash("boostcheatMostKilledPlayerHKRatioCoef") - }, - { - "boostcheatMostKilledPlayerHKRatioMean", - "", - generate_hash("boostcheatMostKilledPlayerHKRatioMean") - }, - { - "boostcheatMostKilledPlayerHKRatioStddev", - "", - generate_hash("boostcheatMostKilledPlayerHKRatioStddev") - }, - { - "boostcheatMostKilledPlayerKillsRatioCoef", - "", - generate_hash("boostcheatMostKilledPlayerKillsRatioCoef") - }, - { - "boostcheatMostKilledPlayerKillsRatioMean", - "", - generate_hash("boostcheatMostKilledPlayerKillsRatioMean") - }, - { - "boostcheatMostKilledPlayerKillsRatioStddev", - "", - generate_hash("boostcheatMostKilledPlayerKillsRatioStddev") - }, - { - "boostcheatMostKilledPlayerKillsTotalCoef", - "", - generate_hash("boostcheatMostKilledPlayerKillsTotalCoef") - }, - { - "boostcheatMostKilledPlayerKillsTotalMean", - "", - generate_hash("boostcheatMostKilledPlayerKillsTotalMean") - }, - { - "boostcheatMostKilledPlayerKillsTotalStddev", - "", - generate_hash("boostcheatMostKilledPlayerKillsTotalStddev") - }, - { - "boostcheatMostKilledPlayerKillTimestampsAnomalyMean", - "", - generate_hash("boostcheatMostKilledPlayerKillTimestampsAnomalyMean") - }, - { - "boostcheatVictimXAnomalyCoef", - "", - generate_hash("boostcheatVictimXAnomalyCoef") - }, - { - "boostcheatVictimXAnomalyMean", - "", - generate_hash("boostcheatVictimXAnomalyMean") - }, - { - "boostcheatVictimXAnomalyStddev", - "", - generate_hash("boostcheatVictimXAnomalyStddev") - }, - { - "boostcheatVictimYAnomalyCoef", - "", - generate_hash("boostcheatVictimYAnomalyCoef") - }, - { - "boostcheatVictimYAnomalyMean", - "", - generate_hash("boostcheatVictimYAnomalyMean") - }, - { - "boostcheatVictimYAnomalyStddev", - "", - generate_hash("boostcheatVictimYAnomalyStddev") - }, - { - "bot_DifficultyDefault", - "default difficulty level of bots", - generate_hash("bot_DifficultyDefault") - }, - { - "ca_auto_signin", - "CoD Anywhere start sign-in task automatically on startup or first party sign-in", - generate_hash("ca_auto_signin") - }, - { - "ca_do_mlc", - "CoD Anywhere Do Multi Login check", - generate_hash("ca_do_mlc") - }, - { - "ca_intra_only", - "CoD Anywhere Intra Network Only", - generate_hash("ca_intra_only") - }, - { - "ca_require_signin", - "CoD Anywhere require sign in to enter MP", - generate_hash("ca_require_signin") - }, - { - "ca_show_signup_request", - "CoD Anywhere should you show new users a popup requesting they create a CoD Account?", - generate_hash("ca_show_signup_request") - }, - { - "camera_thirdPerson", - "Use third person view globally", - generate_hash("camera_thirdPerson") - }, - { - "cameraShakeRemoteHelo_Angles", - "Remote helicopter gunner cam, range to shake the view.", - generate_hash("cameraShakeRemoteHelo_Angles") - }, - { - "cameraShakeRemoteHelo_Freqs", - "Remote helicopter gunner cam, how fast to shake.", - generate_hash("cameraShakeRemoteHelo_Freqs") - }, - { - "cameraShakeRemoteHelo_SpeedRange", - "Remote helicopter gunner cam, range of missile speed to scale the shaking.", - generate_hash("cameraShakeRemoteHelo_SpeedRange") - }, - { - "cameraShakeRemoteMissile_Angles", - "Remote missile-cam, range to shake the view.", - generate_hash("cameraShakeRemoteMissile_Angles") - }, - { - "cameraShakeRemoteMissile_Freqs", - "Remote missile-cam, how fast to shake.", - generate_hash("cameraShakeRemoteMissile_Freqs") - }, - { - "cameraShakeRemoteMissile_SpeedRange", - "Remote missile-cam, range of missile speed to scale the shaking.", - generate_hash("cameraShakeRemoteMissile_SpeedRange") - }, - { - "cg_airstrikeCamFstop", - "Airstrike kill camera aperture. Lower f-stop yields a shallower depth of field. Typical values range from 1 to 22", - generate_hash("cg_airstrikeCamFstop") - }, - { - "cg_airstrikeKillCamFarBlur", - "", - generate_hash("cg_airstrikeKillCamFarBlur") - }, - { - "cg_airstrikeKillCamFarBlurDist", - "", - generate_hash("cg_airstrikeKillCamFarBlurDist") - }, - { - "cg_airstrikeKillCamFarBlurStart", - "", - generate_hash("cg_airstrikeKillCamFarBlurStart") - }, - { - "cg_airstrikeKillCamFov", - "Airstrike kill camera field of view.", - generate_hash("cg_airstrikeKillCamFov") - }, - { - "cg_airstrikeKillCamNearBlur", - "", - generate_hash("cg_airstrikeKillCamNearBlur") - }, - { - "cg_airstrikeKillCamNearBlurEnd", - "", - generate_hash("cg_airstrikeKillCamNearBlurEnd") - }, - { - "cg_airstrikeKillCamNearBlurStart", - "", - generate_hash("cg_airstrikeKillCamNearBlurStart") - }, - { - "cg_blood", - "Show Blood", - generate_hash("cg_blood") - }, - { - "cg_bloodThickColor", - "Color of the blood overlay's thick blood splatter", - generate_hash("cg_bloodThickColor") - }, - { - "cg_bloodThinColor", - "Color of the blood overlay's thin blood splatter", - generate_hash("cg_bloodThinColor") - }, - { - "cg_brass", - "Weapons eject brass", - generate_hash("cg_brass") - }, - { - "cg_centertime", - "The time for a center printed message to fade", - generate_hash("cg_centertime") - }, - { - "cg_chatHeight", - "The font height of a chat message", - generate_hash("cg_chatHeight") - }, - { - "cg_chatTime", - "The amount of time that a chat message is visible", - generate_hash("cg_chatTime") - }, - { - "cg_ColorBlind_EnemyTeam", - "Enemy team color for color blind people", - generate_hash("cg_ColorBlind_EnemyTeam") - }, - { - "cg_ColorBlind_MyParty", - "Player party color for color blind people", - generate_hash("cg_ColorBlind_MyParty") - }, - { - "cg_ColorBlind_MyTeam", - "Player team color for color blind people", - generate_hash("cg_ColorBlind_MyTeam") - }, - { - "cg_connectionIconSize", - "Size of the connection icon", - generate_hash("cg_connectionIconSize") - }, - { - "cg_constantSizeHeadIcons", - "Head icons are the same size regardless of distance from the player", - generate_hash("cg_constantSizeHeadIcons") - }, - { - "cg_crosshairAlpha", - "The alpha value of the crosshair", - generate_hash("cg_crosshairAlpha") - }, - { - "cg_crosshairAlphaMin", - "The minimum alpha value of the crosshair when it fades in", - generate_hash("cg_crosshairAlphaMin") - }, - { - "cg_crosshairDynamic", - "Crosshair is Dynamic", - generate_hash("cg_crosshairDynamic") - }, - { - "cg_crosshairEnemyColor", - "The crosshair color when over an enemy", - generate_hash("cg_crosshairEnemyColor") - }, - { - "cg_crosshairVerticalOffset", - "Amount to vertically offset the crosshair from the center.", - generate_hash("cg_crosshairVerticalOffset") - }, - { - "cg_cullBulletAngle", - "Cull bullet trajectories that don't fall within this fov", - generate_hash("cg_cullBulletAngle") - }, - { - "cg_cullBullets", - "Whether to cull bullet fire prediction if trajectory doesn't pass your view or anywhere near you", - generate_hash("cg_cullBullets") - }, - { - "cg_cursorHints", - "Draw cursor hints where:\n 0: no hints", - generate_hash("cg_cursorHints") - }, - { - "cg_deadChatWithDead", - "If true, dead players can all chat together, regardless of team", - generate_hash("cg_deadChatWithDead") - }, - { - "cg_deadChatWithTeam", - "If true, dead players can talk to living players on their team", - generate_hash("cg_deadChatWithTeam") - }, - { - "cg_deadHearAllLiving", - "If true, dead players can hear all living players talk", - generate_hash("cg_deadHearAllLiving") - }, - { - "cg_deadHearTeamLiving", - "If true, dead players can hear living players on their team talk", - generate_hash("cg_deadHearTeamLiving") - }, - { - "cg_descriptiveText", - "Draw descriptive spectator messages", - generate_hash("cg_descriptiveText") - }, - { - "cg_draw2D", - "Draw 2D screen elements", - generate_hash("cg_draw2D") - }, - { - "cg_drawBreathHint", - "Draw a 'hold breath to steady' hint", - generate_hash("cg_drawBreathHint") - }, - { - "cg_drawBuildName", - "Draw build name", - generate_hash("cg_drawBuildName") - }, - { - "cg_drawCrosshair", - "Turn on weapon crosshair", - generate_hash("cg_drawCrosshair") - }, - { - "cg_drawCrosshairNames", - "Draw the name of an enemy under the crosshair", - generate_hash("cg_drawCrosshairNames") - }, - { - "cg_drawCrosshairNamesPosX", - "", - generate_hash("cg_drawCrosshairNamesPosX") - }, - { - "cg_drawCrosshairNamesPosY", - "", - generate_hash("cg_drawCrosshairNamesPosY") - }, - { - "cg_drawDamageDirection", - "Draw hit direction arrow.", - generate_hash("cg_drawDamageDirection") - }, - { - "cg_drawDamageFlash", - "Draw flash when hit.", - generate_hash("cg_drawDamageFlash") - }, - { - "cg_drawDoubleTapDetonateHint", - "Draw a 'double tap to detonate grenade' hint", - generate_hash("cg_drawDoubleTapDetonateHint") - }, - { - "cg_drawEffectNum", - "Draw counts of effects and elements", - generate_hash("cg_drawEffectNum") - }, - { - "cg_drawFPS", - "Draw frames per second", - generate_hash("cg_drawFPS") - }, - { - "cg_drawFPSLabels", - "Draw FPS Info Labels", - generate_hash("cg_drawFPSLabels") - }, - { - "cg_drawFriendlyHUDGrenades", - "Draw grenade warning indicators for friendly grenades (should be true if friendly-fire is enabled)", - generate_hash("cg_drawFriendlyHUDGrenades") - }, - { - "cg_drawFriendlyNames", - "Whether to show friendly names in game", - generate_hash("cg_drawFriendlyNames") - }, - { - "cg_drawFriendlyNamesAlways", - "Whether to always show friendly names in game (for certain gametypes)", - generate_hash("cg_drawFriendlyNamesAlways") - }, - { - "cg_drawGun", - "Draw the view model", - generate_hash("cg_drawGun") - }, - { - "cg_drawHealth", - "Draw health bar", - generate_hash("cg_drawHealth") - }, - { - "cg_drawMantleHint", - "Draw a 'press key to mantle' hint", - generate_hash("cg_drawMantleHint") - }, - { - "cg_drawMaterial", - "Draw debugging information for materials", - generate_hash("cg_drawMaterial") - }, - { - "cg_drawpaused", - "Draw paused screen", - generate_hash("cg_drawpaused") - }, - { - "cg_drawScriptUsage", - "Draw debugging information for scripts", - generate_hash("cg_drawScriptUsage") - }, - { - "cg_drawSnapshot", - "Draw debugging information for snapshots", - generate_hash("cg_drawSnapshot") - }, - { - "cg_drawStatsSource", - "Draw stats source", - generate_hash("cg_drawStatsSource") - }, - { - "cg_drawTalk", - "Controls which icons CG_TALKER ownerdraw draws", - generate_hash("cg_drawTalk") - }, - { - "cg_drawTurretCrosshair", - "Draw a cross hair when using a turret", - generate_hash("cg_drawTurretCrosshair") - }, - { - "cg_drawViewpos", - "Draw viewpos", - generate_hash("cg_drawViewpos") - }, - { - "cg_e3TrailerHacks", - "Tweaks for trailer recording", - generate_hash("cg_e3TrailerHacks") - }, - { - "cg_equipmentSounds", - "Play equipment sounds", - generate_hash("cg_equipmentSounds") - }, - { - "cg_errordecay", - "Decay for predicted error", - generate_hash("cg_errordecay") - }, - { - "cg_everyoneHearsEveryone", - "If true, all players can all chat together, regardless of team or death", - generate_hash("cg_everyoneHearsEveryone") - }, - { - "cg_explosiveKillCamBackDist", - "Explosive kill camera: distance of camera backwards from explosive.", - generate_hash("cg_explosiveKillCamBackDist") - }, - { - "cg_explosiveKillCamGroundBackDist", - "Explosive kill camera when stuck to ground: distance of camera backwards from explosive.", - generate_hash("cg_explosiveKillCamGroundBackDist") - }, - { - "cg_explosiveKillCamGroundUpDist", - "Explosive kill camera when stuck to ground: distance of camera backwards from explosive.", - generate_hash("cg_explosiveKillCamGroundUpDist") - }, - { - "cg_explosiveKillCamStopDecelDist", - "Rocket and Grenade Launcher kill camera: distance from player to begin coming to rest", - generate_hash("cg_explosiveKillCamStopDecelDist") - }, - { - "cg_explosiveKillCamStopDist", - "Rocket and Grenade Launcher kill camera: distance from player to begin coming to rest", - generate_hash("cg_explosiveKillCamStopDist") - }, - { - "cg_explosiveKillCamUpDist", - "Explosive kill camera: distance of camera backwards from explosive.", - generate_hash("cg_explosiveKillCamUpDist") - }, - { - "cg_explosiveKillCamWallOutDist", - "Explosive kill camera when stuck to wall: distance of camera out from wall.", - generate_hash("cg_explosiveKillCamWallOutDist") - }, - { - "cg_explosiveKillCamWallSideDist", - "Explosive kill camera when stuck to wall: distance of camera out from wall.", - generate_hash("cg_explosiveKillCamWallSideDist") - }, - { - "cg_flashbangNameFadeIn", - "Time in milliseconds to fade in friendly names", - generate_hash("cg_flashbangNameFadeIn") - }, - { - "cg_flashbangNameFadeOut", - "Time in milliseconds to fade out friendly names", - generate_hash("cg_flashbangNameFadeOut") - }, - { - "cg_foliagesnd_alias", - "The sound that plays when an actor or player enters a foliage clip brush.", - generate_hash("cg_foliagesnd_alias") - }, - { - "cg_footsteps", - "Play footstep sounds that are NOT sprint", - generate_hash("cg_footsteps") - }, - { - "cg_footstepsSprint", - "Play sprint footstep sounds", - generate_hash("cg_footstepsSprint") - }, - { - "cg_fov", - "The field of view angle in degrees", - generate_hash("cg_fov") - }, - { - "cg_fovMin", - "The minimum possible field of view", - generate_hash("cg_fovMin") - }, - { - "cg_fovScale", - "Scale applied to the field of view", - generate_hash("cg_fovScale") - }, - { - "cg_friendlyNameFadeIn", - "Time in milliseconds to fade in friendly names", - generate_hash("cg_friendlyNameFadeIn") - }, - { - "cg_friendlyNameFadeOut", - "Time in milliseconds to fade out friendly names", - generate_hash("cg_friendlyNameFadeOut") - }, - { - "cg_gameBoldMessageWidth", - "The maximum character width of the bold game messages", - generate_hash("cg_gameBoldMessageWidth") - }, - { - "cg_gameMessageWidth", - "The maximum character width of the game messages", - generate_hash("cg_gameMessageWidth") - }, - { - "cg_gun_x", - "Forward position of the viewmodel", - generate_hash("cg_gun_x") - }, - { - "cg_gun_y", - "Right position of the viewmodel", - generate_hash("cg_gun_y") - }, - { - "cg_gun_z", - "Up position of the viewmodel", - generate_hash("cg_gun_z") - }, - { - "cg_headIconMinScreenRadius", - "The minumum radius of a head icon on the screen", - generate_hash("cg_headIconMinScreenRadius") - }, - { - "cg_hearKillerTime", - "Duration (in milliseconds) to hear the person you just killed", - generate_hash("cg_hearKillerTime") - }, - { - "cg_hearVictimEnabled", - "If true, you can hear the person you just killed", - generate_hash("cg_hearVictimEnabled") - }, - { - "cg_hearVictimTime", - "Duration (in milliseconds) to hear the person you just killed", - generate_hash("cg_hearVictimTime") - }, - { - "cg_heliKillCamFarBlur", - "", - generate_hash("cg_heliKillCamFarBlur") - }, - { - "cg_heliKillCamFarBlurDist", - "", - generate_hash("cg_heliKillCamFarBlurDist") - }, - { - "cg_heliKillCamFarBlurStart", - "", - generate_hash("cg_heliKillCamFarBlurStart") - }, - { - "cg_heliKillCamFov", - "Helicopter kill camera field of view.", - generate_hash("cg_heliKillCamFov") - }, - { - "cg_heliKillCamFstop", - "Helicopter kill camera aperture. Lower f-stop yields a shallower depth of field. Typical values range from 1 to 22", - generate_hash("cg_heliKillCamFstop") - }, - { - "cg_heliKillCamNearBlur", - "", - generate_hash("cg_heliKillCamNearBlur") - }, - { - "cg_heliKillCamNearBlurEnd", - "", - generate_hash("cg_heliKillCamNearBlurEnd") - }, - { - "cg_heliKillCamNearBlurStart", - "", - generate_hash("cg_heliKillCamNearBlurStart") - }, - { - "cg_hintFadeTime", - "Time in milliseconds for the cursor hint to fade", - generate_hash("cg_hintFadeTime") - }, - { - "cg_hudChatIntermissionPosition", - "Position of the HUD chat box during intermission", - generate_hash("cg_hudChatIntermissionPosition") - }, - { - "cg_hudChatPosition", - "Position of the HUD chat box", - generate_hash("cg_hudChatPosition") - }, - { - "cg_hudDamageIconHeight", - "The height of the damage icon", - generate_hash("cg_hudDamageIconHeight") - }, - { - "cg_hudDamageIconInScope", - "Draw damage icons when aiming down the sight of a scoped weapon", - generate_hash("cg_hudDamageIconInScope") - }, - { - "cg_hudDamageIconOffset", - "The offset from the center of the damage icon", - generate_hash("cg_hudDamageIconOffset") - }, - { - "cg_hudDamageIconOverlayTime", - "The amount of time (in ms) for the overlay portion of the damage icon to stay on screen", - generate_hash("cg_hudDamageIconOverlayTime") - }, - { - "cg_hudDamageIconStartFadeTime", - "The amount of time (in ms) before the damage icon begins to fade", - generate_hash("cg_hudDamageIconStartFadeTime") - }, - { - "cg_hudDamageIconTime", - "The amount of time for the damage icon to stay on screen after damage is taken", - generate_hash("cg_hudDamageIconTime") - }, - { - "cg_hudDamageIconWidth", - "The width of the damage icon", - generate_hash("cg_hudDamageIconWidth") - }, - { - "cg_hudGrenadeIconEnabledFlash", - "Show the grenade indicator for flash grenades", - generate_hash("cg_hudGrenadeIconEnabledFlash") - }, - { - "cg_hudGrenadeIconHeight", - "The height of the grenade indicator icon", - generate_hash("cg_hudGrenadeIconHeight") - }, - { - "cg_hudGrenadeIconInScope", - "Show the grenade indicator when aiming down the sight of a scoped weapon", - generate_hash("cg_hudGrenadeIconInScope") - }, - { - "cg_hudGrenadeIconMaxRangeFlash", - "The minimum distance that a flashbang has to be from a player in order to be shown on the grenade indicator", - generate_hash("cg_hudGrenadeIconMaxRangeFlash") - }, - { - "cg_hudGrenadeIconMaxRangeFrag", - "The minimum distance that a grenade has to be from a player in order to be shown on the grenade indicator", - generate_hash("cg_hudGrenadeIconMaxRangeFrag") - }, - { - "cg_hudGrenadeIconOffset", - "The offset from the center of the screen for a grenade icon", - generate_hash("cg_hudGrenadeIconOffset") - }, - { - "cg_hudGrenadeIconWidth", - "The width of the grenade indicator icon", - generate_hash("cg_hudGrenadeIconWidth") - }, - { - "cg_hudGrenadePointerHeight", - "The height of the grenade indicator pointer", - generate_hash("cg_hudGrenadePointerHeight") - }, - { - "cg_hudGrenadePointerPivot", - "The pivot point of th grenade indicator pointer", - generate_hash("cg_hudGrenadePointerPivot") - }, - { - "cg_hudGrenadePointerPulseFreq", - "The number of times per second that the grenade indicator flashes in Hertz", - generate_hash("cg_hudGrenadePointerPulseFreq") - }, - { - "cg_hudGrenadePointerPulseMax", - "The maximum alpha of the grenade indicator pulse. Values higher than 1 will cause the indicator to remain at full brightness for longer", - generate_hash("cg_hudGrenadePointerPulseMax") - }, - { - "cg_hudGrenadePointerPulseMin", - "The minimum alpha of the grenade indicator pulse. Values lower than 0 will cause the indicator to remain at full transparency for longer", - generate_hash("cg_hudGrenadePointerPulseMin") - }, - { - "cg_hudGrenadePointerWidth", - "The width of the grenade indicator pointer", - generate_hash("cg_hudGrenadePointerWidth") - }, - { - "cg_hudLegacySplitscreenScale", - "Screen scale for hud elements in splitscreen", - generate_hash("cg_hudLegacySplitscreenScale") - }, - { - "cg_hudLighting_basic_additiveLumOffset", - "[basic] Offset applied to additive light color.", - generate_hash("cg_hudLighting_basic_additiveLumOffset") - }, - { - "cg_hudLighting_basic_additiveLumScale", - "[basic] Scale applied to additive light color.", - generate_hash("cg_hudLighting_basic_additiveLumScale") - }, - { - "cg_hudLighting_basic_additiveOffset", - "", - generate_hash("cg_hudLighting_basic_additiveOffset") - }, - { - "cg_hudLighting_basic_additiveScale", - "", - generate_hash("cg_hudLighting_basic_additiveScale") - }, - { - "cg_hudLighting_basic_ambientLumOffset", - "[basic] Offset applied to ambient light color.", - generate_hash("cg_hudLighting_basic_ambientLumOffset") - }, - { - "cg_hudLighting_basic_ambientLumScale", - "[basic] Scale applied to ambient light color.", - generate_hash("cg_hudLighting_basic_ambientLumScale") - }, - { - "cg_hudLighting_basic_ambientOffset", - "", - generate_hash("cg_hudLighting_basic_ambientOffset") - }, - { - "cg_hudLighting_basic_ambientScale", - "", - generate_hash("cg_hudLighting_basic_ambientScale") - }, - { - "cg_hudLighting_basic_diffuseLumOffset", - "[basic] Offset applied to diffuse light color.", - generate_hash("cg_hudLighting_basic_diffuseLumOffset") - }, - { - "cg_hudLighting_basic_diffuseLumScale", - "[basic] Scale applied to diffuse light color.", - generate_hash("cg_hudLighting_basic_diffuseLumScale") - }, - { - "cg_hudLighting_basic_diffuseOffset", - "", - generate_hash("cg_hudLighting_basic_diffuseOffset") - }, - { - "cg_hudLighting_basic_diffuseScale", - "", - generate_hash("cg_hudLighting_basic_diffuseScale") - }, - { - "cg_hudLighting_basic_specExponent", - "[basic] Specular exponent. Higher values result in sharper highlights.", - generate_hash("cg_hudLighting_basic_specExponent") - }, - { - "cg_hudLighting_basic_specLumOffset", - "[basic] Offset applied to spec light luminance.", - generate_hash("cg_hudLighting_basic_specLumOffset") - }, - { - "cg_hudLighting_basic_specLumScale", - "[basic] Scale applied to spec light luminance.", - generate_hash("cg_hudLighting_basic_specLumScale") - }, - { - "cg_hudLighting_basic_specOffset", - "", - generate_hash("cg_hudLighting_basic_specOffset") - }, - { - "cg_hudLighting_basic_specScale", - "", - generate_hash("cg_hudLighting_basic_specScale") - }, - { - "cg_hudLighting_blood_additiveLumOffset", - "[blood] Offset applied to additive light color.", - generate_hash("cg_hudLighting_blood_additiveLumOffset") - }, - { - "cg_hudLighting_blood_additiveLumScale", - "[blood] Scale applied to additive light color.", - generate_hash("cg_hudLighting_blood_additiveLumScale") - }, - { - "cg_hudLighting_blood_additiveOffset", - "", - generate_hash("cg_hudLighting_blood_additiveOffset") - }, - { - "cg_hudLighting_blood_additiveScale", - "", - generate_hash("cg_hudLighting_blood_additiveScale") - }, - { - "cg_hudLighting_blood_ambientLumOffset", - "[blood] Offset applied to ambient light color.", - generate_hash("cg_hudLighting_blood_ambientLumOffset") - }, - { - "cg_hudLighting_blood_ambientLumScale", - "[blood] Scale applied to ambient light color.", - generate_hash("cg_hudLighting_blood_ambientLumScale") - }, - { - "cg_hudLighting_blood_ambientOffset", - "", - generate_hash("cg_hudLighting_blood_ambientOffset") - }, - { - "cg_hudLighting_blood_ambientScale", - "", - generate_hash("cg_hudLighting_blood_ambientScale") - }, - { - "cg_hudLighting_blood_diffuseLumOffset", - "[blood] Offset applied to diffuse light color.", - generate_hash("cg_hudLighting_blood_diffuseLumOffset") - }, - { - "cg_hudLighting_blood_diffuseLumScale", - "[blood] Scale applied to diffuse light color.", - generate_hash("cg_hudLighting_blood_diffuseLumScale") - }, - { - "cg_hudLighting_blood_diffuseOffset", - "", - generate_hash("cg_hudLighting_blood_diffuseOffset") - }, - { - "cg_hudLighting_blood_diffuseScale", - "", - generate_hash("cg_hudLighting_blood_diffuseScale") - }, - { - "cg_hudLighting_blood_specExponent", - "[blood] Specular exponent. Higher values result in sharper highlights.", - generate_hash("cg_hudLighting_blood_specExponent") - }, - { - "cg_hudLighting_blood_specLumOffset", - "[blood] Offset applied to spec light luminance.", - generate_hash("cg_hudLighting_blood_specLumOffset") - }, - { - "cg_hudLighting_blood_specLumScale", - "[blood] Scale applied to spec light luminance.", - generate_hash("cg_hudLighting_blood_specLumScale") - }, - { - "cg_hudLighting_blood_specOffset", - "", - generate_hash("cg_hudLighting_blood_specOffset") - }, - { - "cg_hudLighting_blood_specScale", - "", - generate_hash("cg_hudLighting_blood_specScale") - }, - { - "cg_hudLighting_fadeSharpness", - "This controls how sharp the lines are when fading using the mask alpha. Higher values are sharper.", - generate_hash("cg_hudLighting_fadeSharpness") - }, - { - "cg_hudMapBorderWidth", - "The size of the full map's border, filled by the CG_PLAYER_FULLMAP_BORDER ownerdraw", - generate_hash("cg_hudMapBorderWidth") - }, - { - "cg_hudMapFriendlyHeight", - "", - generate_hash("cg_hudMapFriendlyHeight") - }, - { - "cg_hudMapFriendlyWidth", - "", - generate_hash("cg_hudMapFriendlyWidth") - }, - { - "cg_hudMapPlayerHeight", - "", - generate_hash("cg_hudMapPlayerHeight") - }, - { - "cg_hudMapPlayerWidth", - "", - generate_hash("cg_hudMapPlayerWidth") - }, - { - "cg_hudMapRadarLineThickness", - "Thickness, relative to the map width, of the radar texture that sweeps across the full screen map", - generate_hash("cg_hudMapRadarLineThickness") - }, - { - "cg_hudObjectiveTextScale", - "", - generate_hash("cg_hudObjectiveTextScale") - }, - { - "cg_hudProneY", - "Virtual screen y coordinate of the prone blocked message", - generate_hash("cg_hudProneY") - }, - { - "cg_hudSayPosition", - "Position of the HUD say box", - generate_hash("cg_hudSayPosition") - }, - { - "cg_hudSplitscreenCompassElementScale", - "Scale value to apply to compass elements in splitscreen", - generate_hash("cg_hudSplitscreenCompassElementScale") - }, - { - "cg_hudSplitscreenCompassScale", - "Scale value to apply to the compass in splitscreen", - generate_hash("cg_hudSplitscreenCompassScale") - }, - { - "cg_hudSplitscreenStanceScale", - "Scale value to apply to the stance HUD element in splitscreen", - generate_hash("cg_hudSplitscreenStanceScale") - }, - { - "cg_hudStanceFlash", - "The background color of the flash when the stance changes", - generate_hash("cg_hudStanceFlash") - }, - { - "cg_hudVotePosition", - "Position of the HUD vote box", - generate_hash("cg_hudVotePosition") - }, - { - "cg_invalidCmdHintBlinkInterval", - "Blink rate of an invalid command hint", - generate_hash("cg_invalidCmdHintBlinkInterval") - }, - { - "cg_invalidCmdHintDuration", - "Duration of an invalid command hint", - generate_hash("cg_invalidCmdHintDuration") - }, - { - "cg_javelinKillCamCloseZDist", - "Javelin kill camera: closest distance above the target.", - generate_hash("cg_javelinKillCamCloseZDist") - }, - { - "cg_javelinKillCamDownDist", - "Javelin kill camera: distance to follow during ascent.", - generate_hash("cg_javelinKillCamDownDist") - }, - { - "cg_javelinKillCamFov", - "Javelin kill camera: fov", - generate_hash("cg_javelinKillCamFov") - }, - { - "cg_javelinKillCamLookLerpDist", - "Javelin kill camera: distance over which to lerp to look at player during descent. A value of zero means don't lerp at all.", - generate_hash("cg_javelinKillCamLookLerpDist") - }, - { - "cg_javelinKillCamPassDist", - "Javelin kill camera: distance away when passing.", - generate_hash("cg_javelinKillCamPassDist") - }, - { - "cg_javelinKillCamPassTime", - "Javelin kill camera: time in seconds to pass javelin on the way up", - generate_hash("cg_javelinKillCamPassTime") - }, - { - "cg_javelinKillCamUpDist", - "Javelin kill camera: distance to follow during ascent.", - generate_hash("cg_javelinKillCamUpDist") - }, - { - "cg_killCamDefaultLerpTime", - "Default time used to lerp between killcam entities.", - generate_hash("cg_killCamDefaultLerpTime") - }, - { - "cg_killCamTurretLerpTime", - "Time used to lerp to a killcam entity of the TURRET type.", - generate_hash("cg_killCamTurretLerpTime") - }, - { - "cg_landingSounds", - "Play landing on surface sounds", - generate_hash("cg_landingSounds") - }, - { - "cg_largeExplosiveKillCamBackDist", - "Large Explosive kill camera: distance of camera backwards from explosive.", - generate_hash("cg_largeExplosiveKillCamBackDist") - }, - { - "cg_largeExplosiveKillCamUpDist", - "Large Explosive kill camera: distance of camera backwards from explosive.", - generate_hash("cg_largeExplosiveKillCamUpDist") - }, - { - "cg_mapLocationSelectionCursorSpeed", - "Speed of the cursor when selecting a location on the map", - generate_hash("cg_mapLocationSelectionCursorSpeed") - }, - { - "cg_marks_ents_player_only", - "Marks on entities from players' bullets only.", - generate_hash("cg_marks_ents_player_only") - }, - { - "cg_minCullBulletDist", - "Don't cull bullet trajectories that are within this distance to you.", - generate_hash("cg_minCullBulletDist") - }, - { - "cg_objectiveText", - "", - generate_hash("cg_objectiveText") - }, - { - "cg_overheadIconSize", - "The maximum size to show overhead icons like 'rank'", - generate_hash("cg_overheadIconSize") - }, - { - "cg_overheadNamesFarDist", - "The far distance at which name sizes are scaled by cg_overheadNamesFarScale", - generate_hash("cg_overheadNamesFarDist") - }, - { - "cg_overheadNamesFarScale", - "The amount to scale overhead name sizes at cg_overheadNamesFarDist", - generate_hash("cg_overheadNamesFarScale") - }, - { - "cg_overheadNamesFont", - "Font for overhead names ( see menudefinition.h )", - generate_hash("cg_overheadNamesFont") - }, - { - "cg_overheadNamesGlow", - "Glow color for overhead names", - generate_hash("cg_overheadNamesGlow") - }, - { - "cg_overheadNamesMaxDist", - "The maximum distance for showing friendly player names", - generate_hash("cg_overheadNamesMaxDist") - }, - { - "cg_overheadNamesNearDist", - "The near distance at which names are full size", - generate_hash("cg_overheadNamesNearDist") - }, - { - "cg_overheadNamesSize", - "The maximum size to show overhead names", - generate_hash("cg_overheadNamesSize") - }, - { - "cg_overheadRankSize", - "The size to show rank text", - generate_hash("cg_overheadRankSize") - }, - { - "cg_remoteMissileKillCamBackDist", - "Remote missile kill camera: distance of camera backwards from rocket.", - generate_hash("cg_remoteMissileKillCamBackDist") - }, - { - "cg_remoteMissileKillCamUpDist", - "Remote missile kill camera: distance of camera backwards from rocket.", - generate_hash("cg_remoteMissileKillCamUpDist") - }, - { - "cg_rocketKillCamBackDist", - "Rocket kill camera: distance of camera backwards from rocket.", - generate_hash("cg_rocketKillCamBackDist") - }, - { - "cg_rocketKillCamUpDist", - "Rocket kill camera: distance of camera backwards from rocket.", - generate_hash("cg_rocketKillCamUpDist") - }, - { - "cg_scriptIconSize", - "Size of Icons defined by script", - generate_hash("cg_scriptIconSize") - }, - { - "cg_showmiss", - "Show prediction errors", - generate_hash("cg_showmiss") - }, - { - "cg_sprintMeterDisabledColor", - "The color of the sprint meter when the sprint meter is full", - generate_hash("cg_sprintMeterDisabledColor") - }, - { - "cg_sprintMeterEmptyColor", - "The color of the sprint meter when the sprint meter is full", - generate_hash("cg_sprintMeterEmptyColor") - }, - { - "cg_sprintMeterFullColor", - "The color of the sprint meter when the sprint meter is full", - generate_hash("cg_sprintMeterFullColor") - }, - { - "cg_subtitleMinTime", - "The minimum time that the subtitles are displayed on screen in seconds", - generate_hash("cg_subtitleMinTime") - }, - { - "cg_subtitleWidthStandard", - "The width of the subtitles on a non wide-screen", - generate_hash("cg_subtitleWidthStandard") - }, - { - "cg_subtitleWidthWidescreen", - "The width of the subtitle on a wide-screen", - generate_hash("cg_subtitleWidthWidescreen") - }, - { - "cg_teamChatsOnly", - "Allow chatting only on the same team", - generate_hash("cg_teamChatsOnly") - }, - { - "cg_TeamColor_Allies", - "Allies team color", - generate_hash("cg_TeamColor_Allies") - }, - { - "cg_TeamColor_Axis", - "Axis team color", - generate_hash("cg_TeamColor_Axis") - }, - { - "cg_TeamColor_EnemyTeam", - "Enemy team color", - generate_hash("cg_TeamColor_EnemyTeam") - }, - { - "cg_TeamColor_Free", - "Free Team color", - generate_hash("cg_TeamColor_Free") - }, - { - "cg_TeamColor_MyParty", - "Player team color when in the same party", - generate_hash("cg_TeamColor_MyParty") - }, - { - "cg_TeamColor_MyTeam", - "Player team color", - generate_hash("cg_TeamColor_MyTeam") - }, - { - "cg_TeamColor_Spectator", - "Spectator team color", - generate_hash("cg_TeamColor_Spectator") - }, - { - "cg_turretKillCamBackDist", - "Turret kill camera: distance of camera backwards from Turret.", - generate_hash("cg_turretKillCamBackDist") - }, - { - "cg_turretKillCamFov", - "Turret kill camera field of view.", - generate_hash("cg_turretKillCamFov") - }, - { - "cg_turretKillCamUpDist", - "Turret kill camera: distance of camera backwards from Turret.", - generate_hash("cg_turretKillCamUpDist") - }, - { - "cg_turretRemoteKillCamBackDist", - "Remote Turret kill camera: distance of camera backwards from Turret.", - generate_hash("cg_turretRemoteKillCamBackDist") - }, - { - "cg_turretRemoteKillCamFov", - "Remote Turret kill camera field of view.", - generate_hash("cg_turretRemoteKillCamFov") - }, - { - "cg_turretRemoteKillCamUpDist", - "Remote Turret kill camera: distance of camera backwards from Turret.", - generate_hash("cg_turretRemoteKillCamUpDist") - }, - { - "cg_vectorFieldsForceUniform", - "Forces all vector field assets to represent a single, uniform direction", - generate_hash("cg_vectorFieldsForceUniform") - }, - { - "cg_viewVehicleInfluence", - "The influence on the view angles from being in a vehicle", - generate_hash("cg_viewVehicleInfluence") - }, - { - "cg_viewZSmoothingMax", - "Threshhold for the maximum smoothing distance we'll do", - generate_hash("cg_viewZSmoothingMax") - }, - { - "cg_viewZSmoothingMin", - "Threshhold for the minimum smoothing distance it must move to smooth", - generate_hash("cg_viewZSmoothingMin") - }, - { - "cg_viewZSmoothingTime", - "Amount of time to spread the smoothing over", - generate_hash("cg_viewZSmoothingTime") - }, - { - "cg_voiceIconSize", - "Size of the 'voice' icon", - generate_hash("cg_voiceIconSize") - }, - { - "cg_waterSheeting_distortionScaleFactor", - "Distortion uv scales (Default to 1)", - generate_hash("cg_waterSheeting_distortionScaleFactor") - }, - { - "cg_waterSheeting_magnitude", - "Distortion magnitude", - generate_hash("cg_waterSheeting_magnitude") - }, - { - "cg_waterSheeting_radius", - "Tweak dev var; Glow radius in pixels at 640x480", - generate_hash("cg_waterSheeting_radius") - }, - { - "cg_weapHitCullAngle", - "Angle of cone within which to cull back facing weapon hit effects", - generate_hash("cg_weapHitCullAngle") - }, - { - "cg_weapHitCullEnable", - "When true, cull back facing weapon hit fx.", - generate_hash("cg_weapHitCullEnable") - }, - { - "cg_weaponCycleDelay", - "The delay after cycling to a new weapon to prevent holding down the cycle weapon button from cycling too fast", - generate_hash("cg_weaponCycleDelay") - }, - { - "cg_weaponHintsCoD1Style", - "Draw weapon hints in CoD1 style: with the weapon name, and with the icon below", - generate_hash("cg_weaponHintsCoD1Style") - }, - { - "cg_weaponVisInterval", - "Do weapon vis checks once per this many frames, per centity", - generate_hash("cg_weaponVisInterval") - }, - { - "cg_youInKillCamSize", - "Size of the 'you' Icon in the kill cam", - generate_hash("cg_youInKillCamSize") - }, - { - "cl_anglespeedkey", - "Multiplier for max angle speed for game pad and keyboard", - generate_hash("cl_anglespeedkey") - }, - { - "cl_bypassMouseInput", - "Bypass UI mouse input and send directly to the game", - generate_hash("cl_bypassMouseInput") - }, - { - "cl_connectionAttempts", - "Maximum number of connection attempts before aborting", - generate_hash("cl_connectionAttempts") - }, - { - "cl_connectTimeout", - "Timeout time in seconds while connecting to a server", - generate_hash("cl_connectTimeout") - }, - { - "cl_demo_uploadfb", - "Should we upload to FB", - generate_hash("cl_demo_uploadfb") - }, - { - "cl_dirSelConvergenceTime", - "Time to converge to the new direction when selecting a direction on the map.", - generate_hash("cl_dirSelConvergenceTime") - }, - { - "cl_force_paused", - "Force the client to be paused. Can't be overridden by LUA scripts, the start button, etc.", - generate_hash("cl_force_paused") - }, - { - "cl_freelook", - "Enable looking with mouse", - generate_hash("cl_freelook") - }, - { - "cl_hudDrawsBehindUI", - "Should the HUD draw when the UI is up?", - generate_hash("cl_hudDrawsBehindUI") - }, - { - "cl_ingame", - "True if the game is active", - generate_hash("cl_ingame") - }, - { - "cl_inhibit_stats_upload", - "Inhibit upload of stats during demo playback", - generate_hash("cl_inhibit_stats_upload") - }, - { - "cl_lessprint", - "Print less to the console by filtering out certain spammy channels", - generate_hash("cl_lessprint") - }, - { - "cl_maxpackets", - "Maximum number of packets sent per frame", - generate_hash("cl_maxpackets") - }, - { - "cl_maxPing", - "Maximum ping for the client", - generate_hash("cl_maxPing") - }, - { - "cl_migrationTimeout", - "Seconds to wait to hear from new host during host migration before timeout occurs", - generate_hash("cl_migrationTimeout") - }, - { - "cl_modifiedDebugPlacement", - "Modify the location of debug output (outside of safe area)", - generate_hash("cl_modifiedDebugPlacement") - }, - { - "cl_motdString", - "", - generate_hash("cl_motdString") - }, - { - "cl_mouseAccel", - "Mouse acceleration", - generate_hash("cl_mouseAccel") - }, - { - "cl_noprint", - "Print nothing to the console", - generate_hash("cl_noprint") - }, - { - "cl_packetdup", - "Enable packet duplication", - generate_hash("cl_packetdup") - }, - { - "cl_pauseAudioZoneEnabled", - "Enable the paused audio zone when the menus are up", - generate_hash("cl_pauseAudioZoneEnabled") - }, - { - "cl_paused", - "Pause the game", - generate_hash("cl_paused") - }, - { - "cl_pitchspeed", - "Max pitch speed in degrees for game pad", - generate_hash("cl_pitchspeed") - }, - { - "cl_pranks", - "pranks", - generate_hash("cl_pranks") - }, - { - "cl_pushToTalk", - "Do we have to press a button to talk", - generate_hash("cl_pushToTalk") - }, - { - "cl_serverStatusResendTime", - "Time in milliseconds to resend a server status message", - generate_hash("cl_serverStatusResendTime") - }, - { - "cl_showmouserate", - "Print mouse rate debugging information to the console", - generate_hash("cl_showmouserate") - }, - { - "cl_textChatEnabled", - "Do we want to use text chat", - generate_hash("cl_textChatEnabled") - }, - { - "cl_timeout", - "Seconds with no received packets until a timeout occurs", - generate_hash("cl_timeout") - }, - { - "cl_voice", - "Use voice communications", - generate_hash("cl_voice") - }, - { - "cl_yawspeed", - "Max yaw speed in degrees for game pad and keyboard", - generate_hash("cl_yawspeed") - }, - { - "clientSideEffects", - "Enable loading _fx.gsc files on the client", - generate_hash("clientSideEffects") - }, - { - "cod_anywhere_errorMessage", - "CoD Anywhere error message", - generate_hash("cod_anywhere_errorMessage") - }, - { - "cod_anywhere_showPopup", - "Temp Development: Should we show the CoD Anywhere popup", - generate_hash("cod_anywhere_showPopup") - }, - { - "cod_anywhere_single_task_popup_text", - "CoD Anywhere success message", - generate_hash("cod_anywhere_single_task_popup_text") - }, - { - "com_animCheck", - "Check anim tree", - generate_hash("com_animCheck") - }, - { - "com_cinematicEndInWhite", - "Set by script. True if cinematic ends with a white screen.", - generate_hash("com_cinematicEndInWhite") - }, - { - "com_completionResolveCommand", - "Command to run when the message box successfully closes", - generate_hash("com_completionResolveCommand") - }, - { - "com_errorMessage", - "Most recent error message", - generate_hash("com_errorMessage") - }, - { - "com_errorResolveCommand", - "Command to run when they close the error box", - generate_hash("com_errorResolveCommand") - }, - { - "com_filter_output", - "Use console filters for filtering output.", - generate_hash("com_filter_output") - }, - { - "com_maxfps", - "Cap frames per second", - generate_hash("com_maxfps") - }, - { - "com_maxFrameTime", - "Time slows down if a frame takes longer than this many milliseconds", - generate_hash("com_maxFrameTime") - }, - { - "com_playerProfile", - "Set to the name of the profile", - generate_hash("com_playerProfile") - }, - { - "com_recommendedSet", - "", - generate_hash("com_recommendedSet") - }, - { - "commerce_dl_retry_step", - "Step in m/s for the commerce download retry", - generate_hash("commerce_dl_retry_step") - }, - { - "commerce_manifest_file_max_retry_time", - "Max time that the commerce manifest can retry", - generate_hash("commerce_manifest_file_max_retry_time") - }, - { - "commerce_manifest_file_retry_step", - "Step in m/s for the commerce manifest retry", - generate_hash("commerce_manifest_file_retry_step") - }, - { - "commerce_max_dl_retry_time", - "Max time that the commerce download can retry", - generate_hash("commerce_max_dl_retry_time") - }, - { - "commerce_max_retry_time", - "Max time that the commerce upload can retry", - generate_hash("commerce_max_retry_time") - }, - { - "commerce_retry_step", - "Step in m/s for the commerce upload retry", - generate_hash("commerce_retry_step") - }, - { - "compass", - "Display Compass", - generate_hash("compass") - }, - { - "compassClampIcons", - "If true, friendlies and enemy pings clamp to the edge of the radar. If false, they disappear off the edge.", - generate_hash("compassClampIcons") - }, - { - "compassCoords", - "x = North-South coord base value, \ny = East-West coord base value, \nz = scale (game units per coord unit)", - generate_hash("compassCoords") - }, - { - "compassECoordCutoff", - "Left cutoff for the scrolling east-west coords", - generate_hash("compassECoordCutoff") - }, - { - "compassFriendlyHeight", - "", - generate_hash("compassFriendlyHeight") - }, - { - "compassFriendlyWidth", - "", - generate_hash("compassFriendlyWidth") - }, - { - "compassHideSansObjectivePointer", - "Hide the compass, but leave the obective pointer visible.", - generate_hash("compassHideSansObjectivePointer") - }, - { - "compassHideVehicles", - "When enabled, disables the CG_PLAYER_COMPASS_VEHICLES ownerdraw.", - generate_hash("compassHideVehicles") - }, - { - "compassMaxRange", - "The maximum range from the player in world space that objects will be shown on the compass", - generate_hash("compassMaxRange") - }, - { - "compassMinRadius", - "The minimum radius from the center of the compass that objects will appear.", - generate_hash("compassMinRadius") - }, - { - "compassMinRange", - "The minimum range from the player in world space that objects will appear on the compass", - generate_hash("compassMinRange") - }, - { - "compassObjectiveArrowHeight", - "", - generate_hash("compassObjectiveArrowHeight") - }, - { - "compassObjectiveArrowOffset", - "The offset of the objective arrow inward from the edge of the compass map", - generate_hash("compassObjectiveArrowOffset") - }, - { - "compassObjectiveArrowRotateDist", - "Distance from the corner of the compass map at which the objective arrow rotates to 45 degrees", - generate_hash("compassObjectiveArrowRotateDist") - }, - { - "compassObjectiveArrowWidth", - "", - generate_hash("compassObjectiveArrowWidth") - }, - { - "compassObjectiveDetailDist", - "When an objective is closer than this distance (in meters), the icon will not be drawn on the tickertape.", - generate_hash("compassObjectiveDetailDist") - }, - { - "compassObjectiveDrawLines", - "Draw horizontal and vertical lines to the active target, if it is within the minimap boundries", - generate_hash("compassObjectiveDrawLines") - }, - { - "compassObjectiveHeight", - "", - generate_hash("compassObjectiveHeight") - }, - { - "compassObjectiveIconHeight", - "", - generate_hash("compassObjectiveIconHeight") - }, - { - "compassObjectiveIconWidth", - "", - generate_hash("compassObjectiveIconWidth") - }, - { - "compassObjectiveMaxHeight", - "The maximum height that an objective is considered to be on this level", - generate_hash("compassObjectiveMaxHeight") - }, - { - "compassObjectiveMaxRange", - "The maximum range at which an objective is visible on the compass", - generate_hash("compassObjectiveMaxRange") - }, - { - "compassObjectiveMinAlpha", - "The minimum alpha for an objective at the edge of the compass", - generate_hash("compassObjectiveMinAlpha") - }, - { - "compassObjectiveMinDistRange", - "The distance that objective transition effects play over, centered on compassObjectiveNearbyDist.", - generate_hash("compassObjectiveMinDistRange") - }, - { - "compassObjectiveMinHeight", - "The minimum height that an objective is considered to be on this level", - generate_hash("compassObjectiveMinHeight") - }, - { - "compassObjectiveNearbyDist", - "When an objective is closer than this distance (in meters), the icon will not be drawn on the tickertape.", - generate_hash("compassObjectiveNearbyDist") - }, - { - "compassObjectiveNumRings", - "The number of rings when a new objective appears", - generate_hash("compassObjectiveNumRings") - }, - { - "compassObjectiveRingSize", - "The maximum objective ring sige when a new objective appears on the compass", - generate_hash("compassObjectiveRingSize") - }, - { - "compassObjectiveRingTime", - "The amount of time between each ring when an objective appears", - generate_hash("compassObjectiveRingTime") - }, - { - "compassObjectiveTextHeight", - "Objective text height", - generate_hash("compassObjectiveTextHeight") - }, - { - "compassObjectiveTextScale", - "Scale to apply to hud objectives", - generate_hash("compassObjectiveTextScale") - }, - { - "compassObjectiveWidth", - "", - generate_hash("compassObjectiveWidth") - }, - { - "compassObjectiveWraparoundTime", - "How long it takes for the objective to wrap around the compass from one edge to the other", - generate_hash("compassObjectiveWraparoundTime") - }, - { - "compassPlayerHeight", - "", - generate_hash("compassPlayerHeight") - }, - { - "compassPlayerWidth", - "", - generate_hash("compassPlayerWidth") - }, - { - "compassRadarLineThickness", - "Thickness, relative to the compass size, of the radar texture that sweeps across the map", - generate_hash("compassRadarLineThickness") - }, - { - "compassRadarPingFadeTime", - "How long an enemy is visible on the compass after it is detected by radar", - generate_hash("compassRadarPingFadeTime") - }, - { - "compassRotation", - "Style of compass", - generate_hash("compassRotation") - }, - { - "compassSize", - "Scale the compass", - generate_hash("compassSize") - }, - { - "compassSoundPingFadeTime", - "The time in seconds for the sound overlay on the compass to fade", - generate_hash("compassSoundPingFadeTime") - }, - { - "compassTickertapeStretch", - "How far the tickertape should stretch from its center.", - generate_hash("compassTickertapeStretch") - }, - { - "comscore_active", - "Are we allowed to enable ComScore tracking or not", - generate_hash("comscore_active") - }, - { - "con_gameMsgWindow0FadeInTime", - "", - generate_hash("con_gameMsgWindow0FadeInTime") - }, - { - "con_gameMsgWindow0FadeOutTime", - "", - generate_hash("con_gameMsgWindow0FadeOutTime") - }, - { - "con_gameMsgWindow0Filter", - "", - generate_hash("con_gameMsgWindow0Filter") - }, - { - "con_gameMsgWindow0LineCount", - "", - generate_hash("con_gameMsgWindow0LineCount") - }, - { - "con_gameMsgWindow0MsgTime", - "", - generate_hash("con_gameMsgWindow0MsgTime") - }, - { - "con_gameMsgWindow0ScrollTime", - "", - generate_hash("con_gameMsgWindow0ScrollTime") - }, - { - "con_gameMsgWindow1FadeInTime", - "", - generate_hash("con_gameMsgWindow1FadeInTime") - }, - { - "con_gameMsgWindow1FadeOutTime", - "", - generate_hash("con_gameMsgWindow1FadeOutTime") - }, - { - "con_gameMsgWindow1Filter", - "", - generate_hash("con_gameMsgWindow1Filter") - }, - { - "con_gameMsgWindow1LineCount", - "", - generate_hash("con_gameMsgWindow1LineCount") - }, - { - "con_gameMsgWindow1MsgTime", - "", - generate_hash("con_gameMsgWindow1MsgTime") - }, - { - "con_gameMsgWindow1ScrollTime", - "", - generate_hash("con_gameMsgWindow1ScrollTime") - }, - { - "con_gameMsgWindow2FadeInTime", - "", - generate_hash("con_gameMsgWindow2FadeInTime") - }, - { - "con_gameMsgWindow2FadeOutTime", - "", - generate_hash("con_gameMsgWindow2FadeOutTime") - }, - { - "con_gameMsgWindow2Filter", - "", - generate_hash("con_gameMsgWindow2Filter") - }, - { - "con_gameMsgWindow2LineCount", - "", - generate_hash("con_gameMsgWindow2LineCount") - }, - { - "con_gameMsgWindow2MsgTime", - "", - generate_hash("con_gameMsgWindow2MsgTime") - }, - { - "con_gameMsgWindow2ScrollTime", - "", - generate_hash("con_gameMsgWindow2ScrollTime") - }, - { - "con_gameMsgWindow3FadeInTime", - "", - generate_hash("con_gameMsgWindow3FadeInTime") - }, - { - "con_gameMsgWindow3FadeOutTime", - "", - generate_hash("con_gameMsgWindow3FadeOutTime") - }, - { - "con_gameMsgWindow3Filter", - "", - generate_hash("con_gameMsgWindow3Filter") - }, - { - "con_gameMsgWindow3LineCount", - "", - generate_hash("con_gameMsgWindow3LineCount") - }, - { - "con_gameMsgWindow3MsgTime", - "", - generate_hash("con_gameMsgWindow3MsgTime") - }, - { - "con_gameMsgWindow3ScrollTime", - "", - generate_hash("con_gameMsgWindow3ScrollTime") - }, - { - "con_subtitleLeading", - "Leading for subtitles, calculated as a percentage of the font height", - generate_hash("con_subtitleLeading") - }, - { - "con_typewriterColorGlowCheckpoint", - "", - generate_hash("con_typewriterColorGlowCheckpoint") - }, - { - "con_typewriterColorGlowCompleted", - "", - generate_hash("con_typewriterColorGlowCompleted") - }, - { - "con_typewriterColorGlowFailed", - "", - generate_hash("con_typewriterColorGlowFailed") - }, - { - "con_typewriterColorGlowUpdated", - "", - generate_hash("con_typewriterColorGlowUpdated") - }, - { - "con_typewriterColorInteriorCheckpoint", - "", - generate_hash("con_typewriterColorInteriorCheckpoint") - }, - { - "con_typewriterColorInteriorCompleted", - "", - generate_hash("con_typewriterColorInteriorCompleted") - }, - { - "con_typewriterColorInteriorFailed", - "", - generate_hash("con_typewriterColorInteriorFailed") - }, - { - "con_typewriterColorInteriorUpdated", - "", - generate_hash("con_typewriterColorInteriorUpdated") - }, - { - "con_typewriterDecayDuration", - "Time (in milliseconds) to spend disolving the line away.", - generate_hash("con_typewriterDecayDuration") - }, - { - "con_typewriterDecayStartTime", - "Time (in milliseconds) to spend between the build and disolve phases.", - generate_hash("con_typewriterDecayStartTime") - }, - { - "con_typewriterPrintSpeed", - "Time (in milliseconds) to print each letter in the line.", - generate_hash("con_typewriterPrintSpeed") - }, - { - "counterDownloadInterval", - "Number of minutes before all the global counters are uploaded", - generate_hash("counterDownloadInterval") - }, - { - "counterUploadInterval", - "Number of minutes before all the global counters are uploaded", - generate_hash("counterUploadInterval") - }, - { - "cpu_speed_12players", - "12 player sys_configureGHz req'd", - generate_hash("cpu_speed_12players") - }, - { - "cpu_speed_18players", - "18 player sys_configureGHz req'd", - generate_hash("cpu_speed_18players") - }, - { - "cpu_speed_8players", - "8 player sys_configureGHz req'd", - generate_hash("cpu_speed_8players") - }, - { - "cSplineDebugRender", - "Debug Render the csplines.", - generate_hash("cSplineDebugRender") - }, - { - "cSplineDebugRenderCorridor", - "Debug Render the cspline corridor.", - generate_hash("cSplineDebugRenderCorridor") - }, - { - "cSplineDebugRenderData", - "Debug Render the cspline data.", - generate_hash("cSplineDebugRenderData") - }, - { - "cSplineDebugRenderSplineId", - "Select a cspline - 0 for all.", - generate_hash("cSplineDebugRenderSplineId") - }, - { - "dailychallenge_killswitch", - "daily challenge killswitch - int with bits used to flag individual daily challenges as enabled", - generate_hash("dailychallenge_killswitch") - }, - { - "dailychallenge_killswitch2", - "daily challenge killswitch2 - int with bits used to flag 2nd set of individual daily challenges as enabled", - generate_hash("dailychallenge_killswitch2") - }, - { - "dailychallenge_period", - "daily challenge period - utc value for a day", - generate_hash("dailychallenge_period") - }, - { - "data_validation_allow_drop", - "", - generate_hash("data_validation_allow_drop") - }, - { - "dc_lobbymerge", - "Allows lobby merging across data centres", - generate_hash("dc_lobbymerge") - }, - { - "dcacheSimulateNoHDD", - "When turned on, simulate no HDD for caching.", - generate_hash("dcacheSimulateNoHDD") - }, - { - "dcacheThrottleEnabled", - "Enable or disable dcache upload throttling.", - generate_hash("dcacheThrottleEnabled") - }, - { - "dcacheThrottleKBytesPerSec", - "Dcache upload throttle limit in K Bytes per second.", - generate_hash("dcacheThrottleKBytesPerSec") - }, - { - "dedicated_dhclient", - "True if we're a client playing on a DH server", - generate_hash("dedicated_dhclient") - }, - { - "demonwareConsideredConnectedTime", - "Number of milliseconds after being disconnected from demonware before considering shutting down.", - generate_hash("demonwareConsideredConnectedTime") - }, - { - "developer", - "Enable development options", - generate_hash("developer") - }, - { - "didyouknow", - "", - generate_hash("didyouknow") - }, - { - "discard_playerstats_on_suspend", - "Forces stats discard on suspend", - generate_hash("discard_playerstats_on_suspend") - }, - { - "drawEntityCount", - "Enable entity count drawing", - generate_hash("drawEntityCount") - }, - { - "drawEntityCountPos", - "Where to draw the entity count graph", - generate_hash("drawEntityCountPos") - }, - { - "drawEntityCountSize", - "undefined", - generate_hash("drawEntityCountSize") - }, - { - "drawKillcamData", - "Enable drawing server killcam data", - generate_hash("drawKillcamData") - }, - { - "drawKillcamDataPos", - "Where to draw the server killcam graph", - generate_hash("drawKillcamDataPos") - }, - { - "drawKillcamDataSize", - "How big to draw the killcam data graph", - generate_hash("drawKillcamDataSize") - }, - { - "drawServerBandwidth", - "Enable drawing server bandwidth", - generate_hash("drawServerBandwidth") - }, - { - "drawServerBandwidthPos", - "Where to draw the server bandwidth graph", - generate_hash("drawServerBandwidthPos") - }, - { - "ds_dcid", - "optional datacenter id - from playlist", - generate_hash("ds_dcid") - }, - { - "ds_dcid_override", - "force datacenter id", - generate_hash("ds_dcid_override") - }, - { - "ds_info", - "ds info string", - generate_hash("ds_info") - }, - { - "ds_info_enable", - "Enable ds info string", - generate_hash("ds_info_enable") - }, - { - "ds_introRequestTimeout", - "ds intro request timeout (ms)", - generate_hash("ds_introRequestTimeout") - }, - { - "ds_keepaliveInterval", - "ds keepalive interval (ms)", - generate_hash("ds_keepaliveInterval") - }, - { - "ds_keepaliveTimeout", - "ds keepalive timeout (ms)", - generate_hash("ds_keepaliveTimeout") - }, - { - "ds_pingclient_max_reping_distance", - "don't re-ping a datacenter if it's further away than this (miles)", - generate_hash("ds_pingclient_max_reping_distance") - }, - { - "ds_pingclient_max_repings", - "max # of times to re-ping a datacenter", - generate_hash("ds_pingclient_max_repings") - }, - { - "ds_pingclient_maxpings", - "max pings to send per datacenter", - generate_hash("ds_pingclient_maxpings") - }, - { - "ds_pingclient_maxpings_per_tick", - "max new pings each tick", - generate_hash("ds_pingclient_maxpings_per_tick") - }, - { - "ds_pingclient_min_reping_delay", - "min msec delay between re-pings", - generate_hash("ds_pingclient_min_reping_delay") - }, - { - "ds_pingclient_min_reping_latency", - "don't re-ping a datacenter if latency is less than this", - generate_hash("ds_pingclient_min_reping_latency") - }, - { - "ds_pingclient_minpings", - "min responses required per datacenter", - generate_hash("ds_pingclient_minpings") - }, - { - "ds_pingclient_odsf", - "does dsping set odsf flag", - generate_hash("ds_pingclient_odsf") - }, - { - "dsping_dc_0", - "", - generate_hash("dsping_dc_0") - }, - { - "dsping_dc_1", - "", - generate_hash("dsping_dc_1") - }, - { - "dsping_dc_10", - "", - generate_hash("dsping_dc_10") - }, - { - "dsping_dc_11", - "", - generate_hash("dsping_dc_11") - }, - { - "dsping_dc_12", - "", - generate_hash("dsping_dc_12") - }, - { - "dsping_dc_13", - "", - generate_hash("dsping_dc_13") - }, - { - "dsping_dc_14", - "", - generate_hash("dsping_dc_14") - }, - { - "dsping_dc_15", - "", - generate_hash("dsping_dc_15") - }, - { - "dsping_dc_16", - "", - generate_hash("dsping_dc_16") - }, - { - "dsping_dc_17", - "", - generate_hash("dsping_dc_17") - }, - { - "dsping_dc_18", - "", - generate_hash("dsping_dc_18") - }, - { - "dsping_dc_19", - "", - generate_hash("dsping_dc_19") - }, - { - "dsping_dc_2", - "", - generate_hash("dsping_dc_2") - }, - { - "dsping_dc_20", - "", - generate_hash("dsping_dc_20") - }, - { - "dsping_dc_21", - "", - generate_hash("dsping_dc_21") - }, - { - "dsping_dc_22", - "", - generate_hash("dsping_dc_22") - }, - { - "dsping_dc_23", - "", - generate_hash("dsping_dc_23") - }, - { - "dsping_dc_24", - "", - generate_hash("dsping_dc_24") - }, - { - "dsping_dc_25", - "", - generate_hash("dsping_dc_25") - }, - { - "dsping_dc_26", - "", - generate_hash("dsping_dc_26") - }, - { - "dsping_dc_27", - "", - generate_hash("dsping_dc_27") - }, - { - "dsping_dc_28", - "", - generate_hash("dsping_dc_28") - }, - { - "dsping_dc_29", - "", - generate_hash("dsping_dc_29") - }, - { - "dsping_dc_3", - "", - generate_hash("dsping_dc_3") - }, - { - "dsping_dc_30", - "", - generate_hash("dsping_dc_30") - }, - { - "dsping_dc_31", - "", - generate_hash("dsping_dc_31") - }, - { - "dsping_dc_32", - "", - generate_hash("dsping_dc_32") - }, - { - "dsping_dc_33", - "", - generate_hash("dsping_dc_33") - }, - { - "dsping_dc_34", - "", - generate_hash("dsping_dc_34") - }, - { - "dsping_dc_35", - "", - generate_hash("dsping_dc_35") - }, - { - "dsping_dc_36", - "", - generate_hash("dsping_dc_36") - }, - { - "dsping_dc_37", - "", - generate_hash("dsping_dc_37") - }, - { - "dsping_dc_38", - "", - generate_hash("dsping_dc_38") - }, - { - "dsping_dc_39", - "", - generate_hash("dsping_dc_39") - }, - { - "dsping_dc_4", - "", - generate_hash("dsping_dc_4") - }, - { - "dsping_dc_40", - "", - generate_hash("dsping_dc_40") - }, - { - "dsping_dc_41", - "", - generate_hash("dsping_dc_41") - }, - { - "dsping_dc_42", - "", - generate_hash("dsping_dc_42") - }, - { - "dsping_dc_43", - "", - generate_hash("dsping_dc_43") - }, - { - "dsping_dc_44", - "", - generate_hash("dsping_dc_44") - }, - { - "dsping_dc_45", - "", - generate_hash("dsping_dc_45") - }, - { - "dsping_dc_46", - "", - generate_hash("dsping_dc_46") - }, - { - "dsping_dc_47", - "", - generate_hash("dsping_dc_47") - }, - { - "dsping_dc_48", - "", - generate_hash("dsping_dc_48") - }, - { - "dsping_dc_49", - "", - generate_hash("dsping_dc_49") - }, - { - "dsping_dc_5", - "", - generate_hash("dsping_dc_5") - }, - { - "dsping_dc_50", - "", - generate_hash("dsping_dc_50") - }, - { - "dsping_dc_51", - "", - generate_hash("dsping_dc_51") - }, - { - "dsping_dc_52", - "", - generate_hash("dsping_dc_52") - }, - { - "dsping_dc_53", - "", - generate_hash("dsping_dc_53") - }, - { - "dsping_dc_54", - "", - generate_hash("dsping_dc_54") - }, - { - "dsping_dc_55", - "", - generate_hash("dsping_dc_55") - }, - { - "dsping_dc_56", - "", - generate_hash("dsping_dc_56") - }, - { - "dsping_dc_57", - "", - generate_hash("dsping_dc_57") - }, - { - "dsping_dc_58", - "", - generate_hash("dsping_dc_58") - }, - { - "dsping_dc_59", - "", - generate_hash("dsping_dc_59") - }, - { - "dsping_dc_6", - "", - generate_hash("dsping_dc_6") - }, - { - "dsping_dc_60", - "", - generate_hash("dsping_dc_60") - }, - { - "dsping_dc_61", - "", - generate_hash("dsping_dc_61") - }, - { - "dsping_dc_62", - "", - generate_hash("dsping_dc_62") - }, - { - "dsping_dc_63", - "", - generate_hash("dsping_dc_63") - }, - { - "dsping_dc_7", - "", - generate_hash("dsping_dc_7") - }, - { - "dsping_dc_8", - "", - generate_hash("dsping_dc_8") - }, - { - "dsping_dc_9", - "", - generate_hash("dsping_dc_9") - }, - { - "dvl", - "Enables the data validation system. Only available in non-retail builds.", - generate_hash("dvl") - }, - { - "dw_addrHandleTimeout", - "Delay before destroying an addrHandle after the connection is lost\n", - generate_hash("dw_addrHandleTimeout") - }, - { - "dw_leaderboard_write_active", - "Are leaderboard writes enabled", - generate_hash("dw_leaderboard_write_active") - }, - { - "dw_presence_active", - "Is the demonware presence system enabled", - generate_hash("dw_presence_active") - }, - { - "dw_presence_coop_join_active", - "Do we allow players to join on presence for private coop matches (post session to demonware", - generate_hash("dw_presence_coop_join_active") - }, - { - "dw_presence_get_delay", - "Number of milliseconds to wait after booting the game to fetch demonware presence", - generate_hash("dw_presence_get_delay") - }, - { - "dw_presence_get_rate", - "Number of milliseconds to wait between sending presence state to demonware", - generate_hash("dw_presence_get_rate") - }, - { - "dw_presence_put_delay", - "Number of milliseconds to wait in a presence state before sending to demonware", - generate_hash("dw_presence_put_delay") - }, - { - "dw_presence_put_rate", - "Number of milliseconds to wait between sending presence state to demonware", - generate_hash("dw_presence_put_rate") - }, - { - "dw_region_lookup_timeout", - "Timeout (in MS) after which we will accept not having found a region code and use the default", - generate_hash("dw_region_lookup_timeout") - }, - { - "dw_shared_presence_active", - "Is the demonware shared presence system enabled", - generate_hash("dw_shared_presence_active") - }, - { - "dw_shared_presence_get_delay", - "Number of milliseconds to wait after booting the game to fetch demonware presence", - generate_hash("dw_shared_presence_get_delay") - }, - { - "dw_shared_presence_get_rate", - "Number of milliseconds to wait between sending presence state to demonware", - generate_hash("dw_shared_presence_get_rate") - }, - { - "dw_shared_presence_put_delay", - "Number of milliseconds to wait in a shared presence state before sending to demonware", - generate_hash("dw_shared_presence_put_delay") - }, - { - "dw_shared_presence_put_rate", - "Number of milliseconds to wait between sending presence state to demonware", - generate_hash("dw_shared_presence_put_rate") - }, - { - "dwBandwidthTestTaskTimeout", - "default timeout for the bandwidth test task (in ms). 0 means no timeout", - generate_hash("dwBandwidthTestTaskTimeout") - }, - { - "dynEnt_active", - "Disable/enable dynent reactions", - generate_hash("dynEnt_active") - }, - { - "dynEnt_playerWakeUpRadius", - "Determines threshold distance from player within which all dynents are woken up.", - generate_hash("dynEnt_playerWakeUpRadius") - }, - { - "dynEnt_playerWakeUpZOffset", - "Determines vertical distance from player's feet from which wake up sphere is centered.", - generate_hash("dynEnt_playerWakeUpZOffset") - }, - { - "elite_clan_active", - "Are we allowed to show Elite Clans or not", - generate_hash("elite_clan_active") - }, - { - "elite_clan_cool_off_time", - "Cool off time between calls to fetch the elite clan", - generate_hash("elite_clan_cool_off_time") - }, - { - "elite_clan_delay", - "Delay before the bdTeams calls start to Demonware. -1 means On-Demand and it will wait until the 'starteliteclan' menu call", - generate_hash("elite_clan_delay") - }, - { - "elite_clan_division_icon_active", - "Are we allowed to show Elite Clan division icon or not", - generate_hash("elite_clan_division_icon_active") - }, - { - "elite_clan_get_blob_profile_max_retry_time", - "Max time that the Elite Clan get private profile can retry", - generate_hash("elite_clan_get_blob_profile_max_retry_time") - }, - { - "elite_clan_get_blob_profile_retry_step", - "Step in m/s for the Elite Clan get private profile retry", - generate_hash("elite_clan_get_blob_profile_retry_step") - }, - { - "elite_clan_get_clan_max_retry_time", - "Max time that the Elite Clan get clan can retry", - generate_hash("elite_clan_get_clan_max_retry_time") - }, - { - "elite_clan_get_clan_retry_step", - "Step in m/s for the Elite Clan get clan retry", - generate_hash("elite_clan_get_clan_retry_step") - }, - { - "elite_clan_get_members_max_retry_time", - "Max time that the Elite Clan get members can retry", - generate_hash("elite_clan_get_members_max_retry_time") - }, - { - "elite_clan_get_members_retry_step", - "Step in m/s for the Elite Clan get members retry", - generate_hash("elite_clan_get_members_retry_step") - }, - { - "elite_clan_get_private_member_profile_max_retry_time", - "Max time that the Elite Clan get private profile can retry", - generate_hash("elite_clan_get_private_member_profile_max_retry_time") - }, - { - "elite_clan_get_private_member_profile_retry_step", - "Step in m/s for the Elite Clan get private profile retry", - generate_hash("elite_clan_get_private_member_profile_retry_step") - }, - { - "elite_clan_get_public_profile_max_retry_time", - "Max time that the Elite Clan get public profile can retry", - generate_hash("elite_clan_get_public_profile_max_retry_time") - }, - { - "elite_clan_get_public_profile_retry_step", - "Step in m/s for the Elite Clan get public profile retry", - generate_hash("elite_clan_get_public_profile_retry_step") - }, - { - "elite_clan_get_team_stats_max_retry_time", - "Max time that the Elite Clan get team stats can retry", - generate_hash("elite_clan_get_team_stats_max_retry_time") - }, - { - "elite_clan_get_team_stats_retry_step", - "Step in m/s for the Elite Clan get team stats retry", - generate_hash("elite_clan_get_team_stats_retry_step") - }, - { - "elite_clan_motd_throttle_time", - "Throttle time between motd update calls", - generate_hash("elite_clan_motd_throttle_time") - }, - { - "elite_clan_remote_view_active", - "Are we allowed to view the clans for remote players", - generate_hash("elite_clan_remote_view_active") - }, - { - "elite_clan_remote_view_max_retry_time", - "Max time that the Elite Clan remote viewing can retry", - generate_hash("elite_clan_remote_view_max_retry_time") - }, - { - "elite_clan_remote_view_retry_step", - "Step in m/s for the retry for viewing a remote Elite Clan", - generate_hash("elite_clan_remote_view_retry_step") - }, - { - "elite_clan_send_message_to_members_max_retry_time", - "Max time that the Elite Clan send message to members can retry", - generate_hash("elite_clan_send_message_to_members_max_retry_time") - }, - { - "elite_clan_send_message_to_members_retry_step", - "Step in m/s for the Elite Clan send message to members retry", - generate_hash("elite_clan_send_message_to_members_retry_step") - }, - { - "elite_clan_set_private_member_profile_max_retry_time", - "Max time that the Elite Clan set private member profile can retry", - generate_hash("elite_clan_set_private_member_profile_max_retry_time") - }, - { - "elite_clan_set_private_member_profile_retry_step", - "Step in m/s for the Elite Clan set private member profile retry", - generate_hash("elite_clan_set_private_member_profile_retry_step") - }, - { - "elite_clan_single_task_popup_text", - "String to be displayed on popup when a single task is being performed", - generate_hash("elite_clan_single_task_popup_text") - }, - { - "elite_clan_using_title", - "Stores whether the Elite Clan title is in use by the user", - generate_hash("elite_clan_using_title") - }, - { - "emblems_active", - "Are we allowed to enable Emblems or not", - generate_hash("emblems_active") - }, - { - "enable_recordRecentActivity", - "records the timestamp of when the player was recently active to the tracker leaderboards", - generate_hash("enable_recordRecentActivity") - }, - { - "enableReportingRegisteredParties", - "If true then party membership data and host status will be reported in matchdata blob.", - generate_hash("enableReportingRegisteredParties") - }, - { - "entitlements_active", - "Are we allowed to show Entitlements or not", - generate_hash("entitlements_active") - }, - { - "entitlements_config_file_max_retry_time", - "Max time that the Entitlements config file read can retry", - generate_hash("entitlements_config_file_max_retry_time") - }, - { - "entitlements_config_file_retry_step", - "Step in m/s for the Entitlements config file read retry", - generate_hash("entitlements_config_file_retry_step") - }, - { - "entitlements_cool_off_time", - "Cool off time between calls to fetch the elite clan", - generate_hash("entitlements_cool_off_time") - }, - { - "entitlements_delay", - "Delay before the entitlement calls start to Demonware. -1 means On-Demand and it will wait until the 'startentitlements' menu call", - generate_hash("entitlements_delay") - }, - { - "entitlements_key_archive_max_retry_time", - "Max time that the Entitlements key archive read can retry", - generate_hash("entitlements_key_archive_max_retry_time") - }, - { - "entitlements_key_archive_retry_step", - "Step in m/s for the Entitlements key archive read retry", - generate_hash("entitlements_key_archive_retry_step") - }, - { - "entitlementSystemOk", - "Set by the game to inform that the entitlement system is initialised", - generate_hash("entitlementSystemOk") - }, - { - "facebook_active", - "Are we allowed to show Facebook or not", - generate_hash("facebook_active") - }, - { - "facebook_delay", - "Delay before the Facebook calls start to Demonware. -1 means On-Demand and it will wait until the 'startfacebook' menu call", - generate_hash("facebook_delay") - }, - { - "facebook_friends_active", - "Are we allowed to show Facebook Friends or not", - generate_hash("facebook_friends_active") - }, - { - "facebook_friends_max_retry_time", - "Max time that the Facebook friends read can retry", - generate_hash("facebook_friends_max_retry_time") - }, - { - "facebook_friends_refresh_time", - "Time in seconds between Facebook friend refreshes", - generate_hash("facebook_friends_refresh_time") - }, - { - "facebook_friends_retry_step", - "Step in m/s for the Facebook friends read retry", - generate_hash("facebook_friends_retry_step") - }, - { - "facebook_friends_showing_count", - "Contains how many facebook friends are being shown in the UI.", - generate_hash("facebook_friends_showing_count") - }, - { - "facebook_friends_throttle_time", - "Throttle time between Facebook friend pages", - generate_hash("facebook_friends_throttle_time") - }, - { - "facebook_max_retry_time", - "Max time that the Facebook authentication can retry", - generate_hash("facebook_max_retry_time") - }, - { - "facebook_password", - "Facebook Password", - generate_hash("facebook_password") - }, - { - "facebook_password_asterisk", - "Facebook Password (Asterisk Version)", - generate_hash("facebook_password_asterisk") - }, - { - "facebook_popup_text", - "Facebook Popup Text", - generate_hash("facebook_popup_text") - }, - { - "facebook_retry_step", - "Step in m/s for the Facebook authentication retry", - generate_hash("facebook_retry_step") - }, - { - "facebook_upload_photo_active", - "Are we allowed to Upload Photos to Facebook or not", - generate_hash("facebook_upload_photo_active") - }, - { - "facebook_upload_video_active", - "Are we allowed to Upload Videos to Facebook or not", - generate_hash("facebook_upload_video_active") - }, - { - "facebook_username", - "Facebook Username", - generate_hash("facebook_username") - }, - { - "fixedtime", - "Use a fixed time rate for each frame", - generate_hash("fixedtime") - }, - { - "FoFIconMaxSize", - "Maximum size a Friend-or-Foe icon should ever grow to.", - generate_hash("FoFIconMaxSize") - }, - { - "FoFIconMinSize", - "Minimum size a Friend-or-Foe icon should ever shrink to.", - generate_hash("FoFIconMinSize") - }, - { - "FoFIconScale", - "Base scale of Friend-or-Foe icons.", - generate_hash("FoFIconScale") - }, - { - "FoFIconSpawnTimeDelay", - "How long to wait, after spawning, before showing the Friend-or-Foe icon on a player.", - generate_hash("FoFIconSpawnTimeDelay") - }, - { - "FoFIconSpawnTimeFade", - "Length of the Friend-or-Foe icons' fade-ins.", - generate_hash("FoFIconSpawnTimeFade") - }, - { - "friendsCacheSteamFriends", - "Use cache of steam friends before querying steam api", - generate_hash("friendsCacheSteamFriends") - }, - { - "friendsMaxSteamLookupsPerFrame", - "Number of steam friends to query steam status per frame when doing a refresh.\n", - generate_hash("friendsMaxSteamLookupsPerFrame") - }, - { - "friendsWidgetMinimumRefreshTimer", - "Minimum delay before refreshing friends data if you aren't on the friends screen\n", - generate_hash("friendsWidgetMinimumRefreshTimer") - }, - { - "fs_basegame", - "Base game name", - generate_hash("fs_basegame") - }, - { - "fs_basepath", - "Base game path", - generate_hash("fs_basepath") - }, - { - "fs_basepath_output", - "Base game path", - generate_hash("fs_basepath_output") - }, - { - "fs_cdpath", - "CD path", - generate_hash("fs_cdpath") - }, - { - "fs_copyfiles", - "Copy all used files to another location", - generate_hash("fs_copyfiles") - }, - { - "fs_debug", - "Enable file system debugging information", - generate_hash("fs_debug") - }, - { - "fs_game", - "Game data directory. Must be \"\" or a sub directory of 'mods/'.", - generate_hash("fs_game") - }, - { - "fs_homepath", - "Game home path", - generate_hash("fs_homepath") - }, - { - "fs_ignoreLocalized", - "Ignore localized assets", - generate_hash("fs_ignoreLocalized") - }, - { - "fx_alphaThreshold", - "Don't draw billboard sprites, oriented sprites or tails with alpha below this threshold (0-256).", - generate_hash("fx_alphaThreshold") - }, - { - "fx_cast_shadow", - "Enable transparency shadow mapping from script", - generate_hash("fx_cast_shadow") - }, - { - "fx_count", - "Debug effects count", - generate_hash("fx_count") - }, - { - "fx_cull_elem_draw", - "Culls effect elems for drawing", - generate_hash("fx_cull_elem_draw") - }, - { - "fx_cull_elem_draw_flicker", - "Flicker DPVS culled effect elems", - generate_hash("fx_cull_elem_draw_flicker") - }, - { - "fx_cull_elem_spawn", - "Culls effect elems for spawning", - generate_hash("fx_cull_elem_spawn") - }, - { - "fx_debugBolt", - "Debug effects bolt", - generate_hash("fx_debugBolt") - }, - { - "fx_deferelem", - "Toggles deferred processing of elements instead of effects", - generate_hash("fx_deferelem") - }, - { - "fx_dpvs_cull_elem_draw", - "Culls effect elems for drawing using DPVS(2: ignore per-effect portal culling flag)", - generate_hash("fx_dpvs_cull_elem_draw") - }, - { - "fx_draw", - "", - generate_hash("fx_draw") - }, - { - "fx_draw_omniLight", - "", - generate_hash("fx_draw_omniLight") - }, - { - "fx_draw_simd", - "Draw effects using SIMD / Vector code.", - generate_hash("fx_draw_simd") - }, - { - "fx_draw_spotLight", - "", - generate_hash("fx_draw_spotLight") - }, - { - "fx_drawClouds", - "Toggles the drawing of particle clouds", - generate_hash("fx_drawClouds") - }, - { - "fx_enable", - "Toggles all effects processing", - generate_hash("fx_enable") - }, - { - "fx_flare", - "Toggles fx flare", - generate_hash("fx_flare") - }, - { - "fx_freeze", - "Freeze effects", - generate_hash("fx_freeze") - }, - { - "fx_killEffectOnRewind", - "Causes effects that have been marked for a soft kill (fade out) to be killed immediately on a rewind.", - generate_hash("fx_killEffectOnRewind") - }, - { - "fx_lightGridSampleOffset", - "the length of effect sample's offset along X Axis", - generate_hash("fx_lightGridSampleOffset") - }, - { - "fx_mark_profile", - "Turn on FX profiling for marks (specify which local client, with '1' being the first.)", - generate_hash("fx_mark_profile") - }, - { - "fx_marks", - "Toggles whether bullet hits leave marks", - generate_hash("fx_marks") - }, - { - "fx_marks_ents", - "Toggles whether bullet hits leave marks", - generate_hash("fx_marks_ents") - }, - { - "fx_marks_nearlimit", - "Sets limit of number of decals that can exist at the same location (0 for unlimited)", - generate_hash("fx_marks_nearlimit") - }, - { - "fx_marks_smodels", - "Toggles whether bullet hits leave marks", - generate_hash("fx_marks_smodels") - }, - { - "fx_physicsImpactVelocityThreshold", - "Set the min normal velocity threshold in order for model physics fx to generate child impact effects.", - generate_hash("fx_physicsImpactVelocityThreshold") - }, - { - "fx_profile", - "Turn on FX profiling (specify which local client, with '1' being the first.)", - generate_hash("fx_profile") - }, - { - "fx_profileFilter", - "Only show effects with this as a substring in FX profile", - generate_hash("fx_profileFilter") - }, - { - "fx_profileFilterElemCountZero", - "Do not include FX that have a zero element count", - generate_hash("fx_profileFilterElemCountZero") - }, - { - "fx_profileSkip", - "Skip the first n lines in FX profile (to see ones off bottom of screen)", - generate_hash("fx_profileSkip") - }, - { - "fx_profileSort", - "Choose sort criteria for FX profiling", - generate_hash("fx_profileSort") - }, - { - "fx_showLightGridSampleOffset", - "show light grid sample offset in CreateFX mode", - generate_hash("fx_showLightGridSampleOffset") - }, - { - "fx_visMinTraceDist", - "Minimum visibility trace size", - generate_hash("fx_visMinTraceDist") - }, - { - "g_ai", - "Enable AI", - generate_hash("g_ai") - }, - { - "g_allowVote", - "Enable voting on this server", - generate_hash("g_allowVote") - }, - { - "g_atmosFogDistanceScaleReadOnly", - "scale applied to scene distance used for atmospheric fog calculation", - generate_hash("g_atmosFogDistanceScaleReadOnly") - }, - { - "g_atmosFogEnabledReadOnly", - "use atmospheric fog", - generate_hash("g_atmosFogEnabledReadOnly") - }, - { - "g_atmosFogExtinctionStrengthReadOnly", - "scale out scatter contribution of atmospheric fog", - generate_hash("g_atmosFogExtinctionStrengthReadOnly") - }, - { - "g_atmosFogHalfPlaneDistanceReadOnly", - "distance at which atmospheric fog contributes half the pixels color", - generate_hash("g_atmosFogHalfPlaneDistanceReadOnly") - }, - { - "g_atmosFogHazeSpreadReadOnly", - "directionality of haze (1ReadOnly = all forward scatter, 0ReadOnly = all back scatter)", - generate_hash("g_atmosFogHazeSpreadReadOnly") - }, - { - "g_atmosFogHazeStrengthReadOnly", - "portion of atmospheric fog density that is haze (0ReadOnly = all fog, 1ReadOnly = all haze)", - generate_hash("g_atmosFogHazeStrengthReadOnly") - }, - { - "g_atmosFogHeightFogBaseHeightReadOnly", - "height fog is full density at this world height and below", - generate_hash("g_atmosFogHeightFogBaseHeightReadOnly") - }, - { - "g_atmosFogHeightFogEnabledReadOnly", - "use height for atmospheric fog", - generate_hash("g_atmosFogHeightFogEnabledReadOnly") - }, - { - "g_atmosFogHeightFogHalfPlaneDistanceReadOnly", - "at this distance above g_atmosFogHeightFogBaseHeight, height fog density is half", - generate_hash("g_atmosFogHeightFogHalfPlaneDistanceReadOnly") - }, - { - "g_atmosFogInScatterStrengthReadOnly", - "scale in scatter contribution of atmospheric fog", - generate_hash("g_atmosFogInScatterStrengthReadOnly") - }, - { - "g_atmosFogSkyAngularFalloffEnabledReadOnly", - "use angular sky falloff for atmospheric fog", - generate_hash("g_atmosFogSkyAngularFalloffEnabledReadOnly") - }, - { - "g_atmosFogSkyDistanceReadOnly", - "distance used for sky box when applying atmospheric fog", - generate_hash("g_atmosFogSkyDistanceReadOnly") - }, - { - "g_atmosFogSkyFalloffAngleRangeReadOnly", - "sky fog angular falloff angle range sky fog falls off over this range from the start angle", - generate_hash("g_atmosFogSkyFalloffAngleRangeReadOnly") - }, - { - "g_atmosFogSkyFalloffStartAngleReadOnly", - "sky fog angular falloff start angle (full strength fog at this angle)", - generate_hash("g_atmosFogSkyFalloffStartAngleReadOnly") - }, - { - "g_atmosFogStartDistanceReadOnly", - "distance from camera at which fog contribution begins", - generate_hash("g_atmosFogStartDistanceReadOnly") - }, - { - "g_atmosFogSunDirectionReadOnly", - "sun direction used when calculating atmospheric fog", - generate_hash("g_atmosFogSunDirectionReadOnly") - }, - { - "g_banIPs", - "IP addresses to ban from playing", - generate_hash("g_banIPs") - }, - { - "g_clonePlayerMaxVelocity", - "Maximum velocity in each axis of a cloned player\n(for death animations)", - generate_hash("g_clonePlayerMaxVelocity") - }, - { - "g_deadChat", - "Allow dead players to chat with living players", - generate_hash("g_deadChat") - }, - { - "g_dropForwardSpeed", - "Forward speed of a dropped item", - generate_hash("g_dropForwardSpeed") - }, - { - "g_dropHorzSpeedRand", - "Random component of the initial horizontal speed of a dropped item", - generate_hash("g_dropHorzSpeedRand") - }, - { - "g_dropUpSpeedBase", - "Base component of the initial vertical speed of a dropped item", - generate_hash("g_dropUpSpeedBase") - }, - { - "g_dropUpSpeedRand", - "Random component of the initial vertical speed of a dropped item", - generate_hash("g_dropUpSpeedRand") - }, - { - "g_earthquakeEnable", - "Enable camera shake", - generate_hash("g_earthquakeEnable") - }, - { - "g_fogColorIntensityReadOnly", - "HDR fog color intensity that was set in the most recent call to \"setexpfog\"", - generate_hash("g_fogColorIntensityReadOnly") - }, - { - "g_fogColorReadOnly", - "Fog color that was set in the most recent call to \"setexpfog\"", - generate_hash("g_fogColorReadOnly") - }, - { - "g_fogHalfDistReadOnly", - "", - generate_hash("g_fogHalfDistReadOnly") - }, - { - "g_fogMaxOpacityReadOnly", - "Fog max opacity that was set in the most recent call to \"setexpfog\"", - generate_hash("g_fogMaxOpacityReadOnly") - }, - { - "g_fogStartDistReadOnly", - "", - generate_hash("g_fogStartDistReadOnly") - }, - { - "g_friendlyfireDist", - "Maximum range for disabling fire at a friendly", - generate_hash("g_friendlyfireDist") - }, - { - "g_friendlyNameDist", - "Maximum range for seeing a friendly's name", - generate_hash("g_friendlyNameDist") - }, - { - "g_gametype", - "The current game mode", - generate_hash("g_gametype") - }, - { - "g_giveAll", - "Give all weapons", - generate_hash("g_giveAll") - }, - { - "g_hardcore", - "Hardcore?", - generate_hash("g_hardcore") - }, - { - "g_heightFogBaseHeightReadOnly", - "height fog is full density at this world height and below", - generate_hash("g_heightFogBaseHeightReadOnly") - }, - { - "g_heightFogEnabledReadOnly", - "use height for normal/sun fog, set in the most recent call to \"setexpfog\"", - generate_hash("g_heightFogEnabledReadOnly") - }, - { - "g_heightFogHalfPlaneDistanceReadOnly", - "at this distance above g_heightFogBaseHeight, height fog density is half, set in the most recent call to \"setexpfog\"", - generate_hash("g_heightFogHalfPlaneDistanceReadOnly") - }, - { - "g_inactivity", - "Time delay before player is kicked for inactivity", - generate_hash("g_inactivity") - }, - { - "g_keyboarduseholdtime", - "The time to hold down the 'use' button to activate a 'use' command on a keyboard", - generate_hash("g_keyboarduseholdtime") - }, - { - "g_knockback", - "Maximum knockback", - generate_hash("g_knockback") - }, - { - "g_lagged_damage_threshold", - "Threshold (ms) beyond which we will report a damaged lagged client to the tracker leaderboards.", - generate_hash("g_lagged_damage_threshold") - }, - { - "g_listEntity", - "List the entities", - generate_hash("g_listEntity") - }, - { - "g_mantleBlockTimeBuffer", - "Time that the client think is delayed after mantling", - generate_hash("g_mantleBlockTimeBuffer") - }, - { - "g_maxDroppedWeapons", - "Maximum number of dropped weapons", - generate_hash("g_maxDroppedWeapons") - }, - { - "g_minGrenadeDamageSpeed", - "Minimum speed at which getting hit be a grenade will do damage (not the grenade explosion damage)", - generate_hash("g_minGrenadeDamageSpeed") - }, - { - "g_oldschool", - "Oldschool?", - generate_hash("g_oldschool") - }, - { - "g_password", - "Password", - generate_hash("g_password") - }, - { - "g_playerCollisionEjectSpeed", - "Speed at which to push intersecting players away from each other", - generate_hash("g_playerCollisionEjectSpeed") - }, - { - "g_ScoresColor_Allies", - "Allies team color on scoreboard", - generate_hash("g_ScoresColor_Allies") - }, - { - "g_ScoresColor_Axis", - "Axis team color on scoreboard", - generate_hash("g_ScoresColor_Axis") - }, - { - "g_ScoresColor_EnemyTeam", - "Enemy team color on scoreboard", - generate_hash("g_ScoresColor_EnemyTeam") - }, - { - "g_ScoresColor_Free", - "Free Team color on scoreboard", - generate_hash("g_ScoresColor_Free") - }, - { - "g_ScoresColor_MyParty", - "Player team color on scoreboard when in the same party", - generate_hash("g_ScoresColor_MyParty") - }, - { - "g_ScoresColor_MyTeam", - "Player team color on scoreboard", - generate_hash("g_ScoresColor_MyTeam") - }, - { - "g_ScoresColor_Spectator", - "Spectator team color on scoreboard", - generate_hash("g_ScoresColor_Spectator") - }, - { - "g_scriptMainMenu", - "", - generate_hash("g_scriptMainMenu") - }, - { - "g_sunFogBeginFadeAngleReadOnly", - "Angle from the sun direction to start fade away from the sun fog color that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogBeginFadeAngleReadOnly") - }, - { - "g_sunFogColorIntensityReadOnly", - "HDR sun fog color intensity that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogColorIntensityReadOnly") - }, - { - "g_sunFogColorReadOnly", - "Sun fog color that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogColorReadOnly") - }, - { - "g_sunFogDirReadOnly", - "Sun fog direction that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogDirReadOnly") - }, - { - "g_sunFogEnabledReadOnly", - "Sun fog was enabled in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogEnabledReadOnly") - }, - { - "g_sunFogEndFadeAngleReadOnly", - "Angle from the sun direction to end fade away from the sun fog color that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogEndFadeAngleReadOnly") - }, - { - "g_sunFogScaleReadOnly", - "Distance scale in the sun fog direction that was set in the most recent call to \"setexpfog\"", - generate_hash("g_sunFogScaleReadOnly") - }, - { - "g_TeamIcon_Allies", - "Icon name for the allied scores banner", - generate_hash("g_TeamIcon_Allies") - }, - { - "g_TeamIcon_Axis", - "Icon name for the axis scores banner", - generate_hash("g_TeamIcon_Axis") - }, - { - "g_TeamIcon_EnemyAllies", - "Icon name for the allied scores banner", - generate_hash("g_TeamIcon_EnemyAllies") - }, - { - "g_TeamIcon_EnemyAxis", - "Icon name for the axis scores banner when you're on axis.", - generate_hash("g_TeamIcon_EnemyAxis") - }, - { - "g_TeamIcon_Free", - "Icon name for the scores of players with no team", - generate_hash("g_TeamIcon_Free") - }, - { - "g_TeamIcon_MyAllies", - "Icon name for the allied scores banner", - generate_hash("g_TeamIcon_MyAllies") - }, - { - "g_TeamIcon_MyAxis", - "Icon name for the axis scores banner when you're on axis.", - generate_hash("g_TeamIcon_MyAxis") - }, - { - "g_TeamIcon_Spectator", - "Icon name for the scores of players who are spectators", - generate_hash("g_TeamIcon_Spectator") - }, - { - "g_TeamName_Allies", - "Allied team name", - generate_hash("g_TeamName_Allies") - }, - { - "g_TeamName_Axis", - "Axis team name", - generate_hash("g_TeamName_Axis") - }, - { - "g_TeamTitleColor_EnemyTeam", - "Enemy team color for titles", - generate_hash("g_TeamTitleColor_EnemyTeam") - }, - { - "g_TeamTitleColor_MyTeam", - "Player team color for titles", - generate_hash("g_TeamTitleColor_MyTeam") - }, - { - "g_TeamTitleColor_Spectator", - "Spectator team color for titles", - generate_hash("g_TeamTitleColor_Spectator") - }, - { - "g_useholdspawndelay", - "Time in milliseconds that the player is unable to 'use' after spawning", - generate_hash("g_useholdspawndelay") - }, - { - "g_useholdtime", - "Time to hold the 'use' button to activate use on a gamepad", - generate_hash("g_useholdtime") - }, - { - "g_voiceChatTalkingDuration", - "Time after the last talk packet was received that the player is considered by the\nserver to still be talking in milliseconds", - generate_hash("g_voiceChatTalkingDuration") - }, - { - "g_gravity", - "Game gravity in inches per second squared", - generate_hash("g_gravity") - }, - { - "g_speed", - "changes the speed of the player", - generate_hash("g_speed") - }, - { - "gamedate", - "The date compiled", - generate_hash("gamedate") - }, - { - "gamedvr_active", - "Are we allowed to enable GameDVR or not", - generate_hash("gamedvr_active") - }, - { - "gameMode", - "Current gameMode", - generate_hash("gameMode") - }, - { - "gamename", - "The name of the game", - generate_hash("gamename") - }, - { - "glass_angular_vel", - "Sets the range of angular velocities used by new glass pieces", - generate_hash("glass_angular_vel") - }, - { - "glass_beamDamage", - "The amount of damage beam attacks do to glass", - generate_hash("glass_beamDamage") - }, - { - "glass_break", - "Toggle whether or not glass breaks when shot", - generate_hash("glass_break") - }, - { - "glass_crack_pattern_scale", - "The scale applied to the radius used for the crack pattern", - generate_hash("glass_crack_pattern_scale") - }, - { - "glass_damageToDestroy", - "The amount of damage a piece of glass must take to look damaged", - generate_hash("glass_damageToDestroy") - }, - { - "glass_damageToWeaken", - "The amount of damage a piece of glass must take to look damaged", - generate_hash("glass_damageToWeaken") - }, - { - "glass_edge_angle", - "Sets the range of angle deflections used by new glass pieces on a supported edge", - generate_hash("glass_edge_angle") - }, - { - "glass_fall_delay", - "Sets how long a heavy piece supported by a single edge waits before falling, based on glass_fall_ratio", - generate_hash("glass_fall_delay") - }, - { - "glass_fall_gravity", - "Gravity for falling pieces of glass", - generate_hash("glass_fall_gravity") - }, - { - "glass_fall_ratio", - "Ratio of piece area to supporting edge length squared. Below the min, the piece never falls.", - generate_hash("glass_fall_ratio") - }, - { - "glass_fringe_maxcoverage", - "The maximum portion of the original piece of glass that is allowed to remain after the glass shatters", - generate_hash("glass_fringe_maxcoverage") - }, - { - "glass_fringe_maxsize", - "The maximum area for an edge piece of glass when shattering. Pieces larger than this will be broken into smaller ones", - generate_hash("glass_fringe_maxsize") - }, - { - "glass_fx_chance", - "Chance to play an effect on a small piece of glass when it hits the ground", - generate_hash("glass_fx_chance") - }, - { - "glass_hinge_friction", - "Friction used by moving glass pieces when joined like a hinge to a frame", - generate_hash("glass_hinge_friction") - }, - { - "glass_linear_vel", - "Sets the range of linear velocities used by new glass pieces", - generate_hash("glass_linear_vel") - }, - { - "glass_max_pieces_per_frame", - "Maximum number of pieces to create in one frame. This is a guideline and not a hard limit.", - generate_hash("glass_max_pieces_per_frame") - }, - { - "glass_max_shatter_fx_per_frame", - "Maximum number of shatter effects to play in one frame This is a guideline and not a hard limit.", - generate_hash("glass_max_shatter_fx_per_frame") - }, - { - "glass_meleeDamage", - "The amount of damage melee attacks do to glass", - generate_hash("glass_meleeDamage") - }, - { - "glass_physics_chance", - "The chance for a given shard of glass to use physics", - generate_hash("glass_physics_chance") - }, - { - "glass_physics_maxdist", - "The maximum distance of a glass piece from the player to do physics", - generate_hash("glass_physics_maxdist") - }, - { - "glass_radiusDamageMultiplier", - "The amount to scale damage to glass from grenades and other explosions", - generate_hash("glass_radiusDamageMultiplier") - }, - { - "glass_shard_maxsize", - "The maximum area for a flying piece of glass when shattering. Pieces larger than this will be broken into smaller ones", - generate_hash("glass_shard_maxsize") - }, - { - "glass_shattered_scale", - "The scale of the shattered glass material", - generate_hash("glass_shattered_scale") - }, - { - "glass_trace_interval", - "The length of time, in milliseconds, between glass piece traces", - generate_hash("glass_trace_interval") - }, - { - "gpad_button_deadzone", - "Game pad button deadzone threshhold", - generate_hash("gpad_button_deadzone") - }, - { - "gpad_dpadDebounceTime", - "", - generate_hash("gpad_dpadDebounceTime") - }, - { - "gpad_menu_scroll_delay_first", - "Menu scroll key-repeat delay, for the first repeat, in milliseconds", - generate_hash("gpad_menu_scroll_delay_first") - }, - { - "gpad_menu_scroll_delay_rest_accel", - "Menu scroll key-repeat delay acceleration from start to end, for repeats after the first, in milliseconds per repeat", - generate_hash("gpad_menu_scroll_delay_rest_accel") - }, - { - "gpad_menu_scroll_delay_rest_end", - "Menu scroll key-repeat delay end, for repeats after the first, in milliseconds", - generate_hash("gpad_menu_scroll_delay_rest_end") - }, - { - "gpad_menu_scroll_delay_rest_start", - "Menu scroll key-repeat delay start, for repeats after the first, in milliseconds", - generate_hash("gpad_menu_scroll_delay_rest_start") - }, - { - "gpad_stick_deadzone_max", - "Game pad maximum stick deadzone", - generate_hash("gpad_stick_deadzone_max") - }, - { - "gpad_stick_deadzone_min", - "Game pad minimum stick deadzone", - generate_hash("gpad_stick_deadzone_min") - }, - { - "gpad_stick_pressed", - "Game pad stick pressed threshhold", - generate_hash("gpad_stick_pressed") - }, - { - "gpad_stick_pressed_hysteresis", - "Game pad stick pressed no-change-zone around gpad_stick_pressed to prevent bouncing", - generate_hash("gpad_stick_pressed_hysteresis") - }, - { - "groupDownloadInterval", - "Minimum interval to wait before getting new group counts", - generate_hash("groupDownloadInterval") - }, - { - "groupUploadInterval", - "Minimum interval to wait before setting new group counts", - generate_hash("groupUploadInterval") - }, - { - "heli_barrelMaxVelocity", - "", - generate_hash("heli_barrelMaxVelocity") - }, - { - "heli_barrelRotation", - "How much to rotate the turret barrel when a helicopter fires", - generate_hash("heli_barrelRotation") - }, - { - "heli_barrelSlowdown", - "", - generate_hash("heli_barrelSlowdown") - }, - { - "hiDef", - "True if the game video is running in high-def.", - generate_hash("hiDef") - }, - { - "httpnetfs", - "Stream fastfiles from the specified http server", - generate_hash("httpnetfs") - }, - { - "hud_bloodOverlayLerpRate", - "Rate at which blood overlay fades out", - generate_hash("hud_bloodOverlayLerpRate") - }, - { - "hud_deathQuoteFadeTime", - "The time for the death quote to fade", - generate_hash("hud_deathQuoteFadeTime") - }, - { - "hud_drawHud", - "", - generate_hash("hud_drawHud") - }, - { - "hud_enable", - "Enable hud elements", - generate_hash("hud_enable") - }, - { - "hud_fade_ammodisplay", - "The time for the ammo display to fade in seconds", - generate_hash("hud_fade_ammodisplay") - }, - { - "hud_fade_compass", - "The time for the compass to fade in seconds", - generate_hash("hud_fade_compass") - }, - { - "hud_fade_healthbar", - "The time for the health bar to fade in seconds", - generate_hash("hud_fade_healthbar") - }, - { - "hud_fade_offhand", - "The time for the offhand weapons to fade in seconds", - generate_hash("hud_fade_offhand") - }, - { - "hud_fade_sprint", - "The time for the sprint meter to fade in seconds", - generate_hash("hud_fade_sprint") - }, - { - "hud_flash_period_offhand", - "Offhand weapons flash period on changing weapon", - generate_hash("hud_flash_period_offhand") - }, - { - "hud_flash_time_offhand", - "Offhand weapons flash duration on changing weapon", - generate_hash("hud_flash_time_offhand") - }, - { - "hud_health_pulserate_critical", - "The pulse rate of the 'critical' pulse effect", - generate_hash("hud_health_pulserate_critical") - }, - { - "hud_health_pulserate_injured", - "The pulse rate of the 'injured' pulse effect", - generate_hash("hud_health_pulserate_injured") - }, - { - "hud_health_startpulse_critical", - "The health level at which to start the 'injured' pulse effect", - generate_hash("hud_health_startpulse_critical") - }, - { - "hud_health_startpulse_injured", - "The health level at which to start the 'injured' pulse effect", - generate_hash("hud_health_startpulse_injured") - }, - { - "hudElemPausedBrightness", - "Brightness of the hudelems when the game is paused.", - generate_hash("hudElemPausedBrightness") - }, - { - "hudOutlineDuringADS", - "Turn on the HUD outline (green for friendly, red for enemy) when you are pointing at a player while in ADS.", - generate_hash("hudOutlineDuringADS") - }, - { - "igs_config_dw_filename", - "Name of the configuration file on DW Publisher storage.", - generate_hash("igs_config_dw_filename") - }, - { - "igs_sosp", - "Show Original Season Pass", - generate_hash("igs_sosp") - }, - { - "igs_td", - "Show Trial DLC", - generate_hash("igs_td") - }, - { - "igs_version", - "Version id for the In-game store. Set version number to 0, to disable update.", - generate_hash("igs_version") - }, - { - "in_mouse", - "Initialize the mouse drivers", - generate_hash("in_mouse") - }, - { - "inpubliclobby", - "Currently in a public lobby", - generate_hash("inpubliclobby") - }, - { - "intro", - "Intro movie should play", - generate_hash("intro") - }, - { - "inventory_addEntitlementsToLocalInventory", - "bypass the exchange and directly add entitlements to the local cached player inventory.", - generate_hash("inventory_addEntitlementsToLocalInventory") - }, - { - "inventory_enabled", - "enable/disable the inventory feature", - generate_hash("inventory_enabled") - }, - { - "inventory_enableEntitlementDLCScanning", - "enable scanning of entitlement DLC.", - generate_hash("inventory_enableEntitlementDLCScanning") - }, - { - "inventory_enableRevoke", - "Enable revoke on purchases you no longer have rights to.", - generate_hash("inventory_enableRevoke") - }, - { - "inventory_exchangeEnabled", - "enable/disable the 1st party exchange feature", - generate_hash("inventory_exchangeEnabled") - }, - { - "inventory_exchangeMaxConsumablesPerBoot", - "The maximum number of the same consumable that can be added per boot.", - generate_hash("inventory_exchangeMaxConsumablesPerBoot") - }, - { - "inventory_exchangeRetryBaseMS", - "The amount to delay with each subsequent retry as base value to be multiplied by an exponential factor 1000 = (1000, 2000, 4000, 8000 etc.)", - generate_hash("inventory_exchangeRetryBaseMS") - }, - { - "inventory_exchangeRetryByRound", - "enable/disable retry with exponential delay one round of exchanges at a time (1, 2, 3, 1, 2, 3, 1, 2, 3 etc.), vs exponential delay per exchange (1, 1, 1, 2, 2, 2, 3, 3, 3 etc.)", - generate_hash("inventory_exchangeRetryByRound") - }, - { - "inventory_exchangeRetryMax", - "The number of times to retry for each exchange.", - generate_hash("inventory_exchangeRetryMax") - }, - { - "inventory_excludeEntitlementDLCScanning", - "exclude scanning of entitlement DLC (comma separated list of ids to exclude).", - generate_hash("inventory_excludeEntitlementDLCScanning") - }, - { - "inventory_ignoreDWPushNotification_claimAchievement", - "ignore incoming push notifications from DW to signal item update", - generate_hash("inventory_ignoreDWPushNotification_claimAchievement") - }, - { - "inventory_ignoreDWPushNotification_itemUpdate", - "ignore incoming push notifications from DW to signal item update", - generate_hash("inventory_ignoreDWPushNotification_itemUpdate") - }, - { - "inventory_taskDefaultTimeout", - "default timeout for inventory tasks (in seconds)", - generate_hash("inventory_taskDefaultTimeout") - }, - { - "inventory_taskExchangeTimeout", - "default timeout for inventory exchange tasks (in seconds)", - generate_hash("inventory_taskExchangeTimeout") - }, - { - "inventory_taskGetTimeout", - "default timeout for inventory GET tasks (in seconds)", - generate_hash("inventory_taskGetTimeout") - }, - { - "inventory_triggerExchangeOnContentMount", - "trigger an exchange after mounting new content packs", - generate_hash("inventory_triggerExchangeOnContentMount") - }, - { - "inventory_triggerExchangeOnStoreExit", - "trigger an exchange when exiting the store", - generate_hash("inventory_triggerExchangeOnStoreExit") - }, - { - "iotd_active", - "Is the IOTD system enabled", - generate_hash("iotd_active") - }, - { - "iotd_retry", - "Can the IOTD system retry fetching data from Demonware", - generate_hash("iotd_retry") - }, - { - "jump_slowdownEnable", - "Slow player movement after jumping", - generate_hash("jump_slowdownEnable") - }, - { - "jump_height", - "The maximum height of a player\'s jump", - generate_hash("jump_height") - }, - { - "laserDebug", - "Enables the display of various debug info.", - generate_hash("laserDebug") - }, - { - "laserLightRadius", - "undefined", - generate_hash("laserLightRadius") - }, - { - "laserRadius", - "undefined", - generate_hash("laserRadius") - }, - { - "lb_filter", - "Filter applied to the leaderboard display: ('none','friends','facebook_friends')", - generate_hash("lb_filter") - }, - { - "lb_group", - "GroupID applied to the leaderboard display", - generate_hash("lb_group") - }, - { - "lb_maxrows", - "Maximum number of rows to fetch", - generate_hash("lb_maxrows") - }, - { - "lb_minrefresh", - "Minimum time (in seconds) between leaderboard fetches", - generate_hash("lb_minrefresh") - }, - { - "lb_readDelay", - "Delay time between reads(in milliseconds) between leaderboard fetches", - generate_hash("lb_readDelay") - }, - { - "lb_throttle_time", - "Lobby throttling amount", - generate_hash("lb_throttle_time") - }, - { - "lb_times_in_window", - "Lobby throttling window amount", - generate_hash("lb_times_in_window") - }, - { - "lb_window", - "Lobby throttling window", - generate_hash("lb_window") - }, - { - "live_qosec_firstupdatems", - "MS to wait before deciding to early out qos", - generate_hash("live_qosec_firstupdatems") - }, - { - "live_qosec_lastupdatems", - "MS since last update required to early out qos", - generate_hash("live_qosec_lastupdatems") - }, - { - "live_qosec_minpercent", - "Minimum percentage of probe results required before early outing qos", - generate_hash("live_qosec_minpercent") - }, - { - "live_qosec_minprobes", - "Minimum probe results required before early outing qos", - generate_hash("live_qosec_minprobes") - }, - { - "liveanticheatunknowndvar", - "Live Anti Cheat Unknown Dvar", - generate_hash("liveanticheatunknowndvar") - }, - { - "livestreaming_active", - "Are we allowed to enable LiveStreaming or not", - generate_hash("livestreaming_active") - }, - { - "loading_sre_fatal", - "Loading errors prevent level from loading.", - generate_hash("loading_sre_fatal") - }, - { - "lobby_animationSpeed", - "How long each frame of the animation should draw, in milliseconds", - generate_hash("lobby_animationSpeed") - }, - { - "lobby_animationTilesHigh", - "How many animation tiles high is the searching_for_player texture", - generate_hash("lobby_animationTilesHigh") - }, - { - "lobby_animationTilesWide", - "How many animation tiles wide is the searching_for_player texture", - generate_hash("lobby_animationTilesWide") - }, - { - "lobby_numAnimationFrames", - "How many animation tiles are in the searching_for_player texture", - generate_hash("lobby_numAnimationFrames") - }, - { - "lobby_searchingPartyColor", - "The color to show that we're searching for that slot when shown in lobbies", - generate_hash("lobby_searchingPartyColor") - }, - { - "loc_language", - "Language", - generate_hash("loc_language") - }, - { - "loc_translate", - "Enable translations", - generate_hash("loc_translate") - }, - { - "log_host_migration_chance", - "The % chance of host migration results telemetry", - generate_hash("log_host_migration_chance") - }, - { - "log_party_state", - "Log party state updates to Black Box system", - generate_hash("log_party_state") - }, - { - "lowAmmoWarningColor1", - "Color 1 of 2 to oscilate between", - generate_hash("lowAmmoWarningColor1") - }, - { - "lowAmmoWarningColor2", - "Color 2 of 2 to oscilate between", - generate_hash("lowAmmoWarningColor2") - }, - { - "lowAmmoWarningNoAmmoColor1", - "Like lowAmmoWarningColor1, but when no ammo.", - generate_hash("lowAmmoWarningNoAmmoColor1") - }, - { - "lowAmmoWarningNoAmmoColor2", - "lowAmmoWarningColor2, but when no ammo.", - generate_hash("lowAmmoWarningNoAmmoColor2") - }, - { - "lowAmmoWarningNoReloadColor1", - "Like lowAmmoWarningColor1, but when no ammo.", - generate_hash("lowAmmoWarningNoReloadColor1") - }, - { - "lowAmmoWarningNoReloadColor2", - "lowAmmoWarningColor2, but when no ammo.", - generate_hash("lowAmmoWarningNoReloadColor2") - }, - { - "lowAmmoWarningPulseFreq", - "Frequency of the pulse (oscilation between the 2 colors)", - generate_hash("lowAmmoWarningPulseFreq") - }, - { - "lowAmmoWarningPulseMax", - "Min of oscilation range: 0 is color1 and 1.0 is color2. Can be < 0, and the wave will clip at 0.", - generate_hash("lowAmmoWarningPulseMax") - }, - { - "lowAmmoWarningPulseMin", - "Max of oscilation range: 0 is color1 and 1.0 is color2. Can be > 1.0, and the wave will clip at 1.0.", - generate_hash("lowAmmoWarningPulseMin") - }, - { - "lsp_enumertion_max_retry_time", - "Max time that the LSP enumeration can retry", - generate_hash("lsp_enumertion_max_retry_time") - }, - { - "lsp_enumertion_retry_step", - "Step in m/s for the LSP enumeration retry", - generate_hash("lsp_enumertion_retry_step") - }, - { - "lui_demoMode", - "Check if the game is in demo mode.", - generate_hash("lui_demoMode") - }, - { - "lui_FFotDSupportEnabled", - "Enables lui to update itself via the ffotd", - generate_hash("lui_FFotDSupportEnabled") - }, - { - "lui_hud_motion_angle_ease_speed", - "Hud motion ease percentage of degrees per second", - generate_hash("lui_hud_motion_angle_ease_speed") - }, - { - "lui_hud_motion_bob_scale", - "Hud motion bob scale", - generate_hash("lui_hud_motion_bob_scale") - }, - { - "lui_hud_motion_enabled", - "Enable hud motion", - generate_hash("lui_hud_motion_enabled") - }, - { - "lui_hud_motion_perspective", - "value for hud motion perspective transform in pixels", - generate_hash("lui_hud_motion_perspective") - }, - { - "lui_hud_motion_rotation_max", - "Hud motion rotation max", - generate_hash("lui_hud_motion_rotation_max") - }, - { - "lui_hud_motion_rotation_scale", - "Hud motion rotation scale", - generate_hash("lui_hud_motion_rotation_scale") - }, - { - "lui_hud_motion_trans_ease_speed", - "Hud motion ease percentage of pixels per second", - generate_hash("lui_hud_motion_trans_ease_speed") - }, - { - "lui_hud_motion_translation_max", - "Hud motion translation max", - generate_hash("lui_hud_motion_translation_max") - }, - { - "lui_hud_motion_translation_scale", - "Hud motion translation scale", - generate_hash("lui_hud_motion_translation_scale") - }, - { - "lui_loot_duplicateredemption", - "Whether a user can redeem duplicate loot items in the Armory", - generate_hash("lui_loot_duplicateredemption") - }, - { - "LUI_MemErrorsFatal", - "Out of memory errors cause drops when true, reinits the UI system if false", - generate_hash("LUI_MemErrorsFatal") - }, - { - "lui_menuFlowEnabled", - "Enables LUI menu flow", - generate_hash("lui_menuFlowEnabled") - }, - { - "lui_mlg_rules_unlocked", - "Whether MLG rules are unlocked", - generate_hash("lui_mlg_rules_unlocked") - }, - { - "lui_priv_lobby_team", - "Team selected in private match lobby", - generate_hash("lui_priv_lobby_team") - }, - { - "lui_splitscreensignin_menu", - "Enables the LUI splitscreensignin menu", - generate_hash("lui_splitscreensignin_menu") - }, - { - "lui_splitscreenupscaling", - "Force splitscreen upscaling off/on (-1 off, 1 on) -- requires map change", - generate_hash("lui_splitscreenupscaling") - }, - { - "lui_systemlink_menu", - "Enables the LUI systemlink menu", - generate_hash("lui_systemlink_menu") - }, - { - "lui_waitingforgavelmessagesconfirmed", - "", - generate_hash("lui_waitingforgavelmessagesconfirmed") - }, - { - "lui_waitingfornetworktype", - "value is LuiWaitingForNetworkType enum", - generate_hash("lui_waitingfornetworktype") - }, - { - "lui_waitingforonlinedatafetch_controller", - "the controller index that is fetching the online stats data", - generate_hash("lui_waitingforonlinedatafetch_controller") - }, - { - "LUI_WorkerCmdGC", - "Dev-only flag to enable/disable LUI workerCmd GC thread", - generate_hash("LUI_WorkerCmdGC") - }, - { - "lui_xboxlive_menu", - "Enables the LUI xboxlive menu", - generate_hash("lui_xboxlive_menu") - }, - { - "m_filter", - "Allow mouse movement smoothing", - generate_hash("m_filter") - }, - { - "m_forward", - "Forward speed in units per second", - generate_hash("m_forward") - }, - { - "m_pitch", - "Default pitch", - generate_hash("m_pitch") - }, - { - "m_side", - "Sideways motion in units per second", - generate_hash("m_side") - }, - { - "m_yaw", - "Default yaw", - generate_hash("m_yaw") - }, - { - "manifestfs", - "Use a manifest file to read segmented fastfiles", - generate_hash("manifestfs") - }, - { - "mapname", - "The current map name", - generate_hash("mapname") - }, - { - "mapPackMPGroupFourFlags", - "Map pack flags that comprise MP ala carte map pack 1", - generate_hash("mapPackMPGroupFourFlags") - }, - { - "mapPackMPGroupFreeFlags", - "Map pack flags that comprise the free MP ala carte map pack", - generate_hash("mapPackMPGroupFreeFlags") - }, - { - "mapPackMPGroupOneFlags", - "Map pack flags that comprise MP ala carte map pack 1", - generate_hash("mapPackMPGroupOneFlags") - }, - { - "mapPackMPGroupThreeFlags", - "Map pack flags that comprise MP ala carte map pack 1", - generate_hash("mapPackMPGroupThreeFlags") - }, - { - "mapPackMPGroupTwoFlags", - "Map pack flags that comprise MP ala carte map pack 1", - generate_hash("mapPackMPGroupTwoFlags") - }, - { - "marketing_active", - "Are we allowed to enable Marketing Comms or not", - generate_hash("marketing_active") - }, - { - "marketing_refresh_time", - "time in seconds to wait before refreshing marketing messages from demonware", - generate_hash("marketing_refresh_time") - }, - { - "matchdata_active", - "Are match data uploads enabled", - generate_hash("matchdata_active") - }, - { - "matchdata_maxcompressionbuffer", - "Max SP match data compression buffer to use (in bytes)", - generate_hash("matchdata_maxcompressionbuffer") - }, - { - "matchmaking_debug", - "Enable matchmaking debugging information", - generate_hash("matchmaking_debug") - }, - { - "max_ping_threshold_good", - "max ping value to be considered as good", - generate_hash("max_ping_threshold_good") - }, - { - "max_ping_threshold_medium", - "max ping value to be considered as medium", - generate_hash("max_ping_threshold_medium") - }, - { - "max_xp_per_match", - "", - generate_hash("max_xp_per_match") - }, - { - "maxPrestigeOverride", - "Overrides the maximum prestige level, disabled if 0.", - generate_hash("maxPrestigeOverride") - }, - { - "maxVoicePacketsPerSec", - "", - generate_hash("maxVoicePacketsPerSec") - }, - { - "maxVoicePacketsPerSecForServer", - "", - generate_hash("maxVoicePacketsPerSecForServer") - }, - { - "mdsd", - "enable match data stat delta logging?", - generate_hash("mdsd") - }, - { - "melee_debug", - "Turn on debug lines for melee traces", - generate_hash("melee_debug") - }, - { - "mis_cheat", - "Set when level unlock cheat is performed", - generate_hash("mis_cheat") - }, - { - "migration_dvarErrors", - "Whether to check for illegal script dvar changes.", - generate_hash("migration_dvarErrors") - }, - { - "min_wait_for_players", - "", - generate_hash("min_wait_for_players") - }, - { - "missileRemoteFOV", - "Remote missile-cam, FOV to use.", - generate_hash("missileRemoteFOV") - }, - { - "missileRemoteSteerPitchRange", - "Remote-controlled missile allowed up/down range. To keep players from steering missiles above the horizon.", - generate_hash("missileRemoteSteerPitchRange") - }, - { - "missileRemoteSteerPitchRate", - "Remote-controlled missile up/down steering speed.", - generate_hash("missileRemoteSteerPitchRate") - }, - { - "missileRemoteSteerYawRate", - "Remote-controlled missile left/right steering speed.", - generate_hash("missileRemoteSteerYawRate") - }, - { - "mm_aw_onboarding_rank", - "If a player is at or above this rank in AW, she is not considered onboarding", - generate_hash("mm_aw_onboarding_rank") - }, - { - "mm_blops2_onboarding_skill", - "Used to determine onboarding status for Ghosts", - generate_hash("mm_blops2_onboarding_skill") - }, - { - "mm_bucket_option", - "if using bucketing, describes what pools can join with each other", - generate_hash("mm_bucket_option") - }, - { - "mm_country_code", - "country code", - generate_hash("mm_country_code") - }, - { - "mm_ghosts_onboarding_skill", - "Used to determine onboarding status for Ghosts", - generate_hash("mm_ghosts_onboarding_skill") - }, - { - "mm_past_title_stats_source", - "what type of information do we use from the past titles (rank vs kdr, etc)", - generate_hash("mm_past_title_stats_source") - }, - { - "mm_skill_calculation_type", - "", - generate_hash("mm_skill_calculation_type") - }, - { - "mm_skill_enforcement", - "", - generate_hash("mm_skill_enforcement") - }, - { - "mm_skill_lower_bucket", - "lower mm skill bucket", - generate_hash("mm_skill_lower_bucket") - }, - { - "mm_skill_param_delta", - "Delta parameter for Johnson SU distribution curve", - generate_hash("mm_skill_param_delta") - }, - { - "mm_skill_param_gamma", - "Gamma parameter for Johnson SU distribution curve", - generate_hash("mm_skill_param_gamma") - }, - { - "mm_skill_param_lambda", - "Lambda parameter for Johnson SU distribution curve", - generate_hash("mm_skill_param_lambda") - }, - { - "mm_skill_param_xi", - "Xi parameter for Johnson SU distribution curve", - generate_hash("mm_skill_param_xi") - }, - { - "mm_skill_strict_enforcement", - "", - generate_hash("mm_skill_strict_enforcement") - }, - { - "mm_skill_type", - "mm skill type", - generate_hash("mm_skill_type") - }, - { - "mm_skill_upper_bucket", - "upper mm skill bucket", - generate_hash("mm_skill_upper_bucket") - }, - { - "mm_sph_1", - "", - generate_hash("mm_sph_1") - }, - { - "mm_sph_10", - "", - generate_hash("mm_sph_10") - }, - { - "mm_sph_11", - "", - generate_hash("mm_sph_11") - }, - { - "mm_sph_12", - "", - generate_hash("mm_sph_12") - }, - { - "mm_sph_13", - "", - generate_hash("mm_sph_13") - }, - { - "mm_sph_14", - "", - generate_hash("mm_sph_14") - }, - { - "mm_sph_15", - "", - generate_hash("mm_sph_15") - }, - { - "mm_sph_16", - "", - generate_hash("mm_sph_16") - }, - { - "mm_sph_17", - "", - generate_hash("mm_sph_17") - }, - { - "mm_sph_18", - "", - generate_hash("mm_sph_18") - }, - { - "mm_sph_2", - "", - generate_hash("mm_sph_2") - }, - { - "mm_sph_3", - "", - generate_hash("mm_sph_3") - }, - { - "mm_sph_4", - "", - generate_hash("mm_sph_4") - }, - { - "mm_sph_5", - "", - generate_hash("mm_sph_5") - }, - { - "mm_sph_6", - "", - generate_hash("mm_sph_6") - }, - { - "mm_sph_7", - "", - generate_hash("mm_sph_7") - }, - { - "mm_sph_8", - "", - generate_hash("mm_sph_8") - }, - { - "mm_sph_9", - "", - generate_hash("mm_sph_9") - }, - { - "mm_split_population", - "", - generate_hash("mm_split_population") - }, - { - "mm_test_type", - "mm test type", - generate_hash("mm_test_type") - }, - { - "mm_use_onboarding_skill", - "If set, we will for the player's skill to be the lowest available", - generate_hash("mm_use_onboarding_skill") - }, - { - "monkeytoy", - "Restrict console access", - generate_hash("monkeytoy") - }, - { - "motd", - "", - generate_hash("motd") - }, - { - "motd_store_link", - "Add a link to the in-game store in the MOTD popup", - generate_hash("motd_store_link") - }, - { - "motionTrackerBlurDuration", - "The motion blur duration for motion tracker dots", - generate_hash("motionTrackerBlurDuration") - }, - { - "motionTrackerCenterX", - "", - generate_hash("motionTrackerCenterX") - }, - { - "motionTrackerCenterY", - "", - generate_hash("motionTrackerCenterY") - }, - { - "motionTrackerPingFadeTime", - "How long an enemy is visible on the motion tracker after being detected", - generate_hash("motionTrackerPingFadeTime") - }, - { - "motionTrackerPingPitchAddPerEnemy", - "The added percentage of pitch for each additional enemy that is detected (final pitch = base pitch * (1 + enemy count * this))", - generate_hash("motionTrackerPingPitchAddPerEnemy") - }, - { - "motionTrackerPingPitchBase", - "The pitch of the motion tracker sound for a nearby enemy", - generate_hash("motionTrackerPingPitchBase") - }, - { - "motionTrackerPingPitchNearby", - "The pitch of the motion tracker sound for a nearby enemy", - generate_hash("motionTrackerPingPitchNearby") - }, - { - "motionTrackerPingSize", - "The width and height of the motion tracker's enemy indicators as a percentage of motion tracker scale", - generate_hash("motionTrackerPingSize") - }, - { - "msg_field_delta2", - "enable the delta2 serialization.", - generate_hash("msg_field_delta2") - }, - { - "name", - "Player name", - generate_hash("name") - }, - { - "net_authPort", - "UDP port for Steam authentication", - generate_hash("net_authPort") - }, - { - "net_ip", - "Network IP Address", - generate_hash("net_ip") - }, - { - "net_masterServerPort", - "UDP port for Steam server browser", - generate_hash("net_masterServerPort") - }, - { - "net_noudp", - "Disable UDP", - generate_hash("net_noudp") - }, - { - "net_port", - "Network port", - generate_hash("net_port") - }, - { - "net_socksEnabled", - "Enable network sockets", - generate_hash("net_socksEnabled") - }, - { - "net_socksPassword", - "Network socket password", - generate_hash("net_socksPassword") - }, - { - "net_socksPort", - "Network socket port", - generate_hash("net_socksPort") - }, - { - "net_socksServer", - "Network socket server", - generate_hash("net_socksServer") - }, - { - "net_socksUsername", - "Network socket username", - generate_hash("net_socksUsername") - }, - { - "nextmap", - "Next map to play", - generate_hash("nextmap") - }, - { - "nightVisionDisableEffects", - "", - generate_hash("nightVisionDisableEffects") - }, - { - "nightVisionFadeInOutTime", - "How long the fade to/from black lasts when putting on or removing night vision goggles.", - generate_hash("nightVisionFadeInOutTime") - }, - { - "nightVisionPowerOnTime", - "How long the black-to-nightvision fade lasts when turning on the goggles.", - generate_hash("nightVisionPowerOnTime") - }, - { - "num_available_map_packs", - "Number of map packs available for this platform", - generate_hash("num_available_map_packs") - }, - { - "objectiveFontSize", - "Onscreen Objective Pointer - Fontsize of the icon's text.", - generate_hash("objectiveFontSize") - }, - { - "objectiveTextOffsetY", - "Onscreen Objective Pointer - Offset of the icon's text.", - generate_hash("objectiveTextOffsetY") - }, - { - "onlinegame", - "Current game is an online game with stats, custom classes, unlocks", - generate_hash("onlinegame") - }, - { - "overrideNVGModelWithKnife", - "When true, nightvision animations will attach the weapDef's knife model instead of the night vision goggles.", - generate_hash("overrideNVGModelWithKnife") - }, - { - "overtimeTimeLimit", - "", - generate_hash("overtimeTimeLimit") - }, - { - "p2pAuth_allow_steam_p2p", - "Determines if Steam based P2P is allowed if direct connectivity is not possible.", - generate_hash("p2pAuth_allow_steam_p2p") - }, - { - "p2pAuth_disable", - "", - generate_hash("p2pAuth_disable") - }, - { - "paintExplosionRadius", - "The radius of the paint grenade explosion", - generate_hash("paintExplosionRadius") - }, - { - "painVisionLerpOutRate", - "Rate at which pain vision effect lerps out", - generate_hash("painVisionLerpOutRate") - }, - { - "painVisionTriggerHealth", - "Health (0 to 1) that will trigger the pain vision effect", - generate_hash("painVisionTriggerHealth") - }, - { - "party_alternateMapVoteStatus", - "Alternate map vote progress", - generate_hash("party_alternateMapVoteStatus") - }, - { - "party_dlc_only_intended_mappacks", - "When selecting next map for rotation, should any maps not in the intended search be excluded, even if available?", - generate_hash("party_dlc_only_intended_mappacks") - }, - { - "party_firstSubpartyIndex", - "Determines sort order and coloring of parties in lobbies. Randomly set by code. Dvar provided for debugging.", - generate_hash("party_firstSubpartyIndex") - }, - { - "party_followPartyHostOutOfGames", - "Whether we should follow our party host out of a game in progress.", - generate_hash("party_followPartyHostOutOfGames") - }, - { - "party_gamesize", - "Current maximum game size", - generate_hash("party_gamesize") - }, - { - "party_gametype", - "Current gametype", - generate_hash("party_gametype") - }, - { - "party_inactiveHeartbeatPeriod", - "How often to send inactive party keep alive packets in milliseconds.", - generate_hash("party_inactiveHeartbeatPeriod") - }, - { - "party_initial_dlc_search_timer", - "Time until DLC enabled search should show an error dialog suggesting the user consider going to non dlc search", - generate_hash("party_initial_dlc_search_timer") - }, - { - "party_IsLocalClientSelected", - "True if selected player is a local client. Only valid when used with party feeders.", - generate_hash("party_IsLocalClientSelected") - }, - { - "party_kickplayerquestion", - "String to store the question about kicking the selected player", - generate_hash("party_kickplayerquestion") - }, - { - "party_listFocus", - "True when an item in the party list has focus.", - generate_hash("party_listFocus") - }, - { - "party_lobbyPlayerCount", - "Number of players currently in the party/lobby", - generate_hash("party_lobbyPlayerCount") - }, - { - "party_mapname", - "Current map name", - generate_hash("party_mapname") - }, - { - "party_mapvote_entrya_mapname", - "Primary map vote entry name", - generate_hash("party_mapvote_entrya_mapname") - }, - { - "party_mapvote_entryb_mapname", - "Alternate map vote entry name", - generate_hash("party_mapvote_entryb_mapname") - }, - { - "party_matchedPlayerCount", - "Number of matched players before revealing their true names", - generate_hash("party_matchedPlayerCount") - }, - { - "party_maxplayers", - "Maximum number of players in a party", - generate_hash("party_maxplayers") - }, - { - "party_maxPrivatePartyPlayers", - "Max number of players allowed in a private party.", - generate_hash("party_maxPrivatePartyPlayers") - }, - { - "party_maxTeamDiff", - "Maximum difference allowed between teams before starting a match", - generate_hash("party_maxTeamDiff") - }, - { - "party_membersMissingMapPack", - "Whether any party member is missing one of the enabled map packs. Only valid after running partyUpdateMissingMapPackDvar", - generate_hash("party_membersMissingMapPack") - }, - { - "party_minLobbyTime", - "Minimum time (in seconds) for a lobby to be open before auto starting a match", - generate_hash("party_minLobbyTime") - }, - { - "party_minplayers", - "Minimum number of players in a party", - generate_hash("party_minplayers") - }, - { - "party_nextMapVoteStatus", - "Next map vote progress", - generate_hash("party_nextMapVoteStatus") - }, - { - "party_partyPlayerCount", - "Number of players currently in the party/lobby", - generate_hash("party_partyPlayerCount") - }, - { - "party_partyPlayerCountNum", - "Number of players currently in the party/lobby", - generate_hash("party_partyPlayerCountNum") - }, - { - "party_playersCoop", - "True if the current playlist places all players on the allies team", - generate_hash("party_playersCoop") - }, - { - "party_playervisible", - "Whether selected player in party is showing true info or not. Only valid when used with party feeders.", - generate_hash("party_playervisible") - }, - { - "party_randomMapVoteStatus", - "Random map vote progress", - generate_hash("party_randomMapVoteStatus") - }, - { - "party_resume_dlc_search_timer", - "Time until DLC enabled search should show an error dialog suggesting the user consider going to non dlc search", - generate_hash("party_resume_dlc_search_timer") - }, - { - "party_search_for_dlc_content", - "search for DLC enabled games else standard maps only will be used", - generate_hash("party_search_for_dlc_content") - }, - { - "party_selectedIndex", - "Current selected player index in the feeder.", - generate_hash("party_selectedIndex") - }, - { - "party_selectedIndexChangedTime", - "Time stamp in milliseconds when the selected index last changed.", - generate_hash("party_selectedIndexChangedTime") - }, - { - "party_statusString", - "Party Status (localized )", - generate_hash("party_statusString") - }, - { - "party_teambased", - "True if the current playlist is team based", - generate_hash("party_teambased") - }, - { - "party_teamsVisible", - "teams are visible in UI", - generate_hash("party_teamsVisible") - }, - { - "party_timer", - "Time until game begins in seconds, for UI display", - generate_hash("party_timer") - }, - { - "partyChatDisallowed", - "Whether to disallow ps4 Live Party Chat", - generate_hash("partyChatDisallowed") - }, - { - "partymigrate_broadcast_interval", - "Time between telling people who the new host is when migrating lobby", - generate_hash("partymigrate_broadcast_interval") - }, - { - "partymigrate_cpuBonusPing", - "The ping rewarded to a CPU meeting the bonus threshold when scoring hosts.", - generate_hash("partymigrate_cpuBonusPing") - }, - { - "partymigrate_cpuBonusThreshold", - "The required excess %CPU speed to get a bonus when scoring hosts.", - generate_hash("partymigrate_cpuBonusThreshold") - }, - { - "partymigrate_logResults", - "Instrumentation - Whether to log the best host calculation results. 0 is disabled, 1 for games, 2 for parties, 3 for both.", - generate_hash("partymigrate_logResults") - }, - { - "partymigrate_makeHostTimeout", - "Time before giving up on makeHost requests", - generate_hash("partymigrate_makeHostTimeout") - }, - { - "partymigrate_pingtest_active", - "Whether to do a ping test when lobby migrating", - generate_hash("partymigrate_pingtest_active") - }, - { - "partymigrate_pingtest_filterThreshold", - "Acceptable ping difference from best ping host for host selection (ms)", - generate_hash("partymigrate_pingtest_filterThreshold") - }, - { - "partymigrate_pingtest_minThreshold", - "Minimum meaningful ping delta for host selection (ms)", - generate_hash("partymigrate_pingtest_minThreshold") - }, - { - "partymigrate_pingtest_retry", - "Time between ping tests when migrating lobby", - generate_hash("partymigrate_pingtest_retry") - }, - { - "partymigrate_pingtest_timeout", - "Time to give up on ping tests when migrating lobby", - generate_hash("partymigrate_pingtest_timeout") - }, - { - "partymigrate_preferSameHost", - "When possible, prefer keeping the same host on migrations", - generate_hash("partymigrate_preferSameHost") - }, - { - "partymigrate_selectiontime", - "Time before requiring a host selection when migrating lobby", - generate_hash("partymigrate_selectiontime") - }, - { - "partymigrate_timeout", - "Time before giving up on lobby migration if we hear nothing", - generate_hash("partymigrate_timeout") - }, - { - "partymigrate_timeoutmax", - "Time before giving up on lobby migration if we hear nothing", - generate_hash("partymigrate_timeoutmax") - }, - { - "partymigrate_uploadtest_minThreshold", - "Minimum meaningful upload bandwidth delta for host selection (bps)", - generate_hash("partymigrate_uploadtest_minThreshold") - }, - { - "password", - "", - generate_hash("password") - }, - { - "perk_armorPiercingDamage", - "", - generate_hash("perk_armorPiercingDamage") - }, - { - "perk_blastShieldScale", - "", - generate_hash("perk_blastShieldScale") - }, - { - "perk_blastShieldScale_HC", - "", - generate_hash("perk_blastShieldScale_HC") - }, - { - "perk_bulletPenetrationMultiplier", - "Multiplier for extra bullet penetration", - generate_hash("perk_bulletPenetrationMultiplier") - }, - { - "perk_extendedMagsMGAmmo", - "Number of extra bullets per clip for machine gun weapons with the extended mags perk.", - generate_hash("perk_extendedMagsMGAmmo") - }, - { - "perk_extendedMagsPistolAmmo", - "Number of extra bullets per clip for pistol weapons with the extended mags perk.", - generate_hash("perk_extendedMagsPistolAmmo") - }, - { - "perk_extendedMagsRifleAmmo", - "Number of extra bullets per clip for rifle weapons with the extended mags perk.", - generate_hash("perk_extendedMagsRifleAmmo") - }, - { - "perk_extendedMagsSMGAmmo", - "Number of extra bullets per clip for sub machine gun weapons with the extended mags perk.", - generate_hash("perk_extendedMagsSMGAmmo") - }, - { - "perk_extendedMagsSpreadAmmo", - "Number of extra bullets per clip for spread weapons with the extended mags perk.", - generate_hash("perk_extendedMagsSpreadAmmo") - }, - { - "perk_extraBreath", - "Number of extra seconds a player can hold his breath", - generate_hash("perk_extraBreath") - }, - { - "perk_fastClimb", - "Scales the ladder climb time", - generate_hash("perk_fastClimb") - }, - { - "perk_fastRegenRate", - "", - generate_hash("perk_fastRegenRate") - }, - { - "perk_fastRegenWaitMS", - "", - generate_hash("perk_fastRegenWaitMS") - }, - { - "perk_fastSnipeScale", - "Scales the recovery speed of the view kick when using a sniper.", - generate_hash("perk_fastSnipeScale") - }, - { - "perk_footstepVolumeAlly", - "", - generate_hash("perk_footstepVolumeAlly") - }, - { - "perk_footstepVolumeEnemy", - "", - generate_hash("perk_footstepVolumeEnemy") - }, - { - "perk_footstepVolumePlayer", - "", - generate_hash("perk_footstepVolumePlayer") - }, - { - "perk_footstepVolumeSelectiveHearingMin", - "", - generate_hash("perk_footstepVolumeSelectiveHearingMin") - }, - { - "perk_improvedExtraBreath", - "Number of extra seconds a player can hold his breath", - generate_hash("perk_improvedExtraBreath") - }, - { - "perk_lightWeightViewBobScale", - "Scale for first person view movement while lightweight.", - generate_hash("perk_lightWeightViewBobScale") - }, - { - "perk_numExtraLethal", - "Number of extra lethal grenades", - generate_hash("perk_numExtraLethal") - }, - { - "perk_numExtraTactical", - "Number of extra tactical grenades", - generate_hash("perk_numExtraTactical") - }, - { - "perk_parabolicAngle", - "Eavesdrop perk's effective FOV angle", - generate_hash("perk_parabolicAngle") - }, - { - "perk_parabolicIcon", - "Eavesdrop icon to use when displaying eavesdropped voice chats", - generate_hash("perk_parabolicIcon") - }, - { - "perk_parabolicRadius", - "Eavesdrop perk's effective radius", - generate_hash("perk_parabolicRadius") - }, - { - "perk_quickDrawSpeedScale", - "Scales the 'Hip to ADS' transition speed.", - generate_hash("perk_quickDrawSpeedScale") - }, - { - "perk_quickDrawSpeedScaleSniper", - "Scales the 'Hip to ADS' transition speed.", - generate_hash("perk_quickDrawSpeedScaleSniper") - }, - { - "perk_scavengerMode", - "", - generate_hash("perk_scavengerMode") - }, - { - "perk_sprintMultiplier", - "Multiplier for player_sprinttime", - generate_hash("perk_sprintMultiplier") - }, - { - "perk_sprintRecoveryMultiplierActual", - "", - generate_hash("perk_sprintRecoveryMultiplierActual") - }, - { - "perk_sprintRecoveryMultiplierVisual", - "", - generate_hash("perk_sprintRecoveryMultiplierVisual") - }, - { - "perk_weapRateMultiplier", - "Percentage of weapon firing rate to use", - generate_hash("perk_weapRateMultiplier") - }, - { - "perk_weapReloadMultiplier", - "Percentage of weapon reload time to use", - generate_hash("perk_weapReloadMultiplier") - }, - { - "perk_weapSpreadMultiplier", - "Percentage of weapon spread to use", - generate_hash("perk_weapSpreadMultiplier") - }, - { - "phys_autoDisableLinear", - "A body must have linear velocity less than this to be considered idle.", - generate_hash("phys_autoDisableLinear") - }, - { - "phys_autoDisableTime", - "The amount of time a body must be idle for it to go to sleep.", - generate_hash("phys_autoDisableTime") - }, - { - "phys_bulletSpinScale", - "Scale of the effective offset from the center of mass for the bullet impacts.", - generate_hash("phys_bulletSpinScale") - }, - { - "phys_bulletUpBias", - "Up Bias for the direction of the bullet impact.", - generate_hash("phys_bulletUpBias") - }, - { - "phys_dragAngular", - "The amount of angular drag, applied globally", - generate_hash("phys_dragAngular") - }, - { - "phys_dragLinear", - "The amount of linear drag, applied globally", - generate_hash("phys_dragLinear") - }, - { - "phys_gravity", - "Physics gravity in units/sec^2.", - generate_hash("phys_gravity") - }, - { - "phys_gravity_ragdoll", - "Physics gravity used by ragdolls in units/sec^2.", - generate_hash("phys_gravity_ragdoll") - }, - { - "phys_gravityChangeWakeupRadius", - "The radius around the player within which objects get awakened when gravity changes", - generate_hash("phys_gravityChangeWakeupRadius") - }, - { - "phys_jitterMaxMass", - "Maximum mass to jitter - jitter will fall off up to this mass", - generate_hash("phys_jitterMaxMass") - }, - { - "physVeh_explodeForce", - "The force applied to physics vehicles due to explosions", - generate_hash("physVeh_explodeForce") - }, - { - "physVeh_explodeSpinScale", - "The max (random) offset from the center of mass at which splash damage applies its force", - generate_hash("physVeh_explodeSpinScale") - }, - { - "physVeh_jump", - "Set to 1 to make a vehicle jump.", - generate_hash("physVeh_jump") - }, - { - "physVeh_minContactImpulse", - "The minimum impulse needed to register a contact notification", - generate_hash("physVeh_minContactImpulse") - }, - { - "physVeh_minImpactMomentum", - "The minimum collision momentum needed to register an impact", - generate_hash("physVeh_minImpactMomentum") - }, - { - "physVeh_StepsPerFrame", - "The number of physics timesteps that the server frame will be divided into.", - generate_hash("physVeh_StepsPerFrame") - }, - { - "pickupPrints", - "Print a message to the game window when picking up ammo, etc.", - generate_hash("pickupPrints") - }, - { - "player_breath_snd_delay", - "The delay before playing the breathe in sound", - generate_hash("player_breath_snd_delay") - }, - { - "player_breath_snd_lerp", - "The interpolation rate for the player hold breath sound", - generate_hash("player_breath_snd_lerp") - }, - { - "player_current_floor", - "", - generate_hash("player_current_floor") - }, - { - "player_MGUseRadius", - "The radius within which a player can mount a machine gun", - generate_hash("player_MGUseRadius") - }, - { - "player_stunWhiteFlash", - "If enabled, player's screens will flash white when they are stunned.", - generate_hash("player_stunWhiteFlash") - }, - { - "player_throwbackInnerRadius", - "The radius to a live grenade player must be within initially to do a throwback", - generate_hash("player_throwbackInnerRadius") - }, - { - "player_throwbackOuterRadius", - "The radius player is allow to throwback a grenade once the player has been in the inner radius", - generate_hash("player_throwbackOuterRadius") - }, - { - "player_useRadius", - "The radius within which a player can use things", - generate_hash("player_useRadius") - }, - { - "playercard_cache_download_max_retry_time", - "Max time that the player cache download can retry", - generate_hash("playercard_cache_download_max_retry_time") - }, - { - "playercard_cache_download_retry_step", - "Step in m/s for the player cache download retry", - generate_hash("playercard_cache_download_retry_step") - }, - { - "playercard_cache_upload_max_retry_time", - "Max time that the player cache upload can retry", - generate_hash("playercard_cache_upload_max_retry_time") - }, - { - "playercard_cache_upload_retry_step", - "Step in m/s for the player cache upload retry", - generate_hash("playercard_cache_upload_retry_step") - }, - { - "playercard_cache_validity_life", - "The life of a cached gamercard (it can be re-downloaded after this)", - generate_hash("playercard_cache_validity_life") - }, - { - "playerPositionRecordSampleTime", - "How often to sample player positions and save them into match data.", - generate_hash("playerPositionRecordSampleTime") - }, - { - "playlist", - "The playlist number", - generate_hash("playlist") - }, - { - "playlistAggrFilename", - "Aggregated playlist filename", - generate_hash("playlistAggrFilename") - }, - { - "playlistFilename", - "Playlist filename", - generate_hash("playlistFilename") - }, - { - "playListUpdateCheckMinutes", - "Minutes to wait between checking for updated playlist.", - generate_hash("playListUpdateCheckMinutes") - }, - { - "prestige_shop_active", - "Are we allowed to show the Prestige Shop or not", - generate_hash("prestige_shop_active") - }, - { - "prestige30EasterEggEnabled", - "Enables the easter egg for prestige 30 if 1, disabled if 0.", - generate_hash("prestige30EasterEggEnabled") - }, - { - "privateMatch_joinPassword", - "", - generate_hash("privateMatch_joinPassword") - }, - { - "privateMatch_serverPassword", - "", - generate_hash("privateMatch_serverPassword") - }, - { - "profileMenuOption_blacklevel", - "", - generate_hash("profileMenuOption_blacklevel") - }, - { - "profileMenuOption_offensiveContentMode", - "Mode of the offensive content warning at startup - 0, skip and turn on; 1, skip and turn off; 2, ask user", - generate_hash("profileMenuOption_offensiveContentMode") - }, - { - "profileMenuOption_safeAreaHorz", - "", - generate_hash("profileMenuOption_safeAreaHorz") - }, - { - "profileMenuOption_safeAreaVert", - "", - generate_hash("profileMenuOption_safeAreaVert") - }, - { - "profileMenuOption_volume", - "", - generate_hash("profileMenuOption_volume") - }, - { - "protocol", - "Protocol version", - generate_hash("protocol") - }, - { - "pt_AliensReadyUpPrivateInUse", - "Do we use the co-op Ready Up feature in public lobbies?", - generate_hash("pt_AliensReadyUpPrivateInUse") - }, - { - "pt_AliensReadyUpPublicInUse", - "Do we use the co-op Ready Up feature in public lobbies?", - generate_hash("pt_AliensReadyUpPublicInUse") - }, - { - "pt_AliensReadyUpPublicStartTimerLength", - "co-op Ready Up start timer length in seconds", - generate_hash("pt_AliensReadyUpPublicStartTimerLength") - }, - { - "pt_allMembersDoQoS", - "Whether to send search results to all party/lobby members to get QoS data", - generate_hash("pt_allMembersDoQoS") - }, - { - "pt_backoutOnClientPresence", - "Whether the host should backout the party on client presence. 0=fully disabled, 1=out of game only, 2=in-game also", - generate_hash("pt_backoutOnClientPresence") - }, - { - "pt_connectAttempts", - "Connect timeout when joining another game/party, per attempt", - generate_hash("pt_connectAttempts") - }, - { - "pt_connectTimeout", - "Connect timeout when joining another game/party, per attempt", - generate_hash("pt_connectTimeout") - }, - { - "pt_gameStartTimerLength", - "Time in seconds before a game start once enough party members are ready", - generate_hash("pt_gameStartTimerLength") - }, - { - "pt_logHostSelectionChance", - "Sets the probability that we log our host selection results", - generate_hash("pt_logHostSelectionChance") - }, - { - "pt_memberTimeout", - "Client timeout time in the general case", - generate_hash("pt_memberTimeout") - }, - { - "pt_migrateBeforeAdvertise", - "Whether lobbies made by private parties should migrate host before publishing", - generate_hash("pt_migrateBeforeAdvertise") - }, - { - "pt_migrationBandwidthBonusPing", - "The ping rewarded to the bonus bandwidth threshold when scoring hosts.", - generate_hash("pt_migrationBandwidthBonusPing") - }, - { - "pt_migrationBandwidthBonusThreshold", - "The required excess % upload bandwidth to get a bonus when scoring hosts.", - generate_hash("pt_migrationBandwidthBonusThreshold") - }, - { - "pt_migrationCPUWeight", - "How important CPU speed is when selecting a new host", - generate_hash("pt_migrationCPUWeight") - }, - { - "pt_migrationNotInstalledWeight", - "How important not being done installing is when selecting a new host", - generate_hash("pt_migrationNotInstalledWeight") - }, - { - "pt_migrationPingBad", - "", - generate_hash("pt_migrationPingBad") - }, - { - "pt_migrationPingWeight", - "", - generate_hash("pt_migrationPingWeight") - }, - { - "pt_migrationQuitsBad", - "", - generate_hash("pt_migrationQuitsBad") - }, - { - "pt_migrationQuitsWeight", - "", - generate_hash("pt_migrationQuitsWeight") - }, - { - "pt_migrationRAMWeight", - "How important it is to have the minimum amount of RAM when selecting a new host", - generate_hash("pt_migrationRAMWeight") - }, - { - "pt_migrationThreshold", - "Minimum amount which another client must be better than the current host to trigger a migration", - generate_hash("pt_migrationThreshold") - }, - { - "pt_migrationUploadBad", - "", - generate_hash("pt_migrationUploadBad") - }, - { - "pt_migrationUploadWeight", - "", - generate_hash("pt_migrationUploadWeight") - }, - { - "pt_migrationWifiPenalty", - "How important Wifi is when selecting a new host", - generate_hash("pt_migrationWifiPenalty") - }, - { - "pt_pregameStartTimerLength", - "Time in seconds before showing and starting the game start timer", - generate_hash("pt_pregameStartTimerLength") - }, - { - "pt_rejoin", - "Enable defensive rejoin command", - generate_hash("pt_rejoin") - }, - { - "pt_reservedAnonymousSlotTime", - "Time in milliseconds that ANONYMOUS slots will be reserved.", - generate_hash("pt_reservedAnonymousSlotTime") - }, - { - "pt_reservedCommittedSlotTime", - "Time in milliseconds that COMMITTED slots will be reserved", - generate_hash("pt_reservedCommittedSlotTime") - }, - { - "pt_reservedJoiningSlotTime", - "Time in milliseconds that JOINING slots will be reserved", - generate_hash("pt_reservedJoiningSlotTime") - }, - { - "pt_searchConnectAttempts", - "Connect timeout when joining another game/party, per attempt", - generate_hash("pt_searchConnectAttempts") - }, - { - "pt_searchPauseTime", - "Minimum amount of time to pause between searches", - generate_hash("pt_searchPauseTime") - }, - { - "pt_searchRandomDelay", - "Time period over which the search timers will get randomly delayed.", - generate_hash("pt_searchRandomDelay") - }, - { - "pt_searchResultsLifetime", - "Time at which we consider the search results stale", - generate_hash("pt_searchResultsLifetime") - }, - { - "pt_searchResultsMin", - "Minimum amount of time that has to pass before we'll search again for matches", - generate_hash("pt_searchResultsMin") - }, - { - "pt_stillConnectingWaitTime", - "Amount of time to wait for someone to finish connecting before searching for lobbies to merge with", - generate_hash("pt_stillConnectingWaitTime") - }, - { - "pt_useMigrationWeights", - "Killswitch to turn on or off the host selection by weights", - generate_hash("pt_useMigrationWeights") - }, - { - "publisherFileFetchTimeout", - "default timeout for publisher files FETCH tasks (in seconds)", - generate_hash("publisherFileFetchTimeout") - }, - { - "r_adaptiveSubdiv", - "Enables screen space Catmull Clark adaptive tessellation. If disabled, models tessellate to their designed subdivision level.", - generate_hash("r_adaptiveSubdiv") - }, - { - "r_adaptiveSubdivBaseFactor", - "Screen space Catmull Clark adaptive tessellation factor for the base model. Smaller values mean more tessellation.", - generate_hash("r_adaptiveSubdivBaseFactor") - }, - { - "r_adaptiveSubdivPatchFactor", - "Screen space Catmull Clark adaptive tessellation factor for the base model. Smaller values mean more tessellation.", - generate_hash("r_adaptiveSubdivPatchFactor") - }, - { - "r_allCells", - "Draw all cells. Most useful for seeing if portals or cells are hiding things they should not..", - generate_hash("r_allCells") - }, - { - "r_amdGPU", - "At least on AMD GPU used for rendering.", - generate_hash("r_amdGPU") - }, - { - "r_aoBlurSharpness", - "Controls the tolerance for depth discontinuities during the bilateral blur step. Larger values reduce the depth tolerance and effectively sharpen more edges.", - generate_hash("r_aoBlurSharpness") - }, - { - "r_aoBlurStep", - "Step scale applied to sample offsets during the bilateral blur. A value of 1 results in a normal gaussian blur. Increasing it to 2 or 3 makes the filter larger but causes fine dithering patterns.", - generate_hash("r_aoBlurStep") - }, - { - "r_aoDiminish", - "Decrease the effect of occlusion on brighter colors", - generate_hash("r_aoDiminish") - }, - { - "r_aoPower", - "Power curve applied to AO factor", - generate_hash("r_aoPower") - }, - { - "r_aoStrength", - "Strength of Ambient Occlusion effect", - generate_hash("r_aoStrength") - }, - { - "r_aoUseTweaks", - "Use r_ao* dvars instead of the current light set values for AO common params", - generate_hash("r_aoUseTweaks") - }, - { - "r_artUseTweaks", - "Tells the game that art tweaks is enabled and script is in control (as opposed to ColorEd).", - generate_hash("r_artUseTweaks") - }, - { - "r_aspectRatio", - "Screen aspect ratio. Most widescreen monitors are 16:10 instead of 16:9.", - generate_hash("r_aspectRatio") - }, - { - "r_asyncCompute", - "Enables scheduling GPU compute shader work prior to the graphics frame, improving overlap.", - generate_hash("r_asyncCompute") - }, - { - "r_atlasAnimFPS", - "Speed to animate atlased 2d materials", - generate_hash("r_atlasAnimFPS") - }, - { - "r_autopriority", - "Automatically set the priority of the windows process when the game is minimized", - generate_hash("r_autopriority") - }, - { - "r_balanceLightmapOpaqueLists", - "Split lightmap opaque into multiple draw lists.", - generate_hash("r_balanceLightmapOpaqueLists") - }, - { - "r_blacklevel", - "Black level (negative brightens output)", - generate_hash("r_blacklevel") - }, - { - "r_blur", - "Dev tweak to blur the screen", - generate_hash("r_blur") - }, - { - "r_blurdstGaussianBlurLevel", - "MIP level to start gaussian blur at", - generate_hash("r_blurdstGaussianBlurLevel") - }, - { - "r_blurdstGaussianBlurRadius", - "Amount to gaussian blur blur distortion render target", - generate_hash("r_blurdstGaussianBlurRadius") - }, - { - "r_brightness", - "Brightness adjustment", - generate_hash("r_brightness") - }, - { - "r_cacheModelLighting", - "Speed up model lighting by caching previous results", - generate_hash("r_cacheModelLighting") - }, - { - "r_cacheSModelLighting", - "Speed up static model lighting by caching previous results", - generate_hash("r_cacheSModelLighting") - }, - { - "r_charLightAmbient", - "", - generate_hash("r_charLightAmbient") - }, - { - "r_clampLodScale", - "Clamps the amount that the engine can adjust the LOD distance. 0 the engine can fully adjust. 1 the engine cannot adjust it at all. 0.5 the maximum the engine can adjust the LOD distance is 50% or the default.", - generate_hash("r_clampLodScale") - }, - { - "r_clear", - "Controls how the color buffer is cleared", - generate_hash("r_clear") - }, - { - "r_clearColor", - "Color to clear the screen to when clearing the frame buffer", - generate_hash("r_clearColor") - }, - { - "r_clearColor2", - "Color to clear every second frame to (for use during development)", - generate_hash("r_clearColor2") - }, - { - "r_clutCompositeVisionSet", - "Composite clut with vision set.", - generate_hash("r_clutCompositeVisionSet") - }, - { - "r_cmdbuf_worker", - "Process command buffer in a separate thread", - generate_hash("r_cmdbuf_worker") - }, - { - "r_colorGradingEnable", - "Enable color grading.", - generate_hash("r_colorGradingEnable") - }, - { - "r_colorMap", - "Replace all color maps with pure black or pure white", - generate_hash("r_colorMap") - }, - { - "r_colorScaleUseTweaks", - "Override color scale LightSet settings with tweak dvar values. (MP)", - generate_hash("r_colorScaleUseTweaks") - }, - { - "r_combinePostOpaqueFx", - "", - generate_hash("r_combinePostOpaqueFx") - }, - { - "r_contrast", - "Contrast adjustment", - generate_hash("r_contrast") - }, - { - "r_darkBlur", - "Apply blur (decrease of visual acuity) when dark", - generate_hash("r_darkBlur") - }, - { - "r_darkBlurPower", - "Power curve of blurring (decrease of visual acuity) when dark", - generate_hash("r_darkBlurPower") - }, - { - "r_darkBlurRadius", - "Radius of blurring (decrease of visual acuity) when dark", - generate_hash("r_darkBlurRadius") - }, - { - "r_darkColor", - "Reduce color sensitivity when dark", - generate_hash("r_darkColor") - }, - { - "r_darkColorPower", - "Power curve of color sensitivity when dark", - generate_hash("r_darkColorPower") - }, - { - "r_debugLineWidth", - "Width of server side debug lines", - generate_hash("r_debugLineWidth") - }, - { - "r_defaultPatchCount", - "Patches per thread group for all other surfaces.", - generate_hash("r_defaultPatchCount") - }, - { - "r_depthPrepass", - "Enable depth prepass for various geometries", - generate_hash("r_depthPrepass") - }, - { - "r_depthSortEnable", - "Enable sorting of transparent surfaces.", - generate_hash("r_depthSortEnable") - }, - { - "r_depthSortRange", - "Range to consider depth sort,", - generate_hash("r_depthSortRange") - }, - { - "r_desaturation", - "Desaturation adjustment", - generate_hash("r_desaturation") - }, - { - "r_detailMap", - "Replace all detail maps with an image that effectively disables them", - generate_hash("r_detailMap") - }, - { - "r_diffuseColorScale", - "Globally scale the diffuse color of all point lights", - generate_hash("r_diffuseColorScale") - }, - { - "r_displacementMap", - "Replace all displacement maps with an image that effectively disables them", - generate_hash("r_displacementMap") - }, - { - "r_displacementPatchCount", - "Patches per thread group for displacement surfaces.", - generate_hash("r_displacementPatchCount") - }, - { - "r_distortion", - "Enable distortion", - generate_hash("r_distortion") - }, - { - "r_distortion_script_force_off", - "Force distortion off in script", - generate_hash("r_distortion_script_force_off") - }, - { - "r_dlightLimit", - "Maximum number of dynamic lights drawn simultaneously", - generate_hash("r_dlightLimit") - }, - { - "r_dof_bias", - "Depth of field bias as a power function (like gamma); less than 1 is sharper", - generate_hash("r_dof_bias") - }, - { - "r_dof_enable", - "Enable the depth of field effect", - generate_hash("r_dof_enable") - }, - { - "r_dof_farBlur", - "", - generate_hash("r_dof_farBlur") - }, - { - "r_dof_farEnd", - "Depth of field far end distance, in inches", - generate_hash("r_dof_farEnd") - }, - { - "r_dof_farStart", - "Depth of field far start distance, in inches", - generate_hash("r_dof_farStart") - }, - { - "r_dof_nearBlur", - "", - generate_hash("r_dof_nearBlur") - }, - { - "r_dof_nearEnd", - "Depth of field near end distance, in inches", - generate_hash("r_dof_nearEnd") - }, - { - "r_dof_nearStart", - "Depth of field near start distance, in inches", - generate_hash("r_dof_nearStart") - }, - { - "r_dof_physical_adsFocusSpeed", - "ADS focus speed (focus dist. far to near, focus dist. near to far, aperture opening, aperture closing)", - generate_hash("r_dof_physical_adsFocusSpeed") - }, - { - "r_dof_physical_adsMaxFstop", - "ADS maximum f-stop (optimal aperture and focus distance are automatically calculated for this mode)", - generate_hash("r_dof_physical_adsMaxFstop") - }, - { - "r_dof_physical_adsMinFstop", - "ADS minimum f-stop (optimal aperture and focus distance are automatically calculated for this mode)", - generate_hash("r_dof_physical_adsMinFstop") - }, - { - "r_dof_physical_bokehEnable", - "Enable the bokeh depth of field effect", - generate_hash("r_dof_physical_bokehEnable") - }, - { - "r_dof_physical_bokehPreset", - "Changes dof sampling quality", - generate_hash("r_dof_physical_bokehPreset") - }, - { - "r_dof_physical_bokehRotation", - "Bokeh shape rotation in degrees (hexagonal and octogonal only)", - generate_hash("r_dof_physical_bokehRotation") - }, - { - "r_dof_physical_bokehShape", - "Changes the bokeh shape", - generate_hash("r_dof_physical_bokehShape") - }, - { - "r_dof_physical_bokehSharpness", - "Bokeh shape sharpness, trades sharpness for stability (circular only)", - generate_hash("r_dof_physical_bokehSharpness") - }, - { - "r_dof_physical_enable", - "enable physical camera controls (using aperture priority)", - generate_hash("r_dof_physical_enable") - }, - { - "r_dof_physical_filmDiagonal", - "Diagonal size of the film/sensor (mm). The bigger the sensor size, the bigger the circle of confusion (which means stronger blurring at all distances). Defaults to full-frame 35mm", - generate_hash("r_dof_physical_filmDiagonal") - }, - { - "r_dof_physical_focusDistance", - "Distance to the plane in focus for the scene", - generate_hash("r_dof_physical_focusDistance") - }, - { - "r_dof_physical_fstop", - "Aperture of the camera for the scene. Lower f-stop yields a shallower depth of field. Typical values range from f/1 to f/22. Rare extremes are f/0.75 and f/32", - generate_hash("r_dof_physical_fstop") - }, - { - "r_dof_physical_hipEnable", - "Enable hyperfocal mode", - generate_hash("r_dof_physical_hipEnable") - }, - { - "r_dof_physical_hipFocusSpeed", - "Hyperfocal mode focus speed (focus dist. far to near, focus dist. near to far, aperture opening, aperture closing)", - generate_hash("r_dof_physical_hipFocusSpeed") - }, - { - "r_dof_physical_hipFstop", - "Aperture of the camera for the scene in the hyperfocal mode", - generate_hash("r_dof_physical_hipFstop") - }, - { - "r_dof_physical_hipSharpCocDiameter", - "Defines what circle of confusion can be considered sharp (mm). Defaults to 0.03mm, generally accepted value for 35mm", - generate_hash("r_dof_physical_hipSharpCocDiameter") - }, - { - "r_dof_physical_maxCocDiameter", - "Maximum circle of confusion diameter (virtual units, might be clamped for bokeh dof)", - generate_hash("r_dof_physical_maxCocDiameter") - }, - { - "r_dof_physical_minFocusDistance", - "Minimum focus distance (inches)", - generate_hash("r_dof_physical_minFocusDistance") - }, - { - "r_dof_physical_viewModelFocusDistance", - "Distance to the plane in focus for the scene", - generate_hash("r_dof_physical_viewModelFocusDistance") - }, - { - "r_dof_physical_viewModelFstop", - "Aperture of the camera for the view model. Lower f-stop yields a shallower depth of field. Typical values range from f/1 to f/22. Rare extremes are f/0.75 and f/32", - generate_hash("r_dof_physical_viewModelFstop") - }, - { - "r_dof_tweak", - "Use dvars to set the depth of field effect; overrides r_dof_enable", - generate_hash("r_dof_tweak") - }, - { - "r_dof_viewModelEnd", - "Depth of field viewmodel end distance, in inches", - generate_hash("r_dof_viewModelEnd") - }, - { - "r_dof_viewModelStart", - "Depth of field viewmodel start distance, in inches", - generate_hash("r_dof_viewModelStart") - }, - { - "r_drawSun", - "Enable sun effects", - generate_hash("r_drawSun") - }, - { - "r_drawWater", - "Enable water animation", - generate_hash("r_drawWater") - }, - { - "r_dynamicOPL", - "Enable drawing vfx lights as overlapping primary light for saving gpu performance.", - generate_hash("r_dynamicOPL") - }, - { - "r_dynamicSpotLightShadows", - "Enable shadows for dynamic/VFX spot lights, you should set this dvar then spawn the new light.", - generate_hash("r_dynamicSpotLightShadows") - }, - { - "r_elevatedPriority", - "Utilize priority elevation for process.", - generate_hash("r_elevatedPriority") - }, - { - "r_emblemBrightnessScale", - "Modifier that scales the brightness of the emblem on model materials", - generate_hash("r_emblemBrightnessScale") - }, - { - "r_emissiveMap", - "Replace all emissive maps with pure black or pure white", - generate_hash("r_emissiveMap") - }, - { - "r_enableNoTessBuckets", - "Enables placing triangles that don't need tessellation into additional draw calls using non-tessellated shaders.", - generate_hash("r_enableNoTessBuckets") - }, - { - "r_envBrdfLutMap", - "Replace environment BRDF lookup table with pure black (no secondary specular) or pure white (maximum secondary specular)", - generate_hash("r_envBrdfLutMap") - }, - { - "r_envMapExponent", - "Reflection exponent.", - generate_hash("r_envMapExponent") - }, - { - "r_envMapMaxIntensity", - "Max reflection intensity based on glancing angle.", - generate_hash("r_envMapMaxIntensity") - }, - { - "r_envMapMinIntensity", - "", - generate_hash("r_envMapMinIntensity") - }, - { - "r_envMapOverride", - "", - generate_hash("r_envMapOverride") - }, - { - "r_envMapSunIntensity", - "Max sun specular intensity intensity with env map materials.", - generate_hash("r_envMapSunIntensity") - }, - { - "r_eyePupil", - " Change eye's pupil Size.", - generate_hash("r_eyePupil") - }, - { - "r_eyeRedness", - " Change eye's redness.", - generate_hash("r_eyeRedness") - }, - { - "r_eyeWetness", - " Change eye's wetness.", - generate_hash("r_eyeWetness") - }, - { - "r_fastModelPrimaryLightCheck", - "Reduce R_GetNonSunPrimaryLightForSphere/R_GetNonSunPrimaryLightForBox function calls", - generate_hash("r_fastModelPrimaryLightCheck") - }, - { - "r_fastModelPrimaryLightLink", - "Speed up R_LinkSphereEntityToPrimaryLights and R_LinkBoxEntityToPrimaryLights.", - generate_hash("r_fastModelPrimaryLightLink") - }, - { - "r_filmAltShader", - "Use alternate shader (with middle tint and dark desat) for film color.", - generate_hash("r_filmAltShader") - }, - { - "r_filmTweakBrightness", - "Tweak dev var; film color brightness", - generate_hash("r_filmTweakBrightness") - }, - { - "r_filmTweakContrast", - "Tweak dev var; film color contrast", - generate_hash("r_filmTweakContrast") - }, - { - "r_filmTweakDarkTint", - "", - generate_hash("r_filmTweakDarkTint") - }, - { - "r_filmTweakDesaturation", - "Tweak dev var; Desaturation applied after all 3D drawing to light areas", - generate_hash("r_filmTweakDesaturation") - }, - { - "r_filmTweakDesaturationDark", - "Tweak dev var; Additional desaturation applied after all 3D drawing to dark areas", - generate_hash("r_filmTweakDesaturationDark") - }, - { - "r_filmTweakEnable", - "Tweak dev var; enable film color effects", - generate_hash("r_filmTweakEnable") - }, - { - "r_filmTweakInvert", - "Tweak dev var; enable inverted video", - generate_hash("r_filmTweakInvert") - }, - { - "r_filmTweakLightTint", - "", - generate_hash("r_filmTweakLightTint") - }, - { - "r_filmTweakMediumTint", - "", - generate_hash("r_filmTweakMediumTint") - }, - { - "r_filmUseTweaks", - "Overide film effects with tweak dvar values.", - generate_hash("r_filmUseTweaks") - }, - { - "r_flushAfterExecute", - "Whether to Flush after ExecuteCommandList.", - generate_hash("r_flushAfterExecute") - }, - { - "r_fog", - "Set to 0 to disable fog", - generate_hash("r_fog") - }, - { - "r_fog_depthhack_scale", - "Fog scale for depth hack surfaces", - generate_hash("r_fog_depthhack_scale") - }, - { - "r_fog_ev_adjust", - "Fog color ev adjustment (+2 means fog color is 2 stops brighter)", - generate_hash("r_fog_ev_adjust") - }, - { - "r_font_cache_debug_display", - "Display the current fontcache texture on the HUD for debug purposes", - generate_hash("r_font_cache_debug_display") - }, - { - "r_forceLod", - "Force all level of detail to this level", - generate_hash("r_forceLod") - }, - { - "r_fullbright", - "Toggles rendering without lighting", - generate_hash("r_fullbright") - }, - { - "r_fxaaSubpixel", - "FXAA sub-pixel amount, lower values have more aliasing and less blur", - generate_hash("r_fxaaSubpixel") - }, - { - "r_FXAverageColorFunc", - "How to compute FX system average color? 0 = use IWrad equation, 1 = legacy equation, 2 = spherical harmonics 1 coefficient.", - generate_hash("r_FXAverageColorFunc") - }, - { - "r_globalGenericMaterialScale", - "Hack global generic material constants", - generate_hash("r_globalGenericMaterialScale") - }, - { - "r_glow_allowed", - "Allow glow.", - generate_hash("r_glow_allowed") - }, - { - "r_glow_allowed_script_forced", - "Force 'allow glow' to be treated as true, by script.", - generate_hash("r_glow_allowed_script_forced") - }, - { - "r_gunSightColorEntityScale", - "Scale the gun sight color when over an entity.", - generate_hash("r_gunSightColorEntityScale") - }, - { - "r_gunSightColorNoneScale", - "Scale the gun sight color when not over an entity.", - generate_hash("r_gunSightColorNoneScale") - }, - { - "r_hbaoBias", - "HBAO bias", - generate_hash("r_hbaoBias") - }, - { - "r_hbaoBlurEnable", - "HBAO blur enabled", - generate_hash("r_hbaoBlurEnable") - }, - { - "r_hbaoBlurSharpness", - "HBAO blur sharpness", - generate_hash("r_hbaoBlurSharpness") - }, - { - "r_hbaoCoarseAO", - "HBAO coarse AO", - generate_hash("r_hbaoCoarseAO") - }, - { - "r_hbaoDebug", - "Debug render HBAO occlusion", - generate_hash("r_hbaoDebug") - }, - { - "r_hbaoDetailAO", - "HBAO detail AO", - generate_hash("r_hbaoDetailAO") - }, - { - "r_hbaoPowerExponent", - "HBAO power exponent", - generate_hash("r_hbaoPowerExponent") - }, - { - "r_hbaoRadius", - "HBAO radius", - generate_hash("r_hbaoRadius") - }, - { - "r_hbaoSceneScale", - "HBAO scene scale", - generate_hash("r_hbaoSceneScale") - }, - { - "r_hbaoStrengthBlend", - "Blend factor between the artist-tuned proportional strength r_hbaoStrengthScale*artStrength, and the fixed strength r_hbaoStrengthFixed. A value of 0.0 is fully the proportional value, and a value of 1.0 is fully the fixed value.", - generate_hash("r_hbaoStrengthBlend") - }, - { - "r_hbaoStrengthFixed", - "Fixed HBAO strength. Only used if r_hbaoStrengthBlend > 0.0.", - generate_hash("r_hbaoStrengthFixed") - }, - { - "r_hbaoStrengthScale", - "Scale factor to convert SSAO strength to HBAO strength. Only used if r_hbaoStrengthBlend < 1.0.", - generate_hash("r_hbaoStrengthScale") - }, - { - "r_hbaoUseScriptScale", - "Enable/disable script-controlled strength scale while HBAO is active.", - generate_hash("r_hbaoUseScriptScale") - }, - { - "r_hemiAoBlurTolerance", - "Hemi SSAO Blur Tolerance (log10)", - generate_hash("r_hemiAoBlurTolerance") - }, - { - "r_hemiAoCombineResolutionsBeforeBlur", - "The higher quality modes blend wide and narrow sampling patterns. The wide pattern is due to deinterleaving and requires blurring. The narrow pattern is not on a deinterleaved buffer, but it only samples every other pixel. The blur on it is optional. If you combine the two before blurring, the narrow will get blurred as well. This creates a softer effect but can remove any visible noise from having 50% sample coverage.", - generate_hash("r_hemiAoCombineResolutionsBeforeBlur") - }, - { - "r_hemiAoCombineResolutionsWithMul", - "When combining the wide and narrow patterns, a mul() operation can be used or a min() operation. Multiplication exaggerates the result creating even darker creases. This is an artistic choice. I think it looks less natural, but often art teams prefer more exaggerated contrast. For me, it's more about having the right AO falloff so that it's a smooth gradient rather than falling off precipitously and forming overly dark recesses.", - generate_hash("r_hemiAoCombineResolutionsWithMul") - }, - { - "r_hemiAoDepthSquash", - "Hemi SSAO depth squash. Value is rcp.", - generate_hash("r_hemiAoDepthSquash") - }, - { - "r_hemiAoEnable", - "Enable Hemi SSAO", - generate_hash("r_hemiAoEnable") - }, - { - "r_hemiAoHierarchyDepth", - "Hemi SSAO recursion depth (filter width)", - generate_hash("r_hemiAoHierarchyDepth") - }, - { - "r_hemiAoMaxDepthDownsample", - "Use max depth value when downsampling, instead of pseudo-randomly picking a depth sample? Leaving this at the default false may produce more stable results.", - generate_hash("r_hemiAoMaxDepthDownsample") - }, - { - "r_hemiAoNoiseFilterTolerance", - "This is necessary to filter out pixel shimmer due to bilateral upsampling with too much lost resolution. High frequency detail can sometimes not be reconstructed, and the noise filter fills in the missing pixels with the result of the higher resolution SSAO. Value is log10.", - generate_hash("r_hemiAoNoiseFilterTolerance") - }, - { - "r_hemiAoPower", - "Power curve applied to Hemi SSAO factor, not applied in game yet", - generate_hash("r_hemiAoPower") - }, - { - "r_hemiAoQualityLevel", - "Hemi SSAO quality setting", - generate_hash("r_hemiAoQualityLevel") - }, - { - "r_hemiAoRejectionFalloff", - "Controls how aggressive to fade off samples that occlude spheres but by so much as to be unreliable. This is what gives objects a dark halo around them when placed in front of a wall. If you want to fade off the halo, boost your rejection falloff. The tradeoff is that it reduces overall AO. Value is rcp.", - generate_hash("r_hemiAoRejectionFalloff") - }, - { - "r_hemiAoStrength", - "Strength of Hemi Screen Space Ambient Occlusion effect", - generate_hash("r_hemiAoStrength") - }, - { - "r_hemiAoUpsampleTolerance", - "Hemi SSAO Upsample Tolerance (log10)", - generate_hash("r_hemiAoUpsampleTolerance") - }, - { - "r_heroLighting", - "Enable hero-only lighting", - generate_hash("r_heroLighting") - }, - { - "r_highLodDist", - "Distance for high level of detail", - generate_hash("r_highLodDist") - }, - { - "r_hudFx", - "Draw HUD Effects", - generate_hash("r_hudFx") - }, - { - "r_hudOutlineEnable", - "Enables wireframe outlines to be drawn around DObjs (as a post process).", - generate_hash("r_hudOutlineEnable") - }, - { - "r_hudOutlinePostMode", - "hud outline apply mode", - generate_hash("r_hudOutlinePostMode") - }, - { - "r_hudOutlineWidth", - "Set the width of the Hud Outline", - generate_hash("r_hudOutlineWidth") - }, - { - "r_ignore", - "", - generate_hash("r_ignore") - }, - { - "r_ignoref", - "", - generate_hash("r_ignoref") - }, - { - "r_imageQuality", - "Image quality", - generate_hash("r_imageQuality") - }, - { - "r_inGameVideo", - "Allow in game cinematics", - generate_hash("r_inGameVideo") - }, - { - "r_lateAllocParamCacheAllowed", - "Enable late allocation of parameter cache for VS stage.", - generate_hash("r_lateAllocParamCacheAllowed") - }, - { - "r_lateAllocParamCacheDefault", - "Late allocation of parameter cache value for sub-div materials.", - generate_hash("r_lateAllocParamCacheDefault") - }, - { - "r_lateAllocParamCacheDisplacement", - "Late allocation of parameter cache value for sub-div materials.", - generate_hash("r_lateAllocParamCacheDisplacement") - }, - { - "r_lateAllocParamCacheSubdiv", - "Late allocation of parameter cache value for sub-div materials.", - generate_hash("r_lateAllocParamCacheSubdiv") - }, - { - "r_lightCacheLessFrequentMaxDistance", - "Adjust the distance fx models (and models tagged as less-frequently-lit by script) move before immediately being relit", - generate_hash("r_lightCacheLessFrequentMaxDistance") - }, - { - "r_lightCacheLessFrequentPeriod", - "Adjust how frequently fx models (and models tagged as less-frequently-lit by script) get relit on average (1 is every frame, 8 is every 8th frame)", - generate_hash("r_lightCacheLessFrequentPeriod") - }, - { - "r_lightGridAvgApplyPrimaryLight", - "apply primary light color onto r_showLightGridAvgProbes boxes", - generate_hash("r_lightGridAvgApplyPrimaryLight") - }, - { - "r_lightGridAvgFollowCamera", - "allow the r_showLightGridAvgProbes boxes following current camera position", - generate_hash("r_lightGridAvgFollowCamera") - }, - { - "r_lightGridAvgProbeCount", - "how many light grid avg color probes will show up)", - generate_hash("r_lightGridAvgProbeCount") - }, - { - "r_lightGridAvgTraceGround", - " lock boxes to ground ", - generate_hash("r_lightGridAvgTraceGround") - }, - { - "r_lightGridContrast", - "Adjust the contrast of light color from the light grid", - generate_hash("r_lightGridContrast") - }, - { - "r_lightGridDefaultFXLightingLookup", - "Default FX lighting lookup location\n", - generate_hash("r_lightGridDefaultFXLightingLookup") - }, - { - "r_lightGridDefaultModelLightingLookup", - "Default model lighting lookup location", - generate_hash("r_lightGridDefaultModelLightingLookup") - }, - { - "r_lightGridEnableTweaks", - "Enable tweaks of the light color from the light grid", - generate_hash("r_lightGridEnableTweaks") - }, - { - "r_lightGridIntensity", - "Adjust the intensity of light color from the light grid", - generate_hash("r_lightGridIntensity") - }, - { - "r_lightGridSHBands", - "Spherical harmonics bands being used for evaluating current-gen light grids colors. 0 = default, 1 = 1 band, 2 = 2 bands, 3 = 3 bands.\n", - generate_hash("r_lightGridSHBands") - }, - { - "r_lightGridUseTweakedValues", - "Use tweaked values instead of default", - generate_hash("r_lightGridUseTweakedValues") - }, - { - "r_lightMap", - "Replace all lightmaps with pure black or pure white", - generate_hash("r_lightMap") - }, - { - "r_litSurfaceHDRScalar", - "Vision set based scalar applied to lit surfaces", - generate_hash("r_litSurfaceHDRScalar") - }, - { - "r_loadForRenderer", - "Set to false to disable dx allocations (for dedicated server mode)", - generate_hash("r_loadForRenderer") - }, - { - "r_lockPvs", - "Lock the viewpoint used for determining what is visible to the current position and direction", - generate_hash("r_lockPvs") - }, - { - "r_lod4Dist", - "Distance for lowest level of detail 4", - generate_hash("r_lod4Dist") - }, - { - "r_lod5Dist", - "Distance for lowest level of detail 5", - generate_hash("r_lod5Dist") - }, - { - "r_lodBiasRigid", - "", - generate_hash("r_lodBiasRigid") - }, - { - "r_lodBiasSkinned", - "", - generate_hash("r_lodBiasSkinned") - }, - { - "r_lodScaleRigid", - "", - generate_hash("r_lodScaleRigid") - }, - { - "r_lodScaleSkinned", - "", - generate_hash("r_lodScaleSkinned") - }, - { - "r_lowestLodDist", - "Distance for lowest level of detail", - generate_hash("r_lowestLodDist") - }, - { - "r_lowLodDist", - "Distance for low level of detail", - generate_hash("r_lowLodDist") - }, - { - "r_mbEnable", - "Set of objects which will be enabled for motion blur", - generate_hash("r_mbEnable") - }, - { - "r_mbFastEnable", - "Toggle on/off fast high quality motion blur", - generate_hash("r_mbFastEnable") - }, - { - "r_mbFastPreset", - "Changes motion blur quality", - generate_hash("r_mbFastPreset") - }, - { - "r_mdao", - "Enable the medium distance ambient occlusion feature", - generate_hash("r_mdao") - }, - { - "r_mdaoAsyncOccluderGen", - "The occluder generation step is performed via async compute", - generate_hash("r_mdaoAsyncOccluderGen") - }, - { - "r_mdaoBoneInfluenceRadiusScale", - "Scale for the bone influence radius for mdao", - generate_hash("r_mdaoBoneInfluenceRadiusScale") - }, - { - "r_mdaoCapsuleStrength", - "MDAO strength for capsule occluders", - generate_hash("r_mdaoCapsuleStrength") - }, - { - "r_mdaoMinBoneBoundsToOcclude", - "Minimum volume of the bone collider to create occluders for", - generate_hash("r_mdaoMinBoneBoundsToOcclude") - }, - { - "r_mdaoOccluderCullDistance", - "Culling distance for mdao occluders", - generate_hash("r_mdaoOccluderCullDistance") - }, - { - "r_mdaoOccluderFadeOutStartDistance", - "Fade out distance for mdao occluders", - generate_hash("r_mdaoOccluderFadeOutStartDistance") - }, - { - "r_mdaoUseTweaks", - "Use r_mdao* dvars instead of the current light set values for MDAO", - generate_hash("r_mdaoUseTweaks") - }, - { - "r_mdaoVolumeStrength", - "MDAO strength for volume occluders", - generate_hash("r_mdaoVolumeStrength") - }, - { - "r_mediumLodDist", - "Distance for medium level of detail", - generate_hash("r_mediumLodDist") - }, - { - "r_mode", - "Display mode", - generate_hash("r_mode") - }, - { - "r_modelLightingMap", - "Replace all model lighting maps (light grid) with pure black", - generate_hash("r_modelLightingMap") - }, - { - "r_monitor", - "Index of the monitor to use in a multi monitor system; 0 picks automatically.", - generate_hash("r_monitor") - }, - { - "r_mpRimColor", - "Change character's rim color for multiplayer", - generate_hash("r_mpRimColor") - }, - { - "r_mpRimDiffuseTint", - "Change character's rim diffuse tint for multiplayer.", - generate_hash("r_mpRimDiffuseTint") - }, - { - "r_mpRimStrength", - "Change character's rim color for multiplayer", - generate_hash("r_mpRimStrength") - }, - { - "r_multiGPU", - "Enable multi GPU compat mode.", - generate_hash("r_multiGPU") - }, - { - "r_normalMap", - "Replace all normal maps with a flat normal map", - generate_hash("r_normalMap") - }, - { - "r_nvidiaGPU", - "Enable NV API.", - generate_hash("r_nvidiaGPU") - }, - { - "r_offchipTessellationAllowed", - "Enable off-chip tessellation support.", - generate_hash("r_offchipTessellationAllowed") - }, - { - "r_offchipTessellationTfThreshold", - "Tessellation factor threshold for off-chip.", - generate_hash("r_offchipTessellationTfThreshold") - }, - { - "r_offchipTessellationWaveThreshold", - "Domain shader wave threshold for off-chip.", - generate_hash("r_offchipTessellationWaveThreshold") - }, - { - "r_omitUnusedRenderTargets", - "Omit unused render targets to save memory. Changing this requires a vid_restart.", - generate_hash("r_omitUnusedRenderTargets") - }, - { - "r_outdoor", - "Prevents snow from going indoors", - generate_hash("r_outdoor") - }, - { - "r_outdoorFeather", - "Outdoor z-feathering value", - generate_hash("r_outdoorFeather") - }, - { - "r_particleHdr", - "Enable Hdr Particle Features", - generate_hash("r_particleHdr") - }, - { - "r_patchCountAllowed", - "Enable run-time setting of patch count per draw call.", - generate_hash("r_patchCountAllowed") - }, - { - "r_picmip", - "Picmip level of color maps. If r_picmip_manual is 0, this is read-only.", - generate_hash("r_picmip") - }, - { - "r_picmip_bump", - "Picmip level of normal maps. If r_picmip_manual is 0, this is read-only.", - generate_hash("r_picmip_bump") - }, - { - "r_picmip_spec", - "Picmip level of specular maps. If r_picmip_manual is 0, this is read-only.", - generate_hash("r_picmip_spec") - }, - { - "r_picmip_water", - "Picmip level of water maps.", - generate_hash("r_picmip_water") - }, - { - "r_polygonOffsetBias", - "Offset bias for decal polygons; bigger values z-fight less but poke through walls more", - generate_hash("r_polygonOffsetBias") - }, - { - "r_polygonOffsetClamp", - "Offset clamp for decal polygons; bigger values z-fight less but poke through walls more", - generate_hash("r_polygonOffsetClamp") - }, - { - "r_polygonOffsetScale", - "Offset scale for decal polygons; bigger values z-fight less but poke through walls more", - generate_hash("r_polygonOffsetScale") - }, - { - "r_portalBevels", - "Helps cull geometry by angles of portals that are acute when projected onto the screen, value is the cosine of the angle", - generate_hash("r_portalBevels") - }, - { - "r_portalBevelsOnly", - "Use screen-space bounding box of portals rather than the actual shape of the portal projected onto the screen", - generate_hash("r_portalBevelsOnly") - }, - { - "r_portalMinClipArea", - "Don't clip child portals by a parent portal smaller than this fraction of the screen area.", - generate_hash("r_portalMinClipArea") - }, - { - "r_portalMinRecurseDepth", - "Ignore r_portalMinClipArea for portals with fewer than this many parent portals.", - generate_hash("r_portalMinRecurseDepth") - }, - { - "r_portalWalkLimit", - "Stop portal recursion after this many iterations. Useful for debugging portal errors.", - generate_hash("r_portalWalkLimit") - }, - { - "r_postAA", - "Post process antialiasing mode", - generate_hash("r_postAA") - }, - { - "r_postfx_enable", - "Enable post-processing effects such as color correction, bloom, depth-of-field, etc.", - generate_hash("r_postfx_enable") - }, - { - "r_preloadShaders", - "Force D3D to draw dummy geometry with all shaders during level load; may fix long pauses at level start.", - generate_hash("r_preloadShaders") - }, - { - "r_primaryLightTweakDiffuseStrength", - "Tweak the diffuse intensity for primary lights", - generate_hash("r_primaryLightTweakDiffuseStrength") - }, - { - "r_primaryLightTweakSpecularStrength", - "Tweak the specular intensity for primary lights", - generate_hash("r_primaryLightTweakSpecularStrength") - }, - { - "r_primaryLightUseTweaks", - "", - generate_hash("r_primaryLightUseTweaks") - }, - { - "r_reactiveMotionActorRadius", - "Radial distance from the ai characters that influences reactive motion models (inches)", - generate_hash("r_reactiveMotionActorRadius") - }, - { - "r_reactiveMotionActorVelocityMax", - "AI velocity considered the maximum when determining the length of motion tails (inches/sec)", - generate_hash("r_reactiveMotionActorVelocityMax") - }, - { - "r_reactiveMotionActorZOffset", - "Distance from the actor origin along Z direction where the actor's reactive motion effector sphere is centered at.", - generate_hash("r_reactiveMotionActorZOffset") - }, - { - "r_reactiveMotionEffectorStrengthScale", - "Additional scale for the effector influence, as a factor of the model part distance from the effector center and model part extents", - generate_hash("r_reactiveMotionEffectorStrengthScale") - }, - { - "r_reactiveMotionHelicopterLimit", - "Maximum number of helicopter entities that actively influence reactive motion. Can increase CPU cost of the scene.", - generate_hash("r_reactiveMotionHelicopterLimit") - }, - { - "r_reactiveMotionHelicopterRadius", - "Radial distance from the helicopter that influences reactive motion models (inches)", - generate_hash("r_reactiveMotionHelicopterRadius") - }, - { - "r_reactiveMotionHelicopterStrength", - "Scales the influence of helicopter wind tunnel motion", - generate_hash("r_reactiveMotionHelicopterStrength") - }, - { - "r_reactiveMotionPlayerHeightAdjust", - "Amount to adjust the vertical distance of the effector from the player position (inches)", - generate_hash("r_reactiveMotionPlayerHeightAdjust") - }, - { - "r_reactiveMotionPlayerRadius", - "Radial distance from the player that influences reactive motion models (inches)", - generate_hash("r_reactiveMotionPlayerRadius") - }, - { - "r_reactiveMotionPlayerZOffset", - "Distance from the player origin along Z direction where the player's reactive motion effector sphere is centered at.", - generate_hash("r_reactiveMotionPlayerZOffset") - }, - { - "r_reactiveMotionVelocityTailScale", - "Additional scale for the velocity-based motion tails, as a factor of the effector radius", - generate_hash("r_reactiveMotionVelocityTailScale") - }, - { - "r_reactiveMotionWindAmplitudeScale", - "Scales amplitude of wind wave motion", - generate_hash("r_reactiveMotionWindAmplitudeScale") - }, - { - "r_reactiveMotionWindAreaScale", - "Scales distribution of wind motion", - generate_hash("r_reactiveMotionWindAreaScale") - }, - { - "r_reactiveMotionWindDir", - "Controls the global wind direction", - generate_hash("r_reactiveMotionWindDir") - }, - { - "r_reactiveMotionWindFrequencyScale", - "Scales frequency of wind wave motion", - generate_hash("r_reactiveMotionWindFrequencyScale") - }, - { - "r_reactiveMotionWindStrength", - "Scale of the global wind direction (inches/sec)", - generate_hash("r_reactiveMotionWindStrength") - }, - { - "r_reflectionProbeMap", - "Replace all reflection probes with pure black", - generate_hash("r_reflectionProbeMap") - }, - { - "r_reflectionProbeNmlLuminance", - "Enable/disable shader code for computing luminance during reflection probe denormalization. This is just an experiment.\n", - generate_hash("r_reflectionProbeNmlLuminance") - }, - { - "r_rimLight0Color", - "", - generate_hash("r_rimLight0Color") - }, - { - "r_rimLight0Heading", - "Rim Light 0 heading in degrees", - generate_hash("r_rimLight0Heading") - }, - { - "r_rimLight0Pitch", - "Rim Light 0 pitch in degrees -90 is noon.", - generate_hash("r_rimLight0Pitch") - }, - { - "r_rimLightBias", - "How much to bias the n.l calculation", - generate_hash("r_rimLightBias") - }, - { - "r_rimLightDiffuseIntensity", - "Strength of the diffuse component of the rim light.", - generate_hash("r_rimLightDiffuseIntensity") - }, - { - "r_rimLightFalloffMaxDistance", - "Distance at which the rim light hits intensity of 100%.", - generate_hash("r_rimLightFalloffMaxDistance") - }, - { - "r_rimLightFalloffMinDistance", - "Distance at which the rim light hits intensity of 100%.", - generate_hash("r_rimLightFalloffMinDistance") - }, - { - "r_rimLightFalloffMinIntensity", - "Intensity of the effect at and before minDistance.", - generate_hash("r_rimLightFalloffMinIntensity") - }, - { - "r_rimLightPower", - "Power to raise the n.l calculation", - generate_hash("r_rimLightPower") - }, - { - "r_rimLightSpecIntensity", - "Strength of the spec ( additive) component of the rim light", - generate_hash("r_rimLightSpecIntensity") - }, - { - "r_rimLightUseTweaks", - "Turn on rim lighting tweaks", - generate_hash("r_rimLightUseTweaks") - }, - { - "r_scaleViewport", - "Scale 3D viewports by this fraction. Use this to see if framerate is pixel shader bound.", - generate_hash("r_scaleViewport") - }, - { - "r_sceneMipShowOverlay", - "Toggles scene mip rendertarget overlay", - generate_hash("r_sceneMipShowOverlay") - }, - { - "r_showLightGrid", - "Show light grid debugging information (2: detailed, 3: detailed for this box only)", - generate_hash("r_showLightGrid") - }, - { - "r_showLightGridAvgProbes", - "show an array of boxes which are using the light grid average color at its location", - generate_hash("r_showLightGridAvgProbes") - }, - { - "r_showLightGridDetailInfo", - "Show more details for light grid debugging.", - generate_hash("r_showLightGridDetailInfo") - }, - { - "r_showLightProbes", - "Show the light probes at the light grid sample locations in world space centered around the camera.", - generate_hash("r_showLightProbes") - }, - { - "r_showMissingLightGrid", - "Use rainbow colors for entities that are outside the light grid", - generate_hash("r_showMissingLightGrid") - }, - { - "r_showModelLightingLowWaterMark", - "", - generate_hash("r_showModelLightingLowWaterMark") - }, - { - "r_showPortals", - "Show portals for debugging", - generate_hash("r_showPortals") - }, - { - "r_showPortalsOverview", - "Render 2d XY portal overlay scaled to fit to this distance. Useful for debugging portal errors.", - generate_hash("r_showPortalsOverview") - }, - { - "r_showReflectionProbeSelection", - "Show reflection probe selection", - generate_hash("r_showReflectionProbeSelection") - }, - { - "r_singleCell", - "Only draw things in the same cell as the camera. Most useful for seeing how big the current cell is.", - generate_hash("r_singleCell") - }, - { - "r_skipPvs", - "Skipt the determination of what is in the potentially visible set (disables most drawing)", - generate_hash("r_skipPvs") - }, - { - "r_sky_fog_intensity", - "Amount of sky fog fading", - generate_hash("r_sky_fog_intensity") - }, - { - "r_sky_fog_max_angle", - "End of angular sky fog fading", - generate_hash("r_sky_fog_max_angle") - }, - { - "r_sky_fog_min_angle", - "Start of angular sky fog fading", - generate_hash("r_sky_fog_min_angle") - }, - { - "r_skyFogUseTweaks", - "Enable dvar control of sky fog", - generate_hash("r_skyFogUseTweaks") - }, - { - "r_smaaThreshold", - "SMAA edge detection threshold", - generate_hash("r_smaaThreshold") - }, - { - "r_smodelInstancedRenderer", - "Render static models with instanced renderer", - generate_hash("r_smodelInstancedRenderer") - }, - { - "r_smodelInstancedThreshold", - "Minimum number of static model instances before instanced rendering is used", - generate_hash("r_smodelInstancedThreshold") - }, - { - "r_smp_backend", - "Process renderer back end in a separate thread", - generate_hash("r_smp_backend") - }, - { - "r_smp_worker", - "Process renderer front end in a separate thread", - generate_hash("r_smp_worker") - }, - { - "r_smp_worker_thread0", - "", - generate_hash("r_smp_worker_thread0") - }, - { - "r_smp_worker_thread1", - "", - generate_hash("r_smp_worker_thread1") - }, - { - "r_smp_worker_thread2", - "", - generate_hash("r_smp_worker_thread2") - }, - { - "r_smp_worker_thread3", - "undefined", - generate_hash("r_smp_worker_thread3") - }, - { - "r_smp_worker_thread4", - "undefined", - generate_hash("r_smp_worker_thread4") - }, - { - "r_smp_worker_thread5", - "undefined", - generate_hash("r_smp_worker_thread5") - }, - { - "r_smp_worker_thread6", - "undefined", - generate_hash("r_smp_worker_thread6") - }, - { - "r_smp_worker_thread7", - "undefined", - generate_hash("r_smp_worker_thread7") - }, - { - "r_specOccMap", - "Replace all specular occlusion maps with pure black (fully occluded) or pure white (not occluded)", - generate_hash("r_specOccMap") - }, - { - "r_specularColorScale", - "Set greater than 1 to brighten specular highlights", - generate_hash("r_specularColorScale") - }, - { - "r_specularMap", - "Replace all specular maps with pure black (off) or pure white (super shiny)", - generate_hash("r_specularMap") - }, - { - "r_spotLightEntityShadows", - "Enable entity shadows for spot lights.", - generate_hash("r_spotLightEntityShadows") - }, - { - "r_spotLightShadows", - "Enable shadows for spot lights.", - generate_hash("r_spotLightShadows") - }, - { - "r_ssao", - "Screen Space Ambient Occlusion mode", - generate_hash("r_ssao") - }, - { - "r_ssaoDebug", - "Render calculated or blurred Screen Space Ambient Occlusion values", - generate_hash("r_ssaoDebug") - }, - { - "r_ssaoDebugMip", - "Selects which mip level to render when r_ssaoDebug is enabled. If 0 and r_ssaoDownsample is enabled, will render mip 1.", - generate_hash("r_ssaoDebugMip") - }, - { - "r_ssaoDepthScale", - "Scale applied to depth values used for occlusion tests.", - generate_hash("r_ssaoDepthScale") - }, - { - "r_ssaoDepthScaleViewModel", - "Scale applied to depth values used for occlusion tests.", - generate_hash("r_ssaoDepthScaleViewModel") - }, - { - "r_ssaoDownsample", - "Screen Space Ambient Occlusion calculation occurs at half linear resolution", - generate_hash("r_ssaoDownsample") - }, - { - "r_ssaoFadeDepth", - "Depth at which the SSAO begins to fade out. It fades at even increments of this distance (e.g. it's at 1 for depth r_ssaoFadeDepth, 1/2 for depth 2*r_ssaoFadeDepth, etc.)", - generate_hash("r_ssaoFadeDepth") - }, - { - "r_ssaoGapFalloff", - "Falloff used to blend between creases (that should darken) and silhouettes (that should not darken). Lower values falloff more quickly.", - generate_hash("r_ssaoGapFalloff") - }, - { - "r_ssaoGradientFalloff", - "Falloff used to fade out the effect for steep depth gradients (i.e. surfaces nearly parallel to the camera direction). This fixes sampling artifacts that appear for surfaces nearly parallel to the camera direction (commonly occuring for flat ground planes).", - generate_hash("r_ssaoGradientFalloff") - }, - { - "r_ssaoMaxStrengthDepth", - "Depth at which SSAO strength is at its maximum", - generate_hash("r_ssaoMaxStrengthDepth") - }, - { - "r_ssaoMethod", - "Screen Space Ambient Occlusion method (original or IW6, both are volumetric obscurance)", - generate_hash("r_ssaoMethod") - }, - { - "r_ssaoMinPixelWidth", - "Minimum pixel width of the effect. When the effect is smaller than this, it is culled entirely.", - generate_hash("r_ssaoMinPixelWidth") - }, - { - "r_ssaoMinStrengthDepth", - "Depth at which SSAO strength is zero, effectively disabled", - generate_hash("r_ssaoMinStrengthDepth") - }, - { - "r_ssaoMultiRes", - "Screen Space Ambient Occlusion calculation occurs at half linear resolution", - generate_hash("r_ssaoMultiRes") - }, - { - "r_ssaoPower", - "Power curve applied to SSAO factor", - generate_hash("r_ssaoPower") - }, - { - "r_ssaoRejectDepth", - "Depth at which the SSAO is disabled. Smaller values result in more rejected pixels which is faster, but limits the distance at which the effect is visible.", - generate_hash("r_ssaoRejectDepth") - }, - { - "r_ssaoSampleCount", - "Selects the number of samples used for SSAO", - generate_hash("r_ssaoSampleCount") - }, - { - "r_ssaoScriptScale", - "Allows script to lerp to disable or enable the SSAO. This applies a scalar value to the SSAO strength. When set to 0, this effectively disables SSAO.", - generate_hash("r_ssaoScriptScale") - }, - { - "r_ssaoStrength", - "Strength of Screen Space Ambient Occlusion effect", - generate_hash("r_ssaoStrength") - }, - { - "r_ssaoUseTweaks", - "Use r_ssao* dvars instead of the current light set values for SSAO", - generate_hash("r_ssaoUseTweaks") - }, - { - "r_ssaoWidth", - "The width of the SSAO effect, in pixels at 720p. Larger values increase area but lower effective quality.", - generate_hash("r_ssaoWidth") - }, - { - "r_sse_skinning", - "Use Streaming SIMD Extensions for skinning", - generate_hash("r_sse_skinning") - }, - { - "r_ssrBlendScale", - "Add extra scale to ssr weight versus reflection probe weight, >1 value will make ssr more obvious.", - generate_hash("r_ssrBlendScale") - }, - { - "r_ssrFadeInDuration", - "Duration of the screen-space reflection fade-in, which occurs whenever the reflection source buffer is invalidated due to view changes (in particular, dual-view scope transitions).", - generate_hash("r_ssrFadeInDuration") - }, - { - "r_ssrPositionCorrection", - "Screen space reflection position correction blend factor", - generate_hash("r_ssrPositionCorrection") - }, - { - "r_ssrRoughnessMipParameters", - "X: mirror mip; Y: roughest mip; Z: roughness middle point, may need different value for different screen resolution on PC.", - generate_hash("r_ssrRoughnessMipParameters") - }, - { - "r_sssBlendWeight", - "Controls the blend between the wide (zero) and narrow (one) gaussians", - generate_hash("r_sssBlendWeight") - }, - { - "r_sssDebugMaterial", - "Debug Feature: toggle materials with SSS", - generate_hash("r_sssDebugMaterial") - }, - { - "r_sssEnable", - "Enables the subsurface scattering effect (note that disabling SSS will not prevent the filter from running)", - generate_hash("r_sssEnable") - }, - { - "r_sssGlobalRadius", - "Controls the global radius (in inches)", - generate_hash("r_sssGlobalRadius") - }, - { - "r_sssJitterRadius", - "Percentage of the kernel to be jittered", - generate_hash("r_sssJitterRadius") - }, - { - "r_sssNarrowRadius", - "Controls the narrow Gaussian radius", - generate_hash("r_sssNarrowRadius") - }, - { - "r_sssPreset", - "Changes subsurface scattering quality", - generate_hash("r_sssPreset") - }, - { - "r_sssWideRadius", - "Controls the wide Gaussian radius", - generate_hash("r_sssWideRadius") - }, - { - "r_subdiv", - "Enables Catmull Clark surface subdivision.", - generate_hash("r_subdiv") - }, - { - "r_subdivLimit", - "Set the maximum Catmull Clark subdivision level.", - generate_hash("r_subdivLimit") - }, - { - "r_subdivPatchCount", - "Patches per thread group for sub-division surfaces.", - generate_hash("r_subdivPatchCount") - }, - { - "r_subdomainLimit", - "Maximum number of extra tessellation subdivisions using instancing (max tess amts are 0:64, 1:128, 2:192, 3:256, max instances used are 0:1, 1:4, 2:9, 3:12)", - generate_hash("r_subdomainLimit") - }, - { - "r_subdomainScale", - "Debug only: Scales the extra subdivision amount (for values < 1, not all instanced sub triangles will draw).", - generate_hash("r_subdomainScale") - }, - { - "r_subwindow", - "subwindow to draw: left, right, top, bottom", - generate_hash("r_subwindow") - }, - { - "r_sun_from_dvars", - "Set sun flare values from dvars rather than the level", - generate_hash("r_sun_from_dvars") - }, - { - "r_sun_fx_position", - "Position in degrees of the sun effect", - generate_hash("r_sun_fx_position") - }, - { - "r_sunblind_fadein", - "time in seconds to fade blind from 0% to 100%", - generate_hash("r_sunblind_fadein") - }, - { - "r_sunblind_fadeout", - "time in seconds to fade blind from 100% to 0%", - generate_hash("r_sunblind_fadeout") - }, - { - "r_sunblind_max_angle", - "angle from sun in degrees inside which effect is max", - generate_hash("r_sunblind_max_angle") - }, - { - "r_sunblind_max_darken", - "0-1 fraction for how black the world is at max blind", - generate_hash("r_sunblind_max_darken") - }, - { - "r_sunblind_min_angle", - "angle from sun in degrees outside which effect is 0", - generate_hash("r_sunblind_min_angle") - }, - { - "r_sunflare_fadein", - "time in seconds to fade alpha from 0% to 100%", - generate_hash("r_sunflare_fadein") - }, - { - "r_sunflare_fadeout", - "time in seconds to fade alpha from 100% to 0%", - generate_hash("r_sunflare_fadeout") - }, - { - "r_sunflare_max_alpha", - "0-1 vertex color and alpha of sun at max effect", - generate_hash("r_sunflare_max_alpha") - }, - { - "r_sunflare_max_angle", - "angle from sun in degrees inside which effect is max", - generate_hash("r_sunflare_max_angle") - }, - { - "r_sunflare_max_size", - "largest size of flare effect in pixels at 640x480", - generate_hash("r_sunflare_max_size") - }, - { - "r_sunflare_min_angle", - "angle from sun in degrees outside which effect is 0", - generate_hash("r_sunflare_min_angle") - }, - { - "r_sunflare_min_size", - "smallest size of flare effect in pixels at 640x480", - generate_hash("r_sunflare_min_size") - }, - { - "r_sunflare_shader", - "name for flare effect; can be any material", - generate_hash("r_sunflare_shader") - }, - { - "r_sunglare_fadein", - "time in seconds to fade glare from 0% to 100%", - generate_hash("r_sunglare_fadein") - }, - { - "r_sunglare_fadeout", - "time in seconds to fade glare from 100% to 0%", - generate_hash("r_sunglare_fadeout") - }, - { - "r_sunglare_max_angle", - "angle from sun in degrees inside which effect is max", - generate_hash("r_sunglare_max_angle") - }, - { - "r_sunglare_max_lighten", - "0-1 fraction for how white the world is at max glare", - generate_hash("r_sunglare_max_lighten") - }, - { - "r_sunglare_min_angle", - "angle from sun in degrees inside which effect is max", - generate_hash("r_sunglare_min_angle") - }, - { - "r_sunInfDist", - "Sun infinite distance used to place sun fx", - generate_hash("r_sunInfDist") - }, - { - "r_sunshadowmap_cmdbuf_worker", - "Process shadowmap command buffer in a separate thread", - generate_hash("r_sunshadowmap_cmdbuf_worker") - }, - { - "r_sunsprite_shader", - "name for static sprite; can be any material", - generate_hash("r_sunsprite_shader") - }, - { - "r_sunsprite_size", - "diameter in pixels at 640x480 and 80 fov", - generate_hash("r_sunsprite_size") - }, - { - "r_surfaceHDRScalarUseTweaks", - "Enables lit and unlit surface scalar tweaks", - generate_hash("r_surfaceHDRScalarUseTweaks") - }, - { - "r_tessellation", - "Enables tessellation of world geometry, with an optional cutoff distance.", - generate_hash("r_tessellation") - }, - { - "r_tessellationCutoffDistance", - "Distance at which world geometry ceases to tessellate.", - generate_hash("r_tessellationCutoffDistance") - }, - { - "r_tessellationCutoffFalloff", - "Range over which tessellation is faded out, up to the cutoff.", - generate_hash("r_tessellationCutoffFalloff") - }, - { - "r_tessellationEyeScale", - "Scale applied due to eye * object normal for less tessellation on facing polygons.", - generate_hash("r_tessellationEyeScale") - }, - { - "r_tessellationFactor", - "Target edge length, based on dividing full window height by this factor, for dynamic tessellation. Use zero to disable tessellation.", - generate_hash("r_tessellationFactor") - }, - { - "r_tessellationHeightAuto", - "Correctly auto scale displacement heights for layers to grow as texture is stretched over larger surface areas to preserve feature proportions.", - generate_hash("r_tessellationHeightAuto") - }, - { - "r_tessellationHeightScale", - "Displacement height scale factor.", - generate_hash("r_tessellationHeightScale") - }, - { - "r_tessellationHybrid", - "Hybrid per pixel displacement scale.", - generate_hash("r_tessellationHybrid") - }, - { - "r_tessellationLodBias", - "Displacement map lod bias.", - generate_hash("r_tessellationLodBias") - }, - { - "r_texFilterAnisoMax", - "Maximum anisotropy to use for texture filtering", - generate_hash("r_texFilterAnisoMax") - }, - { - "r_texFilterAnisoMin", - "Minimum anisotropy to use for texture filtering (overridden by max)", - generate_hash("r_texFilterAnisoMin") - }, - { - "r_texFilterDisable", - "Disables all texture filtering (uses nearest only.)", - generate_hash("r_texFilterDisable") - }, - { - "r_texFilterMipBias", - "Change the mipmap bias", - generate_hash("r_texFilterMipBias") - }, - { - "r_texFilterMipMode", - "Forces all mipmaps to use a particular blend between levels (or disables mipping.)", - generate_hash("r_texFilterMipMode") - }, - { - "r_texFilterProbeBilinear", - "Force reflection probe to use bilinear filter", - generate_hash("r_texFilterProbeBilinear") - }, - { - "r_texShowMipMode", - "Forces textures with the specified mip filtering to draw black.", - generate_hash("r_texShowMipMode") - }, - { - "r_thermalColorOffset", - "Offset of the thermal colors (offset + scale*color)", - generate_hash("r_thermalColorOffset") - }, - { - "r_thermalColorScale", - "Scale of the thermal colors (offset + scale*color)", - generate_hash("r_thermalColorScale") - }, - { - "r_thermalDetailScale", - "Scale of the detail that is added to the thermal map from the normal map (multiplies the detail amount from AssetManager)", - generate_hash("r_thermalDetailScale") - }, - { - "r_thermalFadeColor", - "Color the thermal fades to at distance.", - generate_hash("r_thermalFadeColor") - }, - { - "r_thermalFadeControl", - "Select thermal fade mode", - generate_hash("r_thermalFadeControl") - }, - { - "r_thermalFadeMax", - "Distance at which thermal stops fading", - generate_hash("r_thermalFadeMax") - }, - { - "r_thermalFadeMin", - "Distance at which thermal starts fading", - generate_hash("r_thermalFadeMin") - }, - { - "r_tonemap", - "HDR Tonemapping mode", - generate_hash("r_tonemap") - }, - { - "r_tonemapAdaptSpeed", - "HDR Tonemap exposure adaptation speed", - generate_hash("r_tonemapAdaptSpeed") - }, - { - "r_tonemapAuto", - "HDR Tonemapping performs auto-exposure", - generate_hash("r_tonemapAuto") - }, - { - "r_tonemapAutoExposureAdjust", - "HDR Tonemap Auto Exposure Adjust value (set to 0.0 for automatic adjustment)", - generate_hash("r_tonemapAutoExposureAdjust") - }, - { - "r_tonemapBlack", - "HDR Filmic Tonemap black point", - generate_hash("r_tonemapBlack") - }, - { - "r_tonemapBlend", - "HDR Tonemapping blends between exposures", - generate_hash("r_tonemapBlend") - }, - { - "r_tonemapCrossover", - "HDR Filmic Tonemap crossover point", - generate_hash("r_tonemapCrossover") - }, - { - "r_tonemapDarkEv", - "HDR Tonemap Dark EV", - generate_hash("r_tonemapDarkEv") - }, - { - "r_tonemapDarkExposureAdjust", - "HDR Tonemap Dark Exposure Adjust", - generate_hash("r_tonemapDarkExposureAdjust") - }, - { - "r_tonemapExposure", - "HDR Tonemap exposure (in EV) override (only works in non-auto mode)", - generate_hash("r_tonemapExposure") - }, - { - "r_tonemapExposureAdjust", - "HDR Tonemap exposure adjustment (in EV, 0 is no adjustment, works like a camera where +1 reduces EV by 1)", - generate_hash("r_tonemapExposureAdjust") - }, - { - "r_tonemapGamma", - "HDR Tonemap gamma curve power", - generate_hash("r_tonemapGamma") - }, - { - "r_tonemapHighlightRange", - "HDR Tonemap dynamic range, which determines white point luminance", - generate_hash("r_tonemapHighlightRange") - }, - { - "r_tonemapLightEv", - "HDR Tonemap Light EV", - generate_hash("r_tonemapLightEv") - }, - { - "r_tonemapLightExposureAdjust", - "HDR Tonemap Light Exposure Adjust", - generate_hash("r_tonemapLightExposureAdjust") - }, - { - "r_tonemapLockAutoExposureAdjust", - "HDR Tonemapping lock auto exposure adjust", - generate_hash("r_tonemapLockAutoExposureAdjust") - }, - { - "r_tonemapMaxExposure", - "HDR Tonemap maximum exposure (in EV)", - generate_hash("r_tonemapMaxExposure") - }, - { - "r_tonemapMaxExposureAdjust", - "HDR Tonemap Max Exposure Adjust", - generate_hash("r_tonemapMaxExposureAdjust") - }, - { - "r_tonemapMidEv", - "HDR Tonemap Mid EV", - generate_hash("r_tonemapMidEv") - }, - { - "r_tonemapMidExposureAdjust", - "HDR Tonemap Mid Exposure Adjust", - generate_hash("r_tonemapMidExposureAdjust") - }, - { - "r_tonemapMinExposureAdjust", - "HDR Tonemap Min Exposure Adjust", - generate_hash("r_tonemapMinExposureAdjust") - }, - { - "r_tonemapShoulder", - "HDR Filmic Tonemap shoulder control (0 is linear)", - generate_hash("r_tonemapShoulder") - }, - { - "r_tonemapToe", - "HDR Filmic Tonemap toe control (0 is linear)", - generate_hash("r_tonemapToe") - }, - { - "r_tonemapUseCS", - "HDR Tonemapping uses compute shader.", - generate_hash("r_tonemapUseCS") - }, - { - "r_tonemapUseTweaks", - "Override tone map LightSet settings with tweak dvar values.", - generate_hash("r_tonemapUseTweaks") - }, - { - "r_tonemapWhite", - "HDR Filmic Tonemap white point", - generate_hash("r_tonemapWhite") - }, - { - "r_ui3d_debug_display", - "Show UI3D debug overlay", - generate_hash("r_ui3d_debug_display") - }, - { - "r_ui3d_h", - "ui3d texture window height", - generate_hash("r_ui3d_h") - }, - { - "r_ui3d_use_debug_values", - "Use UI debug values", - generate_hash("r_ui3d_use_debug_values") - }, - { - "r_ui3d_w", - "ui3d texture window width", - generate_hash("r_ui3d_w") - }, - { - "r_ui3d_x", - "ui3d texture window x", - generate_hash("r_ui3d_x") - }, - { - "r_ui3d_y", - "ui3d texture window y", - generate_hash("r_ui3d_y") - }, - { - "r_uiBlurDstMode", - "UI blur distortion mode. Fast uses the scene mip map render target, PostSun uses a downsampled post sun resolve buffer, PostSun HQ uses a gaussian blurred post sun resolve buffer.", - generate_hash("r_uiBlurDstMode") - }, - { - "r_umbra", - "Enables Umbra-based portal culling.", - generate_hash("r_umbra") - }, - { - "r_umbraAccurateOcclusionThreshold", - "The distance (in inches) to which accurate occlusion information is gathered. -1.0 = deduced automatically.", - generate_hash("r_umbraAccurateOcclusionThreshold") - }, - { - "r_umbraExclusive", - "Toggle Umbra for exclusive static culling (disables static portal dpvs)", - generate_hash("r_umbraExclusive") - }, - { - "r_umbraQueryParts", - "The number of parts the Umbra query frustum is broken into for async query processing as an M x N grid (0, 0 = all queries are synchronous).", - generate_hash("r_umbraQueryParts") - }, - { - "r_umbraUseBadPlaces", - "Enable/disable ability to disable umbra when inside special volumes defined in mp/umbraBadPlaces.csv.", - generate_hash("r_umbraUseBadPlaces") - }, - { - "r_umbraUseDpvsCullDist", - "Use cull distance from the DPVS instead of the far plane distance.", - generate_hash("r_umbraUseDpvsCullDist") - }, - { - "r_unlitSurfaceHDRScalar", - "Vision set based scalar applied to unlit surfaces to balance those surfaces with the luminance of the scene", - generate_hash("r_unlitSurfaceHDRScalar") - }, - { - "r_useComputeSkinning", - "Enables compute shader (GPU) skinning.", - generate_hash("r_useComputeSkinning") - }, - { - "r_useLayeredMaterials", - "Set to true to use layered materials on shader model 3 hardware", - generate_hash("r_useLayeredMaterials") - }, - { - "r_useLightGridDefaultFXLightingLookup", - "Enable/disable default fx lighting lookup\n", - generate_hash("r_useLightGridDefaultFXLightingLookup") - }, - { - "r_useLightGridDefaultModelLightingLookup", - "Enable/disable default model lighting lookup\n", - generate_hash("r_useLightGridDefaultModelLightingLookup") - }, - { - "r_useShadowGeomOpt", - "Enable iwRad shadow geometry optimization. It only works when we have the data generated in iwRad.", - generate_hash("r_useShadowGeomOpt") - }, - { - "r_useSunShadowPortals", - "Enable sun shadow portals when dir light change and not using cached shadow.", - generate_hash("r_useSunShadowPortals") - }, - { - "r_useXAnimIK", - "Enables IK animation.", - generate_hash("r_useXAnimIK") - }, - { - "r_vc_makelog", - "Enable logging of light grid points for the vis cache. 1 starts from scratch, 2 appends.", - generate_hash("r_vc_makelog") - }, - { - "r_vc_showlog", - "Show this many rows of light grid points for the vis cache", - generate_hash("r_vc_showlog") - }, - { - "r_veil", - "Apply veiling luminance (HDR glow)", - generate_hash("r_veil") - }, - { - "r_veilAntialiasing", - "Veil antialiasing mode (downsample technique used for first mip).", - generate_hash("r_veilAntialiasing") - }, - { - "r_veilBackgroundStrength", - "Strength of background when applying veiling luminance (HDR glow)", - generate_hash("r_veilBackgroundStrength") - }, - { - "r_veilFalloffScale1", - "Controls the size of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)", - generate_hash("r_veilFalloffScale1") - }, - { - "r_veilFalloffScale2", - "Controls the size of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)", - generate_hash("r_veilFalloffScale2") - }, - { - "r_veilFalloffWeight1", - "Controls the weight of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)", - generate_hash("r_veilFalloffWeight1") - }, - { - "r_veilFalloffWeight2", - "Controls the weight of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)", - generate_hash("r_veilFalloffWeight2") - }, - { - "r_veilFilter", - "Changes the veil filtering mode", - generate_hash("r_veilFilter") - }, - { - "r_veilPreset", - "Changes veil sampling quality", - generate_hash("r_veilPreset") - }, - { - "r_veilRadius", - "Controls the radius of the first Gaussian in virtual pixels (remaining Gaussians follow proportionally).", - generate_hash("r_veilRadius") - }, - { - "r_veilStrength", - "Strength of veiling luminance (HDR glow)", - generate_hash("r_veilStrength") - }, - { - "r_veilUseTweaks", - "Override veil LightSet settings with tweak dvar values.", - generate_hash("r_veilUseTweaks") - }, - { - "r_velocityPrepass", - "Perform velocity rendering during the depth prepass", - generate_hash("r_velocityPrepass") - }, - { - "r_viewModelLightAmbient", - "", - generate_hash("r_viewModelLightAmbient") - }, - { - "r_viewModelPrimaryLightTweakDiffuseStrength", - "Tweak the diffuse intensity for view model primary lights", - generate_hash("r_viewModelPrimaryLightTweakDiffuseStrength") - }, - { - "r_viewModelPrimaryLightTweakSpecularStrength", - "Tweak the specular intensity for view model primary lights", - generate_hash("r_viewModelPrimaryLightTweakSpecularStrength") - }, - { - "r_viewModelPrimaryLightUseTweaks", - "", - generate_hash("r_viewModelPrimaryLightUseTweaks") - }, - { - "r_volumeLightScatter", - "Enables volumetric light scattering", - generate_hash("r_volumeLightScatter") - }, - { - "r_volumeLightScatterAngularAtten", - "Distance of sun from center of screen before angular attenuation starts for god rays", - generate_hash("r_volumeLightScatterAngularAtten") - }, - { - "r_volumeLightScatterBackgroundDistance", - "Distance at which pixels are considered background for volume light scatter effect", - generate_hash("r_volumeLightScatterBackgroundDistance") - }, - { - "r_volumeLightScatterColor", - "", - generate_hash("r_volumeLightScatterColor") - }, - { - "r_volumeLightScatterDepthAttenFar", - "Pixels >= than this depth recieve full volume light scatter.", - generate_hash("r_volumeLightScatterDepthAttenFar") - }, - { - "r_volumeLightScatterDepthAttenNear", - "Pixels <= than this depth recieve no volume light scatter.", - generate_hash("r_volumeLightScatterDepthAttenNear") - }, - { - "r_volumeLightScatterEv", - "Light intensity (in EV) for volumetric light scattering", - generate_hash("r_volumeLightScatterEv") - }, - { - "r_volumeLightScatterLinearAtten", - "Coefficient of linear attenuation of god rays", - generate_hash("r_volumeLightScatterLinearAtten") - }, - { - "r_volumeLightScatterQuadraticAtten", - "Coefficient of quadratic attenuation of god rays)", - generate_hash("r_volumeLightScatterQuadraticAtten") - }, - { - "r_volumeLightScatterUseTweaks", - "Enables volumetric light scattering tweaks", - generate_hash("r_volumeLightScatterUseTweaks") - }, - { - "r_vsync", - "Enable v-sync before drawing the next frame to avoid 'tearing' artifacts.", - generate_hash("r_vsync") - }, - { - "r_warningRepeatDelay", - "Number of seconds after displaying a \"per-frame\" warning before it will display again", - generate_hash("r_warningRepeatDelay") - }, - { - "r_wideTessFactorsThreshold", - "If a surface has more than this many triangles, process triangles in parallel instead of surfaces.", - generate_hash("r_wideTessFactorsThreshold") - }, - { - "r_zfar", - "Change the distance at which culling fog reaches 100% opacity; 0 is off", - generate_hash("r_zfar") - }, - { - "r_znear", - "Things closer than this aren't drawn. Reducing this increases z-fighting in the distance.", - generate_hash("r_znear") - }, - { - "radarjamDistMax", - "", - generate_hash("radarjamDistMax") - }, - { - "radarjamDistMin", - "", - generate_hash("radarjamDistMin") - }, - { - "radarjamSinCurve", - "", - generate_hash("radarjamSinCurve") - }, - { - "radius_damage_debug", - "Turn on debug lines for radius damage traces", - generate_hash("radius_damage_debug") - }, - { - "ragdoll_baselerp_time", - "Default time ragdoll baselerp bones take to reach the base pose", - generate_hash("ragdoll_baselerp_time") - }, - { - "ragdoll_bullet_force", - "Bullet force applied to ragdolls", - generate_hash("ragdoll_bullet_force") - }, - { - "ragdoll_bullet_upbias", - "Upward bias applied to ragdoll bullet effects", - generate_hash("ragdoll_bullet_upbias") - }, - { - "ragdoll_dump_anims", - "Dump animation data when ragdoll fails", - generate_hash("ragdoll_dump_anims") - }, - { - "ragdoll_enable", - "Turn on ragdoll death animations", - generate_hash("ragdoll_enable") - }, - { - "ragdoll_explode_force", - "Explosive force applied to ragdolls", - generate_hash("ragdoll_explode_force") - }, - { - "ragdoll_explode_upbias", - "Upwards bias applied to ragdoll explosion effects", - generate_hash("ragdoll_explode_upbias") - }, - { - "ragdoll_exploding_bullet_force", - "Force applied to ragdolls from explosive bullets", - generate_hash("ragdoll_exploding_bullet_force") - }, - { - "ragdoll_exploding_bullet_upbias", - "Upwards bias applied to ragdoll from explosive bullets", - generate_hash("ragdoll_exploding_bullet_upbias") - }, - { - "ragdoll_idle_min_velsq", - "Minimum squared speed a ragdoll body needs to be moving before it will shut down due to time", - generate_hash("ragdoll_idle_min_velsq") - }, - { - "ragdoll_jitter_scale", - "Scale up or down the effect of physics jitter on ragdolls", - generate_hash("ragdoll_jitter_scale") - }, - { - "ragdoll_jointlerp_time", - "Default time taken to lerp down ragdoll joint friction", - generate_hash("ragdoll_jointlerp_time") - }, - { - "ragdoll_link_to_moving_platform", - "Enable client-side linking of ragdolls to script brush models when they go idle.", - generate_hash("ragdoll_link_to_moving_platform") - }, - { - "ragdoll_max_life", - "Max lifetime of a ragdoll system in msec", - generate_hash("ragdoll_max_life") - }, - { - "ragdoll_max_simulating", - "Max number of simultaneous active ragdolls - archived", - generate_hash("ragdoll_max_simulating") - }, - { - "ragdoll_max_stretch_pct", - "Force ragdoll limbs to not stretch more than this percentage in one frame", - generate_hash("ragdoll_max_stretch_pct") - }, - { - "ragdoll_mp_limit", - "Max number of simultaneous active ragdolls - archived", - generate_hash("ragdoll_mp_limit") - }, - { - "ragdoll_mp_resume_share_after_killcam", - "Msec after returning from killcam that splitscreen players will share ragdolls again.", - generate_hash("ragdoll_mp_resume_share_after_killcam") - }, - { - "ragdoll_resolve_penetration_bias", - "Bias value on force to push ragdolls out of environment.", - generate_hash("ragdoll_resolve_penetration_bias") - }, - { - "ragdoll_rotvel_scale", - "Ragdoll rotational velocity estimate scale", - generate_hash("ragdoll_rotvel_scale") - }, - { - "ragdoll_self_collision_scale", - "Scale the size of the collision capsules used to prevent ragdoll limbs from interpenetrating", - generate_hash("ragdoll_self_collision_scale") - }, - { - "ragdoll_stretch_iters", - "Iterations to run the alternate limb solver", - generate_hash("ragdoll_stretch_iters") - }, - { - "rankedPlayEndMatchKeepLobby", - "keep the lobby if the lobby host is in our private party.", - generate_hash("rankedPlayEndMatchKeepLobby") - }, - { - "rankedPlaylistLockoutDuration", - "Time in seconds to lock the ranked play playlist if a player quit the match early.", - generate_hash("rankedPlaylistLockoutDuration") - }, - { - "rate", - "Player's preferred network rate", - generate_hash("rate") - }, - { - "RemoteCameraSounds_DryLevel", - "", - generate_hash("RemoteCameraSounds_DryLevel") - }, - { - "RemoteCameraSounds_RoomType", - "", - generate_hash("RemoteCameraSounds_RoomType") - }, - { - "RemoteCameraSounds_WetLevel", - "", - generate_hash("RemoteCameraSounds_WetLevel") - }, - { - "requireOpenNat", - "", - generate_hash("requireOpenNat") - }, - { - "restrictMapPacksToGroups", - "Restrict map pack usage to needing all maps in an ala carte package in order to use as search criteria", - generate_hash("restrictMapPacksToGroups") - }, - { - "riotshield_bullet_damage_scale", - "Value to scale bullet damage to deployed riotshield.", - generate_hash("riotshield_bullet_damage_scale") - }, - { - "riotshield_deploy_limit_radius", - "Min distance deployed riotshields must be from each other.", - generate_hash("riotshield_deploy_limit_radius") - }, - { - "riotshield_deploy_trace_parallel", - "Report collisions when riotshield traces are parallel to plane of triangle. If disabled traces parallel to triangle planes do not report collisions at all.", - generate_hash("riotshield_deploy_trace_parallel") - }, - { - "riotshield_deployed_health", - "Deployed riotshield health.", - generate_hash("riotshield_deployed_health") - }, - { - "riotshield_destroyed_cleanup_time", - "Time (in seconds) destroyed riotshield model persists before disappearing", - generate_hash("riotshield_destroyed_cleanup_time") - }, - { - "riotshield_explosive_damage_scale", - "Value to scale explosive damage to deployed riotshield..", - generate_hash("riotshield_explosive_damage_scale") - }, - { - "riotshield_melee_damage_scale", - "Value to scale melee damage to deployed riotshield.", - generate_hash("riotshield_melee_damage_scale") - }, - { - "riotshield_projectile_damage_scale", - "Value to scale projectile damage to deployed riotshield.", - generate_hash("riotshield_projectile_damage_scale") - }, - { - "s_aggregate_ping_offset", - "offset to apply to aggregate ping values", - generate_hash("s_aggregate_ping_offset") - }, - { - "s_aggregate_ping_scale", - "8-bit fixed-point aggregate ping scaler value", - generate_hash("s_aggregate_ping_scale") - }, - { - "s_avg_max_weighting", - "weighting from 0-256 of party average ping vs. worst ping", - generate_hash("s_avg_max_weighting") - }, - { - "s_ds_pingclient_reping_wait_db", - "wait this# of frames for the db thread to settle down before repinging", - generate_hash("s_ds_pingclient_reping_wait_db") - }, - { - "s_use_aggregate_datacenter_pings", - "use newer system for aggregating party pings", - generate_hash("s_use_aggregate_datacenter_pings") - }, - { - "safeArea_adjusted_horizontal", - "User-adjustable horizontal safe area as a fraction of the screen width", - generate_hash("safeArea_adjusted_horizontal") - }, - { - "safeArea_adjusted_vertical", - "User-adjustable vertical safe area as a fraction of the screen height", - generate_hash("safeArea_adjusted_vertical") - }, - { - "safeArea_horizontal", - "Horizontal safe area as a fraction of the screen width", - generate_hash("safeArea_horizontal") - }, - { - "safeArea_vertical", - "Vertical safe area as a fraction of the screen height", - generate_hash("safeArea_vertical") - }, - { - "scr_conf_numlives", - "", - generate_hash("scr_conf_numlives") - }, - { - "scr_conf_playerrespawndelay", - "", - generate_hash("scr_conf_playerrespawndelay") - }, - { - "scr_conf_roundlimit", - "", - generate_hash("scr_conf_roundlimit") - }, - { - "scr_conf_scorelimit", - "", - generate_hash("scr_conf_scorelimit") - }, - { - "scr_conf_timelimit", - "", - generate_hash("scr_conf_timelimit") - }, - { - "scr_conf_waverespawndelay", - "", - generate_hash("scr_conf_waverespawndelay") - }, - { - "scr_conf_winlimit", - "", - generate_hash("scr_conf_winlimit") - }, - { - "scr_default_maxagents", - "", - generate_hash("scr_default_maxagents") - }, - { - "scr_diehard", - "", - generate_hash("scr_diehard") - }, - { - "scr_disableClientSpawnTraces", - "", - generate_hash("scr_disableClientSpawnTraces") - }, - { - "scr_dm_numlives", - "", - generate_hash("scr_dm_numlives") - }, - { - "scr_dm_playerrespawndelay", - "", - generate_hash("scr_dm_playerrespawndelay") - }, - { - "scr_dm_roundlimit", - "", - generate_hash("scr_dm_roundlimit") - }, - { - "scr_dm_scorelimit", - "", - generate_hash("scr_dm_scorelimit") - }, - { - "scr_dm_timelimit", - "", - generate_hash("scr_dm_timelimit") - }, - { - "scr_dm_waverespawndelay", - "", - generate_hash("scr_dm_waverespawndelay") - }, - { - "scr_dm_winlimit", - "", - generate_hash("scr_dm_winlimit") - }, - { - "scr_dom_numlives", - "", - generate_hash("scr_dom_numlives") - }, - { - "scr_dom_playerrespawndelay", - "", - generate_hash("scr_dom_playerrespawndelay") - }, - { - "scr_dom_roundlimit", - "", - generate_hash("scr_dom_roundlimit") - }, - { - "scr_dom_scorelimit", - "", - generate_hash("scr_dom_scorelimit") - }, - { - "scr_dom_timelimit", - "", - generate_hash("scr_dom_timelimit") - }, - { - "scr_dom_waverespawndelay", - "", - generate_hash("scr_dom_waverespawndelay") - }, - { - "scr_dom_winlimit", - "", - generate_hash("scr_dom_winlimit") - }, - { - "scr_explBulletMod", - "", - generate_hash("scr_explBulletMod") - }, - { - "scr_game_allowkillcam", - "script allow killcam", - generate_hash("scr_game_allowkillcam") - }, - { - "scr_game_deathpointloss", - "", - generate_hash("scr_game_deathpointloss") - }, - { - "scr_game_forceuav", - "", - generate_hash("scr_game_forceuav") - }, - { - "scr_game_graceperiod", - "", - generate_hash("scr_game_graceperiod") - }, - { - "scr_game_hardpoints", - "", - generate_hash("scr_game_hardpoints") - }, - { - "scr_game_killstreakdelay", - "", - generate_hash("scr_game_killstreakdelay") - }, - { - "scr_game_lockspectatorpov", - "Lock spectator mode globally, 0=freelook/unlocked, 1=first_person, 2=third_person", - generate_hash("scr_game_lockspectatorpov") - }, - { - "scr_game_onlyheadshots", - "", - generate_hash("scr_game_onlyheadshots") - }, - { - "scr_game_perks", - "", - generate_hash("scr_game_perks") - }, - { - "scr_game_spectatetype", - "", - generate_hash("scr_game_spectatetype") - }, - { - "scr_game_suicidepointloss", - "", - generate_hash("scr_game_suicidepointloss") - }, - { - "scr_gameended", - "", - generate_hash("scr_gameended") - }, - { - "scr_hardcore", - "", - generate_hash("scr_hardcore") - }, - { - "scr_horde_difficulty", - "", - generate_hash("scr_horde_difficulty") - }, - { - "scr_horde_maxagents", - "", - generate_hash("scr_horde_maxagents") - }, - { - "scr_horde_numlives", - "", - generate_hash("scr_horde_numlives") - }, - { - "scr_horde_playerrespawndelay", - "", - generate_hash("scr_horde_playerrespawndelay") - }, - { - "scr_horde_roundlimit", - "", - generate_hash("scr_horde_roundlimit") - }, - { - "scr_horde_scorelimit", - "", - generate_hash("scr_horde_scorelimit") - }, - { - "scr_horde_timelimit", - "", - generate_hash("scr_horde_timelimit") - }, - { - "scr_horde_waverespawndelay", - "", - generate_hash("scr_horde_waverespawndelay") - }, - { - "scr_horde_winlimit", - "", - generate_hash("scr_horde_winlimit") - }, - { - "scr_infect_numlives", - "", - generate_hash("scr_infect_numlives") - }, - { - "scr_infect_playerrespawndelay", - "", - generate_hash("scr_infect_playerrespawndelay") - }, - { - "scr_infect_roundlimit", - "", - generate_hash("scr_infect_roundlimit") - }, - { - "scr_infect_timelimit", - "", - generate_hash("scr_infect_timelimit") - }, - { - "scr_infect_waverespawndelay", - "", - generate_hash("scr_infect_waverespawndelay") - }, - { - "scr_infect_winlimit", - "", - generate_hash("scr_infect_winlimit") - }, - { - "scr_isgamescom", - "script use gamescom demo flow", - generate_hash("scr_isgamescom") - }, - { - "scr_maxPerPlayerExplosives", - "", - generate_hash("scr_maxPerPlayerExplosives") - }, - { - "scr_nukeCancelMode", - "", - generate_hash("scr_nukeCancelMode") - }, - { - "scr_nukeTimer", - "", - generate_hash("scr_nukeTimer") - }, - { - "scr_patientZero", - "", - generate_hash("scr_patientZero") - }, - { - "scr_player_forcerespawn", - "", - generate_hash("scr_player_forcerespawn") - }, - { - "scr_player_healthregentime", - "", - generate_hash("scr_player_healthregentime") - }, - { - "scr_player_maxhealth", - "", - generate_hash("scr_player_maxhealth") - }, - { - "scr_player_numlives", - "", - generate_hash("scr_player_numlives") - }, - { - "scr_player_respawndelay", - "", - generate_hash("scr_player_respawndelay") - }, - { - "scr_player_sprinttime", - "", - generate_hash("scr_player_sprinttime") - }, - { - "scr_player_suicidespawndelay", - "", - generate_hash("scr_player_suicidespawndelay") - }, - { - "scr_RequiredMapAspectratio", - "", - generate_hash("scr_RequiredMapAspectratio") - }, - { - "scr_riotShieldXPBullets", - "", - generate_hash("scr_riotShieldXPBullets") - }, - { - "scr_sd_bombtimer", - "", - generate_hash("scr_sd_bombtimer") - }, - { - "scr_sd_defusetime", - "", - generate_hash("scr_sd_defusetime") - }, - { - "scr_sd_multibomb", - "", - generate_hash("scr_sd_multibomb") - }, - { - "scr_sd_numlives", - "", - generate_hash("scr_sd_numlives") - }, - { - "scr_sd_planttime", - "", - generate_hash("scr_sd_planttime") - }, - { - "scr_sd_playerrespawndelay", - "", - generate_hash("scr_sd_playerrespawndelay") - }, - { - "scr_sd_roundlimit", - "", - generate_hash("scr_sd_roundlimit") - }, - { - "scr_sd_roundswitch", - "", - generate_hash("scr_sd_roundswitch") - }, - { - "scr_sd_scorelimit", - "", - generate_hash("scr_sd_scorelimit") - }, - { - "scr_sd_timelimit", - "", - generate_hash("scr_sd_timelimit") - }, - { - "scr_sd_waverespawndelay", - "", - generate_hash("scr_sd_waverespawndelay") - }, - { - "scr_sd_winlimit", - "", - generate_hash("scr_sd_winlimit") - }, - { - "scr_sr_bombtimer", - "", - generate_hash("scr_sr_bombtimer") - }, - { - "scr_sr_defusetime", - "", - generate_hash("scr_sr_defusetime") - }, - { - "scr_sr_multibomb", - "", - generate_hash("scr_sr_multibomb") - }, - { - "scr_sr_numlives", - "", - generate_hash("scr_sr_numlives") - }, - { - "scr_sr_planttime", - "", - generate_hash("scr_sr_planttime") - }, - { - "scr_sr_playerrespawndelay", - "", - generate_hash("scr_sr_playerrespawndelay") - }, - { - "scr_sr_roundlimit", - "", - generate_hash("scr_sr_roundlimit") - }, - { - "scr_sr_roundswitch", - "", - generate_hash("scr_sr_roundswitch") - }, - { - "scr_sr_scorelimit", - "", - generate_hash("scr_sr_scorelimit") - }, - { - "scr_sr_timelimit", - "", - generate_hash("scr_sr_timelimit") - }, - { - "scr_sr_waverespawndelay", - "", - generate_hash("scr_sr_waverespawndelay") - }, - { - "scr_sr_winlimit", - "", - generate_hash("scr_sr_winlimit") - }, - { - "scr_team_fftype", - "script team friendly fire type", - generate_hash("scr_team_fftype") - }, - { - "scr_team_respawntime", - "", - generate_hash("scr_team_respawntime") - }, - { - "scr_team_teamkillpointloss", - "", - generate_hash("scr_team_teamkillpointloss") - }, - { - "scr_team_teamkillspawndelay", - "", - generate_hash("scr_team_teamkillspawndelay") - }, - { - "scr_thirdPerson", - "", - generate_hash("scr_thirdPerson") - }, - { - "scr_tispawndelay", - "", - generate_hash("scr_tispawndelay") - }, - { - "scr_war_halftime", - "", - generate_hash("scr_war_halftime") - }, - { - "scr_war_numlives", - "", - generate_hash("scr_war_numlives") - }, - { - "scr_war_playerrespawndelay", - "", - generate_hash("scr_war_playerrespawndelay") - }, - { - "scr_war_roundlimit", - "", - generate_hash("scr_war_roundlimit") - }, - { - "scr_war_roundswitch", - "", - generate_hash("scr_war_roundswitch") - }, - { - "scr_war_scorelimit", - "", - generate_hash("scr_war_scorelimit") - }, - { - "scr_war_timelimit", - "", - generate_hash("scr_war_timelimit") - }, - { - "scr_war_waverespawndelay", - "", - generate_hash("scr_war_waverespawndelay") - }, - { - "scr_war_winlimit", - "", - generate_hash("scr_war_winlimit") - }, - { - "scr_xpscale", - "", - generate_hash("scr_xpscale") - }, - { - "screenshots_active", - "Are we allowed to enable Screenshots or not", - generate_hash("screenshots_active") - }, - { - "search_weight_asn", - "The weight used for the asn in weighted matchmaking.", - generate_hash("search_weight_asn") - }, - { - "search_weight_country_code", - "The weight used for the country code in weighted matchmaking.", - generate_hash("search_weight_country_code") - }, - { - "search_weight_lat_long", - "The weight used for the lat long in weighted matchmaking.", - generate_hash("search_weight_lat_long") - }, - { - "sensitivity", - "Mouse sensitivity", - generate_hash("sensitivity") - }, - { - "sentry_placement_feet_offset", - "Position of the feet from the center axis.", - generate_hash("sentry_placement_feet_offset") - }, - { - "sentry_placement_feet_trace_dist_z", - "Max distance for a foot to be considered touching the ground", - generate_hash("sentry_placement_feet_trace_dist_z") - }, - { - "sentry_placement_trace_dist", - "Distance along the trace axis where the sentry will attempt to position itself", - generate_hash("sentry_placement_trace_dist") - }, - { - "sentry_placement_trace_min_normal", - "Minimum normal to accept a sentry position", - generate_hash("sentry_placement_trace_min_normal") - }, - { - "sentry_placement_trace_parallel", - "Enable turret traces that are parallel to plane of triangle. If 0, traces parallel to triangle planes do not report collisions at all. If 2 (debug-only), then trace code ping pongs between new and old.", - generate_hash("sentry_placement_trace_parallel") - }, - { - "sentry_placement_trace_pitch", - "Pitch used for the trace axis", - generate_hash("sentry_placement_trace_pitch") - }, - { - "sentry_placement_trace_radius", - "Radius of the bound used for the placement trace", - generate_hash("sentry_placement_trace_radius") - }, - { - "sentry_placement_trace_radius_canon_safety", - "Extra radius used in the forward direction to compensate for the canon length", - generate_hash("sentry_placement_trace_radius_canon_safety") - }, - { - "server1", - "", - generate_hash("server1") - }, - { - "server10", - "", - generate_hash("server10") - }, - { - "server11", - "", - generate_hash("server11") - }, - { - "server12", - "", - generate_hash("server12") - }, - { - "server13", - "", - generate_hash("server13") - }, - { - "server14", - "", - generate_hash("server14") - }, - { - "server15", - "", - generate_hash("server15") - }, - { - "server16", - "", - generate_hash("server16") - }, - { - "server2", - "", - generate_hash("server2") - }, - { - "server3", - "", - generate_hash("server3") - }, - { - "server4", - "", - generate_hash("server4") - }, - { - "server5", - "", - generate_hash("server5") - }, - { - "server6", - "", - generate_hash("server6") - }, - { - "server7", - "", - generate_hash("server7") - }, - { - "server8", - "", - generate_hash("server8") - }, - { - "server9", - "", - generate_hash("server9") - }, - { - "session_immediateDeleteTinySessions", - "Whether to immediately delete sessions with 1 user", - generate_hash("session_immediateDeleteTinySessions") - }, - { - "session_modify_retry_on_failure", - "Enable session modify retry on failures.", - generate_hash("session_modify_retry_on_failure") - }, - { - "session_nonblocking", - "Non-blocking Session code", - generate_hash("session_nonblocking") - }, - { - "sf_use_bw", - "Intel Cheat - CoD Noir.", - generate_hash("sf_use_bw") - }, - { - "sf_use_chaplin", - "Intel Cheat - Ragtime Warfare.", - generate_hash("sf_use_chaplin") - }, - { - "sf_use_clustergrenade", - "Intel Cheat - Cluster Bombs.", - generate_hash("sf_use_clustergrenade") - }, - { - "sf_use_contrast", - "Intel Cheat - Super Contrast.", - generate_hash("sf_use_contrast") - }, - { - "sf_use_ignoreammo", - "Intel Cheat - Infinite Ammo.", - generate_hash("sf_use_ignoreammo") - }, - { - "sf_use_invert", - "Intel Cheat - Photo-Negative.", - generate_hash("sf_use_invert") - }, - { - "sf_use_lemonade_mode", - "Intel Cheat - Lemon-nade.", - generate_hash("sf_use_lemonade_mode") - }, - { - "sf_use_melon_mode", - "Intel Cheat - Melon Heads.", - generate_hash("sf_use_melon_mode") - }, - { - "sf_use_ragdoll_mode", - "Intel Cheat - Ragdoll Impact.", - generate_hash("sf_use_ragdoll_mode") - }, - { - "sf_use_slowmo", - "Intel Cheat - Slow-Mo Ability.", - generate_hash("sf_use_slowmo") - }, - { - "sf_use_tire_explosion", - "Intel Cheat - A Bad Year.", - generate_hash("sf_use_tire_explosion") - }, - { - "sf_use_tracksuit_mode", - "Intel Cheat - Zakhaev's Sons.", - generate_hash("sf_use_tracksuit_mode") - }, - { - "shortversion", - "Short game version", - generate_hash("shortversion") - }, - { - "showDebugAmmoCounter", - "Show the debug ammo counter when unable to show ar ammo counter", - generate_hash("showDebugAmmoCounter") - }, - { - "showPlaylistTotalPlayers", - "Toggle the display of the total number of players in a playlist and online", - generate_hash("showPlaylistTotalPlayers") - }, - { - "sm_cacheSpotShadows", - "Cache spot shadow maps, improves shadow map performance at the cost of memory (requires vid_restart)", - generate_hash("sm_cacheSpotShadows") - }, - { - "sm_cacheSpotShadowsEnabled", - "Enables caching of spot shadows.", - generate_hash("sm_cacheSpotShadowsEnabled") - }, - { - "sm_cacheSunShadow", - "Cache sun shadow map, improves shadow map performance at the cost of memory (requires vid_restart)", - generate_hash("sm_cacheSunShadow") - }, - { - "sm_cacheSunShadowEnabled", - "Enables caching of sun-based shadows.", - generate_hash("sm_cacheSunShadowEnabled") - }, - { - "sm_cameraOffset", - "", - generate_hash("sm_cameraOffset") - }, - { - "sm_dynlightAllSModels", - "Enable, from script, rendering all static models in dynamic light volume when shadow mapping", - generate_hash("sm_dynlightAllSModels") - }, - { - "sm_enable", - "Enable shadow mapping", - generate_hash("sm_enable") - }, - { - "sm_fastSunShadow", - "Fast sun shadow", - generate_hash("sm_fastSunShadow") - }, - { - "sm_lightScore_eyeProjectDist", - "When picking shadows for primary lights, measure distance from a point this far in front of the camera.", - generate_hash("sm_lightScore_eyeProjectDist") - }, - { - "sm_lightScore_spotProjectFrac", - "When picking shadows for primary lights, measure distance from a point this far in front of the camera.", - generate_hash("sm_lightScore_spotProjectFrac") - }, - { - "sm_maxLightsWithShadows", - "Limits how many primary lights can have shadow maps", - generate_hash("sm_maxLightsWithShadows") - }, - { - "sm_minSpotLightScore", - "Minimum score (based on intensity, radius, and position relative to the camera) for a spot light to have shadow maps.", - generate_hash("sm_minSpotLightScore") - }, - { - "sm_polygonOffsetBias", - "Shadow map offset bias", - generate_hash("sm_polygonOffsetBias") - }, - { - "sm_polygonOffsetClamp", - "Shadow map offset clamp", - generate_hash("sm_polygonOffsetClamp") - }, - { - "sm_polygonOffsetPreset", - "Shadow map polygon offset preset.", - generate_hash("sm_polygonOffsetPreset") - }, - { - "sm_polygonOffsetScale", - "Shadow map offset scale", - generate_hash("sm_polygonOffsetScale") - }, - { - "sm_qualitySpotShadow", - "Fast spot shadow", - generate_hash("sm_qualitySpotShadow") - }, - { - "sm_shadowUseTweaks", - "Override shadow LightSet settings with tweak dvar values.", - generate_hash("sm_shadowUseTweaks") - }, - { - "sm_spotDistCull", - "Distance cull spot shadows", - generate_hash("sm_spotDistCull") - }, - { - "sm_spotEnable", - "Enable spot shadow mapping from script", - generate_hash("sm_spotEnable") - }, - { - "sm_spotFilterRadius", - "Spot soft shadows filter radius", - generate_hash("sm_spotFilterRadius") - }, - { - "sm_spotLightScoreModelScale", - "Scale the calculated spot light score by this value if the light currently only affects static or script brush models.", - generate_hash("sm_spotLightScoreModelScale") - }, - { - "sm_spotLightScoreRadiusPower", - "Power to apply to light radius when determining spot light shadowing score (1.0 means radius scales up score a lot, 0.0 means don't scale score using radius)", - generate_hash("sm_spotLightScoreRadiusPower") - }, - { - "sm_spotLimit", - "Limit number of spot shadows from script", - generate_hash("sm_spotLimit") - }, - { - "sm_spotShadowFadeTime", - "How many seconds it takes for a primary light shadow map to fade in or out", - generate_hash("sm_spotShadowFadeTime") - }, - { - "sm_strictCull", - "Strict shadow map cull", - generate_hash("sm_strictCull") - }, - { - "sm_sunEnable", - "Enable sun shadow mapping from script", - generate_hash("sm_sunEnable") - }, - { - "sm_sunFilterRadius", - "Sun soft shadows filter radius", - generate_hash("sm_sunFilterRadius") - }, - { - "sm_sunSampleSizeNear", - "Shadow sample size", - generate_hash("sm_sunSampleSizeNear") - }, - { - "sm_sunShadowBoundsMax", - "Max Shadow Bounds", - generate_hash("sm_sunShadowBoundsMax") - }, - { - "sm_sunShadowBoundsMin", - "Min Shadow Bounds", - generate_hash("sm_sunShadowBoundsMin") - }, - { - "sm_sunShadowBoundsOverride", - "Override Shadow Bounds", - generate_hash("sm_sunShadowBoundsOverride") - }, - { - "sm_sunShadowCenter", - "Sun shadow center, 0 0 0 means don't override", - generate_hash("sm_sunShadowCenter") - }, - { - "sm_sunShadowCenterMode", - "When false center value only used for far map, when true sets both maps", - generate_hash("sm_sunShadowCenterMode") - }, - { - "sm_sunShadowScale", - "Sun shadow scale optimization", - generate_hash("sm_sunShadowScale") - }, - { - "sm_sunShadowScaleLocked", - "Lock usage of sm_sunShadowScale at 1", - generate_hash("sm_sunShadowScaleLocked") - }, - { - "sm_usedSunCascadeCount", - "How many shadow cascade we are using", - generate_hash("sm_usedSunCascadeCount") - }, - { - "snd_allowHeadphoneHRTF", - "Enable HRTF over headphones", - generate_hash("snd_allowHeadphoneHRTF") - }, - { - "snd_announcerDisabled", - "Disable all in-game announcers", - generate_hash("snd_announcerDisabled") - }, - { - "snd_announcerVoicePrefix", - "Local mp announcer voice to use", - generate_hash("snd_announcerVoicePrefix") - }, - { - "snd_battlechatterDisabled", - "Disable all in-game battle chatter", - generate_hash("snd_battlechatterDisabled") - }, - { - "snd_cinematicVolumeScale", - "Scales the volume of Bink videos.", - generate_hash("snd_cinematicVolumeScale") - }, - { - "snd_detectedSpeakerConfig", - "speaker configuration:\n0: autodetect\n1: mono\n2: stereo\n4: quadrophonic\n6: 5.1 surround\n8: 7.1 surround", - generate_hash("snd_detectedSpeakerConfig") - }, - { - "snd_dopplerAuditionEnable", - "Enables doppler calculation preview mode", - generate_hash("snd_dopplerAuditionEnable") - }, - { - "snd_dopplerBaseSpeedOfSound", - "The base speed of sound used in doppler calculation", - generate_hash("snd_dopplerBaseSpeedOfSound") - }, - { - "snd_dopplerEnable", - "Enables doppler calculation", - generate_hash("snd_dopplerEnable") - }, - { - "snd_dopplerPitchMax", - "Maximum pitch that can be legally applied by doppler", - generate_hash("snd_dopplerPitchMax") - }, - { - "snd_dopplerPitchMin", - "Minimum pitch that can be legally applied by doppler", - generate_hash("snd_dopplerPitchMin") - }, - { - "snd_dopplerPlayerVelocityScale", - "The scale of the player velocity, relative the the sound source velocity, when applied to the doppler calculation", - generate_hash("snd_dopplerPlayerVelocityScale") - }, - { - "snd_dopplerSmoothing", - "Smoothing factor applied to doppler to eliminate jitter or sudden acceleration changes", - generate_hash("snd_dopplerSmoothing") - }, - { - "snd_draw3D", - "Draw the position and info of world sounds", - generate_hash("snd_draw3D") - }, - { - "snd_drawInfo", - "Draw debugging information for sounds", - generate_hash("snd_drawInfo") - }, - { - "snd_enable2D", - "Enable 2D sounds", - generate_hash("snd_enable2D") - }, - { - "snd_enable3D", - "Enable 3D sounds", - generate_hash("snd_enable3D") - }, - { - "snd_enableEq", - "Enable equalization filter", - generate_hash("snd_enableEq") - }, - { - "snd_enableReverb", - "Enable sound reverberation", - generate_hash("snd_enableReverb") - }, - { - "snd_enableStream", - "Enable streamed sounds", - generate_hash("snd_enableStream") - }, - { - "snd_envFollowerBuffScale", - "Amount of buffer to use for envelope follower. Smaller value indicates faster envelope.", - generate_hash("snd_envFollowerBuffScale") - }, - { - "snd_errorOnMissing", - "Cause a Com_Error if a sound file is missing.", - generate_hash("snd_errorOnMissing") - }, - { - "snd_hitsoundDisabled", - "Disable the hit indicator sound", - generate_hash("snd_hitsoundDisabled") - }, - { - "snd_inheritSecondaryPitchVol", - "Set to true for secondary aliases to inherit the pitch of the parent", - generate_hash("snd_inheritSecondaryPitchVol") - }, - { - "snd_levelFadeTime", - "The amout of time in milliseconds for all audio to fade in at the start of a level", - generate_hash("snd_levelFadeTime") - }, - { - "snd_loadFadeTime", - "Fade time for loading from a checkpoint after death.", - generate_hash("snd_loadFadeTime") - }, - { - "snd_loopFadeTime", - "Fade-in time for looping sounds.", - generate_hash("snd_loopFadeTime") - }, - { - "snd_musicDisabled", - "Disable all in-game music", - generate_hash("snd_musicDisabled") - }, - { - "snd_musicDisabledForCustomSoundtrack", - "Disable all in-game music due to user playing a custom soundtrack", - generate_hash("snd_musicDisabledForCustomSoundtrack") - }, - { - "snd_occlusionDelay", - "Minimum delay in (ms) between occlusion updates", - generate_hash("snd_occlusionDelay") - }, - { - "snd_occlusionLerpTime", - "Time to lerp to target occlusion lerp when occluded", - generate_hash("snd_occlusionLerpTime") - }, - { - "snd_peakLimiterCompression", - "Peak limiter compression factor. The output data is scaled by this and then normalized: F < 1 = disabled; F >= 1 enabled.", - generate_hash("snd_peakLimiterCompression") - }, - { - "snd_peakLimiterDecay", - "Peak limiter compression decay ratio.", - generate_hash("snd_peakLimiterDecay") - }, - { - "snd_peakLimiterSustainFrames", - "Number of frames to sustain the limiter peak. 1 frame = 10 msec.", - generate_hash("snd_peakLimiterSustainFrames") - }, - { - "snd_premixVolume", - "Game sound pre-mix volume", - generate_hash("snd_premixVolume") - }, - { - "snd_reverbZoneOutsideFactor", - "When a 3d sound is played in a different reverb zone than the player, this factor will be applied to its wet level.", - generate_hash("snd_reverbZoneOutsideFactor") - }, - { - "snd_slaveFadeTime", - "The amount of time in milliseconds for a 'slave' sound\nto fade its volumes when a master sound starts or stops", - generate_hash("snd_slaveFadeTime") - }, - { - "snd_speakerConfig", - "speaker configuration:\n0: autodetect\n1: mono\n2: stereo\n4: quadrophonic\n6: 5.1 surround\n8: 7.1 surround", - generate_hash("snd_speakerConfig") - }, - { - "snd_touchStreamFilesOnLoad", - "Check whether stream sound files exist while loading", - generate_hash("snd_touchStreamFilesOnLoad") - }, - { - "snd_useOldPanning", - "Use old and busted panning", - generate_hash("snd_useOldPanning") - }, - { - "snd_virtualChannelInfo", - "Display virtual voice info.", - generate_hash("snd_virtualChannelInfo") - }, - { - "snd_virtualMinDur", - "The minimum duration (in seconds) of a sound if it is to be added to the virtual voice buffer.", - generate_hash("snd_virtualMinDur") - }, - { - "snd_virtualMinPri", - "The minimum priority of an alias if it is to be added to the virtual voice buffer.", - generate_hash("snd_virtualMinPri") - }, - { - "snd_virtualMinTimeLeftToRevive", - "The minimum time (in ms) left in a sample in order to attempt to revive it.", - generate_hash("snd_virtualMinTimeLeftToRevive") - }, - { - "snd_virtualReviveVoices", - "Whether or not to restore virtual voices.", - generate_hash("snd_virtualReviveVoices") - }, - { - "snd_virtualWaitToReviveTime", - "The minimum time (in ms) to wait before trying to revive the voice.", - generate_hash("snd_virtualWaitToReviveTime") - }, - { - "snd_volume", - "Game sound master volume", - generate_hash("snd_volume") - }, - { - "speech_active", - "Are we allowed to enable Speech or not", - generate_hash("speech_active") - }, - { - "splitscreen", - "Current game is a splitscreen game", - generate_hash("splitscreen") - }, - { - "steam_ingame_p2p_throttle", - "Time, in MS, to wait between P2P packet lookups when in-game", - generate_hash("steam_ingame_p2p_throttle") - }, - { - "stringtable_debug", - "spam debug info for stringtable lookups", - generate_hash("stringtable_debug") - }, - { - "sv_allowClientConsole", - "Allow remote clients to access the console", - generate_hash("sv_allowClientConsole") - }, - { - "sv_allowedClan1", - "", - generate_hash("sv_allowedClan1") - }, - { - "sv_allowedClan2", - "", - generate_hash("sv_allowedClan2") - }, - { - "sv_archiveClientsPositions", - "Archive the client positions to speed up SV_GetClientPositionsAtTime", - generate_hash("sv_archiveClientsPositions") - }, - { - "sv_checkMinPlayers", - "Check min players. 0 disables", - generate_hash("sv_checkMinPlayers") - }, - { - "sv_clientArchive", - "Have the clients archive data to save bandwidth on the server", - generate_hash("sv_clientArchive") - }, - { - "sv_connectTimeout", - "seconds without any message when a client is loading", - generate_hash("sv_connectTimeout") - }, - { - "sv_cumulThinkTime", - "Max client think per server 50 msec frame", - generate_hash("sv_cumulThinkTime") - }, - { - "sv_error_on_baseline_failure", - "Throw an error if the const baseline data is invalid.", - generate_hash("sv_error_on_baseline_failure") - }, - { - "sv_exponentialBackoffAfterNonAckedMsgs", - "start exponential backoff on msg frequency if the client has not acked the last X messages", - generate_hash("sv_exponentialBackoffAfterNonAckedMsgs") - }, - { - "sv_hostname", - "Host name of the server", - generate_hash("sv_hostname") - }, - { - "sv_hugeSnapshotDelay", - "How long to wait before building a new snapshot after a 'huge' snapshot is sent", - generate_hash("sv_hugeSnapshotDelay") - }, - { - "sv_hugeSnapshotSize", - "Size of a snapshot to be considered 'huge'", - generate_hash("sv_hugeSnapshotSize") - }, - { - "sv_kickBanTime", - "Time in seconds for a player to be banned from the server after being kicked", - generate_hash("sv_kickBanTime") - }, - { - "sv_local_client_snapshot_msec", - "Local client snapshot rate, add to cl_penaltyTime", - generate_hash("sv_local_client_snapshot_msec") - }, - { - "sv_maxclients", - "The maximum number of clients that can connect to a server", - generate_hash("sv_maxclients") - }, - { - "sv_minPingClamp", - "Clamp the minimum ping to this value", - generate_hash("sv_minPingClamp") - }, - { - "sv_network_fps", - "Number of times per second the server checks for net messages", - generate_hash("sv_network_fps") - }, - { - "sv_numExpBackoffBeforeReleasingCachedSnapshots", - "if a client is under an exponential backoff over this dvar, then we will release all the cached snapshot data he owns and will send him a baseline if he reconnects", - generate_hash("sv_numExpBackoffBeforeReleasingCachedSnapshots") - }, - { - "sv_paused", - "Pause the server", - generate_hash("sv_paused") - }, - { - "sv_privateClients", - "Maximum number of private clients allowed on the server", - generate_hash("sv_privateClients") - }, - { - "sv_privateClientsForClients", - "The # of private clients (we send this to clients)", - generate_hash("sv_privateClientsForClients") - }, - { - "sv_privatePassword", - "password for the privateClient slots", - generate_hash("sv_privatePassword") - }, - { - "sv_reconnectlimit", - "minimum seconds between connect messages", - generate_hash("sv_reconnectlimit") - }, - { - "sv_rejoinTimeout", - "seconds without any message before allowing a rejoin", - generate_hash("sv_rejoinTimeout") - }, - { - "sv_remote_client_snapshot_joiningstate_msec", - "Remote client snapshot rate during join (until the client acked his first delta message)", - generate_hash("sv_remote_client_snapshot_joiningstate_msec") - }, - { - "sv_remote_client_snapshot_msec", - "Remote client snapshot rate, add to cl_penaltyTime", - generate_hash("sv_remote_client_snapshot_msec") - }, - { - "sv_resetOnSpawn", - "Have clients reset some player state fields when spawning rather than sending them over the network", - generate_hash("sv_resetOnSpawn") - }, - { - "sv_running", - "Server is running", - generate_hash("sv_running") - }, - { - "sv_sayName", - "", - generate_hash("sv_sayName") - }, - { - "sv_showAverageBPS", - "Show average bytes per second for net debugging", - generate_hash("sv_showAverageBPS") - }, - { - "sv_testValue", - "Max antilag rewind", - generate_hash("sv_testValue") - }, - { - "sv_timeout", - "seconds without any message", - generate_hash("sv_timeout") - }, - { - "sv_trackFrameMsecThreshold", - "server frame time that will trigger script time tracking.", - generate_hash("sv_trackFrameMsecThreshold") - }, - { - "sv_useExtraCompress", - "Use zlib compress for gamestate/baseline/score packets", - generate_hash("sv_useExtraCompress") - }, - { - "sv_zlib_threshold", - "Message size threshold which triggers more aggressive compression", - generate_hash("sv_zlib_threshold") - }, - { - "sv_zombietime", - "seconds to sync messages after disconnect", - generate_hash("sv_zombietime") - }, - { - "svwp", - "playerdata server write protection: 0 = disable, 1 = silent, 2 = kick", - generate_hash("svwp") - }, - { - "syncTimeTimeout", - "default timeout for sync time task (in seconds)", - generate_hash("syncTimeTimeout") - }, - { - "sys_configSum", - "Configuration checksum", - generate_hash("sys_configSum") - }, - { - "sys_configureGHz", - "Normalized total CPU power, based on cpu type, count, and speed; used in autoconfigure", - generate_hash("sys_configureGHz") - }, - { - "sys_cpuGHz", - "Measured CPU speed", - generate_hash("sys_cpuGHz") - }, - { - "sys_cpuName", - "CPU name description", - generate_hash("sys_cpuName") - }, - { - "sys_gpu", - "GPU description", - generate_hash("sys_gpu") - }, - { - "sys_lockThreads", - "Prevents specified threads from changing CPUs; improves profiling and may fix some bugs, but can hurt performance", - generate_hash("sys_lockThreads") - }, - { - "sys_quitMigrateTime", - "Time in msec to wait for host migration when user closes the window", - generate_hash("sys_quitMigrateTime") - }, - { - "sys_smp_allowed", - "Allow multi-threading", - generate_hash("sys_smp_allowed") - }, - { - "sys_SSE", - "Operating system allows Streaming SIMD Extensions", - generate_hash("sys_SSE") - }, - { - "sys_sysMB", - "Physical memory in the system", - generate_hash("sys_sysMB") - }, - { - "systemlink", - "Current game is a system link game", - generate_hash("systemlink") - }, - { - "systemlink_host", - "Local client is hosting system link game", - generate_hash("systemlink_host") - }, - { - "tb_report", - "tb event record", - generate_hash("tb_report") - }, - { - "team_rebalance", - "rebalance", - generate_hash("team_rebalance") - }, - { - "teambalance_option", - "Selects active teambalance algorithm. 0 = heuristic 1 = exhaustive", - generate_hash("teambalance_option") - }, - { - "theater_active", - "Are we allowed to show theater or not.", - generate_hash("theater_active") - }, - { - "thermal_playerModel", - "Model to draw for players when in thermal vision mode", - generate_hash("thermal_playerModel") - }, - { - "thermalBlurFactorNoScope", - "Amount of blur to use when drawing blur through a weapon's thermal scope.", - generate_hash("thermalBlurFactorNoScope") - }, - { - "thermalBlurFactorScope", - "Amount of blur to use when drawing blur through a weapon's thermal scope.", - generate_hash("thermalBlurFactorScope") - }, - { - "timescale", - "Set the game speed.", - generate_hash("timescale") - }, - { - "tokensEnabled", - "Is token economy enabled", - generate_hash("tokensEnabled") - }, - { - "tracer_explosiveColor1", - "The 1st color of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveColor1") - }, - { - "tracer_explosiveColor2", - "The 2nd color of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveColor2") - }, - { - "tracer_explosiveColor3", - "The 3rd color of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveColor3") - }, - { - "tracer_explosiveColor4", - "The 4th color of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveColor4") - }, - { - "tracer_explosiveColor5", - "The 5th color of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveColor5") - }, - { - "tracer_explosiveOverride", - "When turned on, will apply an override to the tracer settings when shooting explosive bullets.", - generate_hash("tracer_explosiveOverride") - }, - { - "tracer_explosiveWidth", - "The width of a bullet tracer when using explosive bullets", - generate_hash("tracer_explosiveWidth") - }, - { - "tracer_firstPersonMaxWidth", - "The maximum width our OWN tracers can be when looking through our ADS", - generate_hash("tracer_firstPersonMaxWidth") - }, - { - "tracer_stoppingPowerColor1", - "The 1st color of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerColor1") - }, - { - "tracer_stoppingPowerColor2", - "The 2nd color of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerColor2") - }, - { - "tracer_stoppingPowerColor3", - "The 3rd color of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerColor3") - }, - { - "tracer_stoppingPowerColor4", - "The 4th color of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerColor4") - }, - { - "tracer_stoppingPowerColor5", - "The 5th color of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerColor5") - }, - { - "tracer_stoppingPowerOverride", - "When turned on, will apply an override to the tracer settings when shooting explosive bullets.", - generate_hash("tracer_stoppingPowerOverride") - }, - { - "tracer_stoppingPowerWidth", - "The width of a bullet tracer when using explosive bullets", - generate_hash("tracer_stoppingPowerWidth") - }, - { - "tracer_thermalWidthMult", - "The multiplier applied to the base width when viewed in thermal vision", - generate_hash("tracer_thermalWidthMult") - }, - { - "transients_verbose", - "Verbose logging information for transient fastfiles.", - generate_hash("transients_verbose") - }, - { - "triggerDLCEnumerationOnSocialConfigLoad", - "Triggers a new DLC enumeration after social config has loaded.", - generate_hash("triggerDLCEnumerationOnSocialConfigLoad") - }, - { - "ui_allow_controlschange", - "", - generate_hash("ui_allow_controlschange") - }, - { - "ui_allow_teamchange", - "", - generate_hash("ui_allow_teamchange") - }, - { - "ui_autodetectGamepad", - "undefined", - generate_hash("ui_autodetectGamepad") - }, - { - "ui_autodetectGamepadDone", - "undefined", - generate_hash("ui_autodetectGamepadDone") - }, - { - "ui_bigFont", - "Big font scale", - generate_hash("ui_bigFont") - }, - { - "ui_blurAmount", - "Max amount to blur background menu items.", - generate_hash("ui_blurAmount") - }, - { - "ui_blurDarkenAmount", - "Amount to darken blurred UI.", - generate_hash("ui_blurDarkenAmount") - }, - { - "ui_blurTime", - "Time in milliseconds to fade in/out the blur.", - generate_hash("ui_blurTime") - }, - { - "ui_borderLowLightScale", - "Scales the border color for the lowlight color on certain UI borders", - generate_hash("ui_borderLowLightScale") - }, - { - "ui_browserFriendlyfire", - "Friendly fire is active", - generate_hash("ui_browserFriendlyfire") - }, - { - "ui_browserKillcam", - "Kill cam is active", - generate_hash("ui_browserKillcam") - }, - { - "ui_browserMod", - "UI Mod value", - generate_hash("ui_browserMod") - }, - { - "ui_browserShowDedicated", - "Show dedicated servers only", - generate_hash("ui_browserShowDedicated") - }, - { - "ui_browserShowEmpty", - "Show empty servers", - generate_hash("ui_browserShowEmpty") - }, - { - "ui_browserShowFull", - "Show full servers", - generate_hash("ui_browserShowFull") - }, - { - "ui_browserShowPassword", - "Show servers that are password protected", - generate_hash("ui_browserShowPassword") - }, - { - "ui_browserShowPure", - "Show pure servers only", - generate_hash("ui_browserShowPure") - }, - { - "ui_buildLocation", - "Where to draw the build number", - generate_hash("ui_buildLocation") - }, - { - "ui_buildSize", - "Font size to use for the build number", - generate_hash("ui_buildSize") - }, - { - "ui_challenge_1_ref", - "", - generate_hash("ui_challenge_1_ref") - }, - { - "ui_challenge_2_ref", - "", - generate_hash("ui_challenge_2_ref") - }, - { - "ui_challenge_3_ref", - "", - generate_hash("ui_challenge_3_ref") - }, - { - "ui_challenge_4_ref", - "", - generate_hash("ui_challenge_4_ref") - }, - { - "ui_challenge_5_ref", - "", - generate_hash("ui_challenge_5_ref") - }, - { - "ui_challenge_6_ref", - "", - generate_hash("ui_challenge_6_ref") - }, - { - "ui_challenge_7_ref", - "", - generate_hash("ui_challenge_7_ref") - }, - { - "ui_changeclass_menu_open", - "", - generate_hash("ui_changeclass_menu_open") - }, - { - "ui_changeteam_menu_open", - "", - generate_hash("ui_changeteam_menu_open") - }, - { - "ui_cinematicsTimestamp", - "Shows cinematics timestamp on subtitle UI elements.", - generate_hash("ui_cinematicsTimestamp") - }, - { - "ui_class_menu_open", - "", - generate_hash("ui_class_menu_open") - }, - { - "ui_connectScreenTextGlowColor", - "Glow color applied to the mode and map name strings on the connect screen.", - generate_hash("ui_connectScreenTextGlowColor") - }, - { - "ui_contextualMenuLocation", - "Contextual menu location from where you entered the store.", - generate_hash("ui_contextualMenuLocation") - }, - { - "ui_controls_menu_open", - "", - generate_hash("ui_controls_menu_open") - }, - { - "ui_currentFeederMapIndex", - "Currently selected map", - generate_hash("ui_currentFeederMapIndex") - }, - { - "ui_currentMap", - "Current map index", - generate_hash("ui_currentMap") - }, - { - "ui_customClassName", - "Custom Class name", - generate_hash("ui_customClassName") - }, - { - "ui_customModeEditName", - "Name to give the currently edited custom game mode when editing is complete", - generate_hash("ui_customModeEditName") - }, - { - "ui_customModeName", - "Custom game mode name", - generate_hash("ui_customModeName") - }, - { - "ui_danger_team", - "", - generate_hash("ui_danger_team") - }, - { - "ui_debugMode", - "Draw ui debug info on the screen.", - generate_hash("ui_debugMode") - }, - { - "ui_disableInGameStore", - "This will disable the ingame store button on the xbox live menu.", - generate_hash("ui_disableInGameStore") - }, - { - "ui_disableTokenRedemption", - "This will disable the token redemption option in the in-game store menu.", - generate_hash("ui_disableTokenRedemption") - }, - { - "ui_drawCrosshair", - "Whether to draw crosshairs.", - generate_hash("ui_drawCrosshair") - }, - { - "ui_editSquadMemberIndex", - "Which squad member is currently being edited", - generate_hash("ui_editSquadMemberIndex") - }, - { - "ui_extraBigFont", - "Extra big font scale", - generate_hash("ui_extraBigFont") - }, - { - "ui_game_state", - "", - generate_hash("ui_game_state") - }, - { - "ui_gametype", - "Current game type", - generate_hash("ui_gametype") - }, - { - "ui_halftime", - "", - generate_hash("ui_halftime") - }, - { - "ui_hitloc_0", - "", - generate_hash("ui_hitloc_0") - }, - { - "ui_hitloc_1", - "", - generate_hash("ui_hitloc_1") - }, - { - "ui_hitloc_2", - "", - generate_hash("ui_hitloc_2") - }, - { - "ui_hitloc_3", - "", - generate_hash("ui_hitloc_3") - }, - { - "ui_hitloc_4", - "", - generate_hash("ui_hitloc_4") - }, - { - "ui_hitloc_5", - "", - generate_hash("ui_hitloc_5") - }, - { - "ui_hitloc_damage_0", - "", - generate_hash("ui_hitloc_damage_0") - }, - { - "ui_hitloc_damage_1", - "", - generate_hash("ui_hitloc_damage_1") - }, - { - "ui_hitloc_damage_2", - "", - generate_hash("ui_hitloc_damage_2") - }, - { - "ui_hitloc_damage_3", - "", - generate_hash("ui_hitloc_damage_3") - }, - { - "ui_hitloc_damage_4", - "", - generate_hash("ui_hitloc_damage_4") - }, - { - "ui_hitloc_damage_5", - "", - generate_hash("ui_hitloc_damage_5") - }, - { - "ui_hud_hardcore", - "Whether the HUD should be suppressed for hardcore mode", - generate_hash("ui_hud_hardcore") - }, - { - "ui_hud_obituaries", - "", - generate_hash("ui_hud_obituaries") - }, - { - "ui_inactiveBaseColor", - "The local player's rank/stats font color when shown in lobbies and parties", - generate_hash("ui_inactiveBaseColor") - }, - { - "ui_inactivePartyColor", - "", - generate_hash("ui_inactivePartyColor") - }, - { - "ui_inGameStoreOpen", - "is the InGameStore open", - generate_hash("ui_inGameStoreOpen") - }, - { - "ui_inhostmigration", - "", - generate_hash("ui_inhostmigration") - }, - { - "ui_joinGametype", - "Game join type", - generate_hash("ui_joinGametype") - }, - { - "ui_loadMenuName", - "Frontend menu will start on this level instead of lockout", - generate_hash("ui_loadMenuName") - }, - { - "ui_mapname", - "Current map name", - generate_hash("ui_mapname") - }, - { - "ui_mapvote_entrya_gametype", - "Primary map vote entry game type", - generate_hash("ui_mapvote_entrya_gametype") - }, - { - "ui_mapvote_entrya_mapname", - "Primary map vote entry map name", - generate_hash("ui_mapvote_entrya_mapname") - }, - { - "ui_mapvote_entryb_gametype", - "Secondary map vote entry game type", - generate_hash("ui_mapvote_entryb_gametype") - }, - { - "ui_mapvote_entryb_mapname", - "Secondary map vote entry map name", - generate_hash("ui_mapvote_entryb_mapname") - }, - { - "ui_maxclients", - "undefined", - generate_hash("ui_maxclients") - }, - { - "ui_missingMapName", - "Name of map to show in missing content error", - generate_hash("ui_missingMapName") - }, - { - "ui_mousePitch", - "", - generate_hash("ui_mousePitch") - }, - { - "ui_multiplayer", - "True if the game is multiplayer", - generate_hash("ui_multiplayer") - }, - { - "ui_myPartyColor", - "Player name font color when in the same party as the local player", - generate_hash("ui_myPartyColor") - }, - { - "ui_netGametype", - "Game type", - generate_hash("ui_netGametype") - }, - { - "ui_netGametypeName", - "Displayed game type name", - generate_hash("ui_netGametypeName") - }, - { - "ui_netSource", - "The network source where:\n 0:Local\n 1:Internet\n 2:Favourites", - generate_hash("ui_netSource") - }, - { - "ui_onlineRequired", - "UI requires online connection to be present.", - generate_hash("ui_onlineRequired") - }, - { - "ui_opensummary", - "", - generate_hash("ui_opensummary") - }, - { - "ui_override_halftime", - "", - generate_hash("ui_override_halftime") - }, - { - "ui_partyFull", - "True if the current party is full.", - generate_hash("ui_partyFull") - }, - { - "ui_playerPartyColor", - "", - generate_hash("ui_playerPartyColor") - }, - { - "ui_playlistActionButtonAlpha", - "The current alpha of the playlist selection button", - generate_hash("ui_playlistActionButtonAlpha") - }, - { - "ui_playlistCategoryDisabledColor", - "The color of playlist categories when disabled", - generate_hash("ui_playlistCategoryDisabledColor") - }, - { - "ui_playlistCategoryEnabledColor", - "The color of playlist categories when enabled", - generate_hash("ui_playlistCategoryEnabledColor") - }, - { - "ui_promotion", - "", - generate_hash("ui_promotion") - }, - { - "ui_remoteTankUseTime", - "", - generate_hash("ui_remoteTankUseTime") - }, - { - "ui_scorelimit", - "", - generate_hash("ui_scorelimit") - }, - { - "ui_selectedFeederMap", - "Current preview game type", - generate_hash("ui_selectedFeederMap") - }, - { - "ui_serverStatusTimeOut", - "Time in milliseconds before a server status request times out", - generate_hash("ui_serverStatusTimeOut") - }, - { - "ui_show_store", - "Use to enable the store button", - generate_hash("ui_show_store") - }, - { - "ui_showDLCMaps", - "Whether to display the DLC maps.", - generate_hash("ui_showDLCMaps") - }, - { - "ui_showInfo", - "", - generate_hash("ui_showInfo") - }, - { - "ui_showList", - "Show onscreen list of currently visible menus", - generate_hash("ui_showList") - }, - { - "ui_showmap", - "", - generate_hash("ui_showmap") - }, - { - "ui_showMenuOnly", - "If set, only menus using this name will draw.", - generate_hash("ui_showMenuOnly") - }, - { - "ui_showMinimap", - "", - generate_hash("ui_showMinimap") - }, - { - "ui_sliderSteps", - "The number of steps for a slider itemdef", - generate_hash("ui_sliderSteps") - }, - { - "ui_smallFont", - "Small font scale", - generate_hash("ui_smallFont") - }, - { - "ui_textScrollFadeTime", - "Text scrolling takes this long (seconds) to fade out at the end before restarting", - generate_hash("ui_textScrollFadeTime") - }, - { - "ui_textScrollPauseEnd", - "Text scrolling waits this long (seconds) before starting", - generate_hash("ui_textScrollPauseEnd") - }, - { - "ui_textScrollPauseStart", - "Text scrolling waits this long (seconds) before starting", - generate_hash("ui_textScrollPauseStart") - }, - { - "ui_textScrollSpeed", - "Speed at which text scrolls vertically", - generate_hash("ui_textScrollSpeed") - }, - { - "ui_timelimit", - "", - generate_hash("ui_timelimit") - }, - { - "uiscript_debug", - "spam debug info for the ui script", - generate_hash("uiscript_debug") - }, - { - "unlock_breadcrumb_killswitch", - "True to enable unlock breadcrumbs", - generate_hash("unlock_breadcrumb_killswitch") - }, - { - "uno_current_tos_version", - "Current Uno Terms of Service Version", - generate_hash("uno_current_tos_version") - }, - { - "use_filtered_query_pass", - "Dictates whether to use the filtered query for MMing or not", - generate_hash("use_filtered_query_pass") - }, - { - "use_weighted_dlc_exactmatch_pass", - "Dictates whether to use a search weighted pass with the DLC match set to exact for MMing or not", - generate_hash("use_weighted_dlc_exactmatch_pass") - }, - { - "use_weighted_pass", - "Dictates whether to use the search weighted pass for MMing or not", - generate_hash("use_weighted_pass") - }, - { - "useCPMarkerForCPOwnership", - "If set, we will check the player inventory to see if he owns the redeemedItem for a contentPack if this contentPack is not available for the player", - generate_hash("useCPMarkerForCPOwnership") - }, - { - "useonlinestats", - "Whether to use online stats when in offline modes", - generate_hash("useonlinestats") - }, - { - "useRelativeTeamColors", - "Whether to use relative team colors.", - generate_hash("useRelativeTeamColors") - }, - { - "userFileFetchTimeout", - "default timeout for user files FETCH tasks (in seconds)", - generate_hash("userFileFetchTimeout") - }, - { - "userGroup_active", - "Are we allowed to show Usergroups or not", - generate_hash("userGroup_active") - }, - { - "userGroup_cool_off_time", - "Cool off time between calls to fetch the elite clan", - generate_hash("userGroup_cool_off_time") - }, - { - "userGroup_coop_delay", - "Delay between a player joining a coop lobby and the DW user group task starting", - generate_hash("userGroup_coop_delay") - }, - { - "userGroup_max_retry_time", - "Max time that the usergroup read find can retry", - generate_hash("userGroup_max_retry_time") - }, - { - "userGroup_refresh_time_secs", - "Time in seconds between re-sending lobby group data to confirmed users.", - generate_hash("userGroup_refresh_time_secs") - }, - { - "userGroup_retry_step", - "Step in m/s for the usegroup read retry", - generate_hash("userGroup_retry_step") - }, - { - "userGroup_RetryTime", - "Time in ms between sending lobby group data retrys.", - generate_hash("userGroup_RetryTime") - }, - { - "useStatsGroups", - "If true then StatsGroups are in use for all playerdata.ddl accessing.", - generate_hash("useStatsGroups") - }, - { - "useTagFlashSilenced", - "When true, silenced weapons will use \"tag_flash_silenced\" instead of \"tag_flash\".", - generate_hash("useTagFlashSilenced") - }, - { - "using_mlg", - "MLG feature on/off", - generate_hash("using_mlg") - }, - { - "validate_apply_clamps", - "True if individual stat validation failure reverts the value", - generate_hash("validate_apply_clamps") - }, - { - "validate_apply_revert", - "True if individual stat validation failure reverts the value", - generate_hash("validate_apply_revert") - }, - { - "validate_apply_revert_full", - "True if any individual stat validation failure causes a full stats revert", - generate_hash("validate_apply_revert_full") - }, - { - "validate_clamp_assists", - "The maximum number of assists a player can make in a match", - generate_hash("validate_clamp_assists") - }, - { - "validate_clamp_experience", - "The maximum experience a player can gain in a match", - generate_hash("validate_clamp_experience") - }, - { - "validate_clamp_headshots", - "The maximum number of headshots a player can make in a match", - generate_hash("validate_clamp_headshots") - }, - { - "validate_clamp_hits", - "The maximum number of hits player can make in a match", - generate_hash("validate_clamp_hits") - }, - { - "validate_clamp_kills", - "The maximum number of kills a player can make in a match", - generate_hash("validate_clamp_kills") - }, - { - "validate_clamp_losses", - "The maximum number of losses a player can make in a match", - generate_hash("validate_clamp_losses") - }, - { - "validate_clamp_misses", - "The maximum number of misses player can make in a match", - generate_hash("validate_clamp_misses") - }, - { - "validate_clamp_ties", - "The maximum number of ties a player can make in a match", - generate_hash("validate_clamp_ties") - }, - { - "validate_clamp_totalshots", - "The maximum number of totalshots player can make in a match", - generate_hash("validate_clamp_totalshots") - }, - { - "validate_clamp_weaponXP", - "The maximum experience a weapon can gain in a match", - generate_hash("validate_clamp_weaponXP") - }, - { - "validate_clamp_wins", - "The maximum number of wins a player can make in a match", - generate_hash("validate_clamp_wins") - }, - { - "validate_drop_on_fail", - "True if stats validation failure results in dropping from the match", - generate_hash("validate_drop_on_fail") - }, - { - "veh_aiOverSteerScale", - "Scaler used to cause ai vehicles to over steer", - generate_hash("veh_aiOverSteerScale") - }, - { - "veh_boneControllerLodDist", - "Distance at which bone controllers are not updated.", - generate_hash("veh_boneControllerLodDist") - }, - { - "veh_boneControllerUnLodDist", - "Distance at which bone controllers start updating when not moving.", - generate_hash("veh_boneControllerUnLodDist") - }, - { - "vehAudio_inAirPitchDownLerp", - "Rate at which the pitch lerps down", - generate_hash("vehAudio_inAirPitchDownLerp") - }, - { - "vehAudio_inAirPitchUpLerp", - "Rate at which the pitch lerps up", - generate_hash("vehAudio_inAirPitchUpLerp") - }, - { - "vehAudio_spawnVolumeTime", - "Seconds it takes for spawned vehicles to reach full volume.", - generate_hash("vehAudio_spawnVolumeTime") - }, - { - "vehCam_freeLook", - "Enables free look mode", - generate_hash("vehCam_freeLook") - }, - { - "vehCam_mode", - "Camera modes: 1st person, 3rd person, or both", - generate_hash("vehCam_mode") - }, - { - "vehDroneDebugDrawPath", - "Debug render the drone draw paths.", - generate_hash("vehDroneDebugDrawPath") - }, - { - "vehHelicopterBoundsRadius", - "The radius of the collision volume to be used when colliding with world geometry.", - generate_hash("vehHelicopterBoundsRadius") - }, - { - "vehHelicopterDecelerationFwd", - "Set the deceleration of the player helicopter (as a fraction of acceleration) in the direction the chopper is facing. So 1.0 makes it equal to the acceleration.", - generate_hash("vehHelicopterDecelerationFwd") - }, - { - "vehHelicopterDecelerationSide", - "Set the side-to-side deceleration of the player helicopter (as a fraction of acceleration). So 1.0 makes it equal to the acceleration.", - generate_hash("vehHelicopterDecelerationSide") - }, - { - "vehHelicopterDecelerationUp", - "Set the vertical deceleration of the player helicopter (as a fraction of acceleration). So 1.0 makes it equal to the acceleration.", - generate_hash("vehHelicopterDecelerationUp") - }, - { - "vehHelicopterHeadSwayDontSwayTheTurret", - "If set, the turret will not fire through the crosshairs, but straight ahead of the vehicle, when the player is not freelooking.", - generate_hash("vehHelicopterHeadSwayDontSwayTheTurret") - }, - { - "vehHelicopterHoverSpeedThreshold", - "The speed below which the player helicopter begins to jitter the tilt, for hovering", - generate_hash("vehHelicopterHoverSpeedThreshold") - }, - { - "vehHelicopterInvertUpDown", - "Invert the altitude control on the player helicopter.", - generate_hash("vehHelicopterInvertUpDown") - }, - { - "vehHelicopterJitterJerkyness", - "Specifies how jerky the tilt jitter should be", - generate_hash("vehHelicopterJitterJerkyness") - }, - { - "vehHelicopterLookaheadTime", - "How far ahead (in seconds) the player helicopter looks ahead, to avoid hard collisions. (Like driving down the highway, you should keep 2 seconds distance between you and the vehicle in front of you)", - generate_hash("vehHelicopterLookaheadTime") - }, - { - "vehHelicopterMaxAccel", - "Maximum horizontal acceleration of the player helicopter (in MPH per second)", - generate_hash("vehHelicopterMaxAccel") - }, - { - "vehHelicopterMaxAccelVertical", - "Maximum vertical acceleration of the player helicopter (in MPH per second)", - generate_hash("vehHelicopterMaxAccelVertical") - }, - { - "vehHelicopterMaxPitch", - "Maximum pitch of the player helicopter", - generate_hash("vehHelicopterMaxPitch") - }, - { - "vehHelicopterMaxRoll", - "Maximum roll of the player helicopter", - generate_hash("vehHelicopterMaxRoll") - }, - { - "vehHelicopterMaxSpeed", - "Maximum horizontal speed of the player helicopter (in MPH)", - generate_hash("vehHelicopterMaxSpeed") - }, - { - "vehHelicopterMaxSpeedVertical", - "Maximum vertical speed of the player helicopter (in MPH)", - generate_hash("vehHelicopterMaxSpeedVertical") - }, - { - "vehHelicopterMaxYawAccel", - "Maximum yaw acceleration of the player helicopter", - generate_hash("vehHelicopterMaxYawAccel") - }, - { - "vehHelicopterMaxYawRate", - "Maximum yaw speed of the player helicopter", - generate_hash("vehHelicopterMaxYawRate") - }, - { - "vehHelicopterPitchOffset", - "The resting pitch of the helicopter", - generate_hash("vehHelicopterPitchOffset") - }, - { - "vehHelicopterRightStickDeadzone", - "Dead-zone for the axes of the right thumbstick. This helps to better control the two axes separately.", - generate_hash("vehHelicopterRightStickDeadzone") - }, - { - "vehHelicopterScaleMovement", - "Scales down the smaller of the left stick axes.", - generate_hash("vehHelicopterScaleMovement") - }, - { - "vehHelicopterSoftCollisions", - "Player helicopters have soft collisions (slow down before they collide).", - generate_hash("vehHelicopterSoftCollisions") - }, - { - "vehHelicopterStrafeDeadzone", - "Dead-zone so that you can fly straight forward easily without accidentally strafing (and thus rolling).", - generate_hash("vehHelicopterStrafeDeadzone") - }, - { - "vehHelicopterTiltFromAcceleration", - "The amount of tilt caused by acceleration", - generate_hash("vehHelicopterTiltFromAcceleration") - }, - { - "vehHelicopterTiltFromControllerAxes", - "The amount of tilt caused by the desired velocity (i.e., the amount of controller stick deflection)", - generate_hash("vehHelicopterTiltFromControllerAxes") - }, - { - "vehHelicopterTiltFromDeceleration", - "The amount of tilt caused by deceleration", - generate_hash("vehHelicopterTiltFromDeceleration") - }, - { - "vehHelicopterTiltFromFwdAndYaw", - "The amount of roll caused by yawing while moving forward.", - generate_hash("vehHelicopterTiltFromFwdAndYaw") - }, - { - "vehHelicopterTiltFromFwdAndYaw_VelAtMaxTilt", - "The forward speed (as a fraction of top speed) at which the tilt due to yaw reaches is maximum value.", - generate_hash("vehHelicopterTiltFromFwdAndYaw_VelAtMaxTilt") - }, - { - "vehHelicopterTiltFromVelocity", - "The amount of tilt caused by the current velocity", - generate_hash("vehHelicopterTiltFromVelocity") - }, - { - "vehHelicopterTiltMomentum", - "The amount of rotational momentum the helicopter has with regards to tilting.", - generate_hash("vehHelicopterTiltMomentum") - }, - { - "vehHelicopterTiltSpeed", - "The rate at which the player helicopter's tilt responds", - generate_hash("vehHelicopterTiltSpeed") - }, - { - "vehHelicopterYawOnLeftStick", - "The yaw speed created by the left stick when pushing the stick diagonally (e.g., moving forward and strafing slightly).", - generate_hash("vehHelicopterYawOnLeftStick") - }, - { - "vehicle_debug_render_spline_plane", - "Do we want to render the spline plane data", - generate_hash("vehicle_debug_render_spline_plane") - }, - { - "vehicle_pathsmooth", - "Smoothed vehicle pathing.", - generate_hash("vehicle_pathsmooth") - }, - { - "vehUGVPitchTrack", - "UGV body pitch orientation speed", - generate_hash("vehUGVPitchTrack") - }, - { - "vehUGVRollTrack", - "UGV body roll orientation speed", - generate_hash("vehUGVRollTrack") - }, - { - "vehUGVWheelInfluence", - "UGV wheel influence on the orientation of the body", - generate_hash("vehUGVWheelInfluence") - }, - { - "vehWalkerControlMode", - "Walker controls (0==move no turn, 1=move and turn, 2=move relative(tank))", - generate_hash("vehWalkerControlMode") - }, - { - "version", - "Game version", - generate_hash("version") - }, - { - "vid_xpos", - "Game window horizontal position", - generate_hash("vid_xpos") - }, - { - "vid_ypos", - "Game window vertical position", - generate_hash("vid_ypos") - }, - { - "viewangNow", - "", - generate_hash("viewangNow") - }, - { - "viewModelDebugNotetracks", - "Enable display of viewmodel notetrack debug info.", - generate_hash("viewModelDebugNotetracks") - }, - { - "viewModelHacks", - "Enabled depth hack and remove viewmodel from shadows.", - generate_hash("viewModelHacks") - }, - { - "viewposNow", - "", - generate_hash("viewposNow") - }, - { - "virtualLobbyActive", - "Indicates the VL is actively being displayed.", - generate_hash("virtualLobbyActive") - }, - { - "virtualLobbyAllocated", - "Indicates the first VL zone has been loaded.", - generate_hash("virtualLobbyAllocated") - }, - { - "virtualLobbyEnabled", - "VirtualLobby is enabled (must be true before loading UI zone)", - generate_hash("virtualLobbyEnabled") - }, - { - "virtualLobbyInFiringRange", - "VirtualLobby is in firing range mode", - generate_hash("virtualLobbyInFiringRange") - }, - { - "virtualLobbyMap", - "VirtualLobby map to load (must be set before starting vl)", - generate_hash("virtualLobbyMap") - }, - { - "virtualLobbyMembers", - "Number of members in the VirtualLobby (set by script)", - generate_hash("virtualLobbyMembers") - }, - { - "virtualLobbyPresentable", - "Indicates to LUA the VirtualLobby is ready to be displayed (set by script).", - generate_hash("virtualLobbyPresentable") - }, - { - "virtualLobbyReady", - "Indicates to LUA the VirtualLobby is loaded and running (set by script).", - generate_hash("virtualLobbyReady") - }, - { - "vl_clan_models_loaded", - "Indicates to LUA when all models are loaded for the clan highlights so it can begin the fade-in without any popping(set by script).", - generate_hash("vl_clan_models_loaded") - }, - { - "voMtxEnable", - "When set (e.g. via config), will enable voice over packs", - generate_hash("voMtxEnable") - }, - { - "waypointAerialIconMaxSize", - "Max size of aerial targeting waypoints.", - generate_hash("waypointAerialIconMaxSize") - }, - { - "waypointAerialIconMinSize", - "Min size of aerial targeting waypoints.", - generate_hash("waypointAerialIconMinSize") - }, - { - "waypointAerialIconScale", - "Base scale of aerial targeting waypoints.", - generate_hash("waypointAerialIconScale") - }, - { - "waypointDebugDraw", - "", - generate_hash("waypointDebugDraw") - }, - { - "waypointDistScaleRangeMax", - "Distance from player that icon distance scaling ends.", - generate_hash("waypointDistScaleRangeMax") - }, - { - "waypointDistScaleRangeMin", - "Distance from player that icon distance scaling ends.", - generate_hash("waypointDistScaleRangeMin") - }, - { - "waypointDistScaleSmallest", - "Smallest scale that the distance effect uses.", - generate_hash("waypointDistScaleSmallest") - }, - { - "waypointIconHeight", - "", - generate_hash("waypointIconHeight") - }, - { - "waypointIconWidth", - "", - generate_hash("waypointIconWidth") - }, - { - "waypointOffscreenCornerRadius", - "Size of the rounded corners.", - generate_hash("waypointOffscreenCornerRadius") - }, - { - "waypointOffscreenDistanceThresholdAlpha", - "Distance from the threshold over which offscreen objective icons lerp their alpha.", - generate_hash("waypointOffscreenDistanceThresholdAlpha") - }, - { - "waypointOffscreenPadBottom", - "", - generate_hash("waypointOffscreenPadBottom") - }, - { - "waypointOffscreenPadLeft", - "", - generate_hash("waypointOffscreenPadLeft") - }, - { - "waypointOffscreenPadRight", - "", - generate_hash("waypointOffscreenPadRight") - }, - { - "waypointOffscreenPadTop", - "", - generate_hash("waypointOffscreenPadTop") - }, - { - "waypointOffscreenPointerDistance", - "Distance from the center of the offscreen objective icon to the center its arrow.", - generate_hash("waypointOffscreenPointerDistance") - }, - { - "waypointOffscreenPointerHeight", - "", - generate_hash("waypointOffscreenPointerHeight") - }, - { - "waypointOffscreenPointerWidth", - "", - generate_hash("waypointOffscreenPointerWidth") - }, - { - "waypointOffscreenRoundedCorners", - "Off-screen icons take rounded corners when true. 90-degree corners when false.", - generate_hash("waypointOffscreenRoundedCorners") - }, - { - "waypointOffscreenScaleLength", - "How far the offscreen icon scale travels from full to smallest scale.", - generate_hash("waypointOffscreenScaleLength") - }, - { - "waypointOffscreenScaleSmallest", - "Smallest scale that the offscreen effect uses.", - generate_hash("waypointOffscreenScaleSmallest") - }, - { - "waypointPlayerOffsetCrouch", - "For waypoints pointing to players, how high to offset off of their origin when they are prone.", - generate_hash("waypointPlayerOffsetCrouch") - }, - { - "waypointPlayerOffsetProne", - "For waypoints pointing to players, how high to offset off of their origin when they are prone.", - generate_hash("waypointPlayerOffsetProne") - }, - { - "waypointPlayerOffsetStand", - "For waypoints pointing to players, how high to offset off of their origin when they are prone.", - generate_hash("waypointPlayerOffsetStand") - }, - { - "waypointScreenCenterFadeAdsMin", - "When 'waypointScreenCenterFadeRadius' enabled, minimum amount that waypoint will fade when in ads", - generate_hash("waypointScreenCenterFadeAdsMin") - }, - { - "waypointScreenCenterFadeHipMin", - "When 'waypointScreenCenterFadeRadius' enabled, minimum amount that waypoint will fade when in ads", - generate_hash("waypointScreenCenterFadeHipMin") - }, - { - "waypointScreenCenterFadeRadius", - "Radius from screen center that a waypoint will start fading out. Setting to 0 will turn this off", - generate_hash("waypointScreenCenterFadeRadius") - }, - { - "waypointSplitscreenScale", - "Scale applied to waypoint icons in splitscreen views.", - generate_hash("waypointSplitscreenScale") - }, - { - "waypointTweakY", - "", - generate_hash("waypointTweakY") - }, - { - "weap_thermoDebuffMod", - "", - generate_hash("weap_thermoDebuffMod") - }, - { - "wideScreen", - "True if the game video is running in 16x9 aspect, false if 4x3.", - generate_hash("wideScreen") - }, - { - "winvoice_loopback", - "Echo microphone input locally", - generate_hash("winvoice_loopback") - }, - { - "winvoice_mic_mute", - "Mute the microphone", - generate_hash("winvoice_mic_mute") - }, - { - "winvoice_mic_outTime", - "Microphone voice amount of silence before we cut the mic", - generate_hash("winvoice_mic_outTime") - }, - { - "winvoice_mic_reclevel", - "Microphone recording level", - generate_hash("winvoice_mic_reclevel") - }, - { - "winvoice_mic_scaler", - "Microphone scaler value", - generate_hash("winvoice_mic_scaler") - }, - { - "winvoice_mic_threshold", - "Microphone voice threshold", - generate_hash("winvoice_mic_threshold") - }, - { - "winvoice_save_voice", - "Write voice data to a file", - generate_hash("winvoice_save_voice") - }, - { - "xanim_disableIK", - "Disable inverse kinematics solvers", - generate_hash("xanim_disableIK") - }, - { - "xblive_competitionmatch", - "MLG Rules?", - generate_hash("xblive_competitionmatch") - }, - { - "xblive_hostingprivateparty", - "true only if we're hosting a party", - generate_hash("xblive_hostingprivateparty") - }, - { - "xblive_loggedin", - "User is logged into xbox live", - generate_hash("xblive_loggedin") - }, - { - "xblive_privatematch", - "Current game is a private match", - generate_hash("xblive_privatematch") - }, - { - "xblive_privatematch_solo", - "Current game is an Extinction solo match", - generate_hash("xblive_privatematch_solo") - }, - { - "xphys_maxJointPositionError", - "If a joints with position error exceeding this value is detected, then the whole xphys system gets snapped back to the animation pose", - generate_hash("xphys_maxJointPositionError") - }, - { - "aa_player_kills", - "Player kills", - generate_hash("aa_player_kills") - }, - { - "aa_player_damage_dealt", - "Player damages dealt", - generate_hash("aa_player_damage_dealt") - }, - { - "aa_ads_damage_dealt", - "Player damages dealt in ads", - generate_hash("aa_ads_damage_dealt") - }, - { - "aa_deaths", - "Player deaths", - generate_hash("aa_deaths") - }, - { - "aa_time_tracking", - "Time in game", - generate_hash("aa_time_tracking") - }, - }; + dvar_info info{}; + info.hash = hash; + info.name = name; + info.description = description; + dvar_map.insert(std::make_pair(hash, info)); + } + + void insert_dvar_info(const std::string& name, const std::string& description) + { + insert_dvar_info(generate_hash(name), name, description); + } + + std::optional get_dvar_info_from_hash(const std::int32_t hash) + { + const auto iter = dvar_map.find(hash); + if (iter != dvar_map.end()) + { + return iter->second; + } + + return {}; + } + + std::optional get_dvar_info(const std::string& name) + { + const auto hash = generate_hash(name); + return get_dvar_info_from_hash(hash); + } std::string dvar_get_description(const std::string& name) { - const auto lower = utils::string::to_lower(name); - for (std::uint32_t i = 0; i < dvar_list.size(); i++) + const auto info = get_dvar_info(name); + if (info.has_value()) { - if (utils::string::to_lower(dvar_list[i].name) == lower) - { - return dvar_list[i].description; - } + return info->description; } return {}; } - bool can_add_dvar_to_list(std::string name) - { - const auto lower = utils::string::to_lower(name); - for (std::uint32_t i = 0; i < dvar_list.size(); i++) - { - if (utils::string::to_lower(dvar_list[i].name) == lower) - { - return false; - } - } - - return true; - } - - std::optional get_dvar_info_from_hash(const int hash) - { - for (std::uint32_t i = 0; i < dvar_list.size(); i++) - { - if (dvar_list[i].hash == hash) - { - return {dvar_list[i]}; - } - } - - return {}; - } - - std::string hash_to_string(const int hash) + std::string hash_to_string(const std::int32_t hash) { return utils::string::va("0x%lX", hash); } @@ -11872,78 +233,48 @@ namespace dvars game::dvar_t* register_int(const std::string& name, int value, int min, int max, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({name, description}); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterInt(hash, "", value, min, max, flags); } game::dvar_t* register_bool(const std::string& name, bool value, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({name, description}); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterBool(hash, "", value, flags); } game::dvar_t* register_string(const std::string& name, const char* value, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({name, description}); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterString(hash, "", value, flags); } game::dvar_t* register_float(const std::string& name, float value, float min, float max, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({name, description}); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterFloat(hash, "", value, min, max, flags); } game::dvar_t* register_float_hashed(const std::string& name, float value, float min, float max, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({ name, description }); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterFloatHashed(hash, "", value, min, max, flags); } game::dvar_t* register_vec4(const std::string& name, float x, float y, float z, float w, float min, float max, unsigned int flags, const std::string& description) { - const auto hash = game::generateHashValue(name.data()); - - if (can_add_dvar_to_list(name)) - { - dvar_list.push_back({name, description}); - } - + const auto hash = generate_hash(name); + insert_dvar_info(hash, name, description); return game::Dvar_RegisterVec4(hash, "", x, y, z, w, min, max, flags); } } diff --git a/src/client/game/dvars.hpp b/src/client/game/dvars.hpp index d884877f..42f12ec9 100644 --- a/src/client/game/dvars.hpp +++ b/src/client/game/dvars.hpp @@ -10,9 +10,11 @@ namespace dvars { std::string name; std::string description; - int hash; + std::int32_t hash; }; + extern std::unordered_map dvar_map; + extern game::dvar_t* aimassist_enabled; extern game::dvar_t* con_inputBoxColor; @@ -48,12 +50,57 @@ namespace dvars extern game::dvar_t* cg_legacyCrashHandling; - extern std::vector dvar_list; + constexpr int generate_hash(const char* string) + { + const char* v1; + char v2, v6; + int v4, v5, v7; + char* end_ptr; + + v1 = string; + v2 = *string; + + if (v2 == 48 && v1[1] == 120) + { + return strtoul(v1 + 2, &end_ptr, 16); + } + + v4 = v2; + + if ((v2 - 65) <= 0x19u) + { + v4 = v2 + 32; + } + + v5 = 0xB3CB2E29 * static_cast(v4 ^ 0x319712C3); + + if (v2) + { + do + { + v6 = *++v1; + v7 = v6; + if ((v6 - 65) <= 0x19u) + { + v7 = v6 + 32; + } + + v5 = 0xB3CB2E29 * static_cast(v5 ^ v7); + } while (v6); + } + + return v5; + } + + std::int32_t generate_hash(const std::string& string); + + void insert_dvar_info(const std::int32_t hash, const std::string& name, const std::string& description); + void insert_dvar_info(const std::string& name, const std::string& description); std::string dvar_get_vector_domain(const int components, const game::dvar_limits& domain); std::string dvar_get_domain(const game::dvar_type type, const game::dvar_limits& domain); std::string dvar_get_description(const std::string& name); - std::optional get_dvar_info_from_hash(const int hash); + std::optional get_dvar_info_from_hash(const std::int32_t hash); game::dvar_t* register_int(const std::string& name, int value, int min, int max, unsigned int flags, const std::string& description); diff --git a/src/client/game/scripting/functions.cpp b/src/client/game/scripting/functions.cpp index 165e56a9..daf766ee 100644 --- a/src/client/game/scripting/functions.cpp +++ b/src/client/game/scripting/functions.cpp @@ -1,80 +1,79 @@ #include #include "functions.hpp" -#include "component/console.hpp" -#include "component/gsc/script_extension.hpp" - -#include -#include - #include +#include "component/gsc/script_extension.hpp" +#include "component/gsc/script_loading.hpp" + namespace scripting { namespace { - int find_function_index(const std::string& name, const bool prefer_global) + int find_function_index(const std::string& name, [[maybe_unused]] const bool prefer_global) { const auto target = utils::string::to_lower(name); - auto first = xsk::gsc::h1::resolver::function_id; - auto second = xsk::gsc::h1::resolver::method_id; + auto const& first = gsc::gsc_ctx->func_map(); + auto const& second = gsc::gsc_ctx->meth_map(); + if (!prefer_global) { - std::swap(first, second); + if (const auto itr = second.find(name); itr != second.end()) + { + return static_cast(itr->second); + } + + if (const auto itr = first.find(name); itr != first.end()) + { + return static_cast(itr->second); + } } - const auto first_res = first(target); - if (first_res) + if (const auto itr = first.find(name); itr != first.end()) { - return first_res; + return static_cast(itr->second); } - const auto second_res = second(target); - if (second_res) + if (const auto itr = second.find(name); itr != second.end()) { - return second_res; + return static_cast(itr->second); } return -1; } - - script_function get_function_by_index(const unsigned index) - { - if (index < 0x1000) - { - return reinterpret_cast(gsc::func_table)[index - 1]; - } - - return reinterpret_cast(gsc::meth_table)[index - 0x8000]; - } - - unsigned int parse_token_id(const std::string& name) - { - if (name.starts_with("_ID")) - { - return static_cast(std::strtol(name.substr(3).data(), nullptr, 10)); - } - - if (name.starts_with("_id_")) - { - return static_cast(std::strtol(name.substr(4).data(), nullptr, 16)); - } - - return 0; - } } - std::string find_token(unsigned int id) + std::uint32_t parse_token_id(const std::string& name) { - return xsk::gsc::h1::resolver::token_name(static_cast(id)); + if (name.starts_with("_ID")) + { + return static_cast(std::strtol(name.substr(3).data(), nullptr, 10)); + } + + if (name.starts_with("_id_")) + { + return static_cast(std::strtol(name.substr(4).data(), nullptr, 16)); + } + + return 0; + } + + std::string find_token(std::uint32_t id) + { + return gsc::gsc_ctx->token_name(id); + } + + std::string find_token_single(std::uint32_t id) + { + return gsc::gsc_ctx->token_name(id); } unsigned int find_token_id(const std::string& name) { - const auto result = xsk::gsc::h1::resolver::token_id(name); - if (result) + const auto id = gsc::gsc_ctx->token_id(name); + if (id) { - return result; + return id; } const auto parsed_id = parse_token_id(name); @@ -86,14 +85,24 @@ namespace scripting return game::SL_GetCanonicalString(name.data()); } + script_function get_function_by_index(const std::uint32_t index) + { + static const auto function_table = &gsc::func_table; + static const auto method_table = &gsc::meth_table; + + if (index < 0x1000) + { + return reinterpret_cast(function_table)[index - 1]; + } + + return reinterpret_cast(method_table)[index - 0x8000]; + } + script_function find_function(const std::string& name, const bool prefer_global) { const auto index = find_function_index(name, prefer_global); - if (index < 0) - { - return nullptr; - } + if (index < 0) return nullptr; return get_function_by_index(index); } -} +} \ No newline at end of file diff --git a/src/client/game/scripting/functions.hpp b/src/client/game/scripting/functions.hpp index 0685a92e..5bd1c764 100644 --- a/src/client/game/scripting/functions.hpp +++ b/src/client/game/scripting/functions.hpp @@ -5,8 +5,10 @@ namespace scripting { using script_function = void(*)(game::scr_entref_t); - std::string find_token(unsigned int id); + std::string find_token(std::uint32_t id); + std::string find_token_single(std::uint32_t id); unsigned int find_token_id(const std::string& name); + script_function get_function_by_index(std::uint32_t index); script_function find_function(const std::string& name, const bool prefer_global); } diff --git a/src/client/game/scripting/lua/context.cpp b/src/client/game/scripting/lua/context.cpp index 8a719e0e..8c483551 100644 --- a/src/client/game/scripting/lua/context.cpp +++ b/src/client/game/scripting/lua/context.cpp @@ -11,8 +11,7 @@ #include "component/fastfiles.hpp" #include "component/scheduler.hpp" -#include -#include +#include "component/gsc/script_loading.hpp" #include #include @@ -217,7 +216,7 @@ namespace scripting::lua auto entity_type = state.new_usertype("entity"); - for (const auto& func : xsk::gsc::h1::resolver::get_methods()) + for (auto const& func : gsc::gsc_ctx->meth_map()) { const auto name = std::string(func.first); entity_type[name] = [name](const entity& entity, const sol::this_state s, sol::variadic_args va) @@ -336,7 +335,7 @@ namespace scripting::lua auto game_type = state.new_usertype("game_"); state["game"] = game(); - for (const auto& func : xsk::gsc::h1::resolver::get_functions()) + for (auto const& func : gsc::gsc_ctx->func_map()) { const auto name = std::string(func.first); game_type[name] = [name](const game&, const sol::this_state s, sol::variadic_args va) diff --git a/src/client/game/structs.hpp b/src/client/game/structs.hpp index aa953275..f2f57509 100644 --- a/src/client/game/structs.hpp +++ b/src/client/game/structs.hpp @@ -1,7 +1,15 @@ #pragma once #include -#define PROTOCOL 1 +#define PROTOCOL 2 + +#ifdef DEBUG +#define assert_sizeof(__ASSET__, __SIZE__) static_assert(sizeof(__ASSET__) == __SIZE__) +#define assert_offsetof(__ASSET__, __VARIABLE__, __OFFSET__) static_assert(offsetof(__ASSET__, __VARIABLE__) == __OFFSET__) +#else +#define assert_sizeof(__ASSET__, __SIZE__) +#define assert_offsetof(__ASSET__, __VARIABLE__, __OFFSET__) +#endif namespace game { @@ -223,6 +231,40 @@ namespace game SF_COUNT = 0x6, }; + enum FileSysResult : std::int32_t + { + FILESYSRESULT_SUCCESS = 0x0, + FILESYSRESULT_EOF = 0x1, + FILESYSRESULT_ERROR = 0x2, + }; + + struct DB_IFileSysFile + { + void* file; + uint64_t last_read; + uint64_t bytes_read; + }; + + static_assert(sizeof(DB_IFileSysFile) == 24); + + struct DB_FileSysInterface; + + // this is a best guess, interface doesn't match up exactly w/other games (IW8, T9) + struct DB_FileSysInterface_vtbl + { + DB_IFileSysFile* (__fastcall* OpenFile)(DB_FileSysInterface* _this, Sys_Folder folder, const char* filename); + FileSysResult(__fastcall* Read)(DB_FileSysInterface* _this, DB_IFileSysFile* handle, unsigned __int64 offset, unsigned __int64 size, void* dest); + FileSysResult(__fastcall* Tell)(DB_FileSysInterface* _this, DB_IFileSysFile* handle, unsigned __int64* bytesRead); + __int64(__fastcall* Size)(DB_FileSysInterface* _this, DB_IFileSysFile* handle); + void(__fastcall* Close)(DB_FileSysInterface* _this, DB_IFileSysFile* handle); + bool(__fastcall* Exists)(DB_FileSysInterface* _this, Sys_Folder folder, const char* filename); + }; + + struct DB_FileSysInterface + { + DB_FileSysInterface_vtbl* vftbl; + }; + enum CodPlayMode { CODPLAYMODE_NONE = 0x0, @@ -1052,6 +1094,13 @@ namespace game int freeFlags; }; + struct XZoneInfoInternal + { + char name[64]; + int flags; + int isBaseMap; + }; + enum XAssetType { ASSET_TYPE_PHYSPRESET, @@ -1217,22 +1266,133 @@ namespace game char rasterizerState; }; - struct Material + struct Packed128 + { + std::uint64_t p0; + std::uint64_t p1; + }; + + struct GfxDrawSurfFields + { + unsigned __int64 objectId : 16; // p0 >> 0 + unsigned __int64 pad0 : 20; + unsigned __int64 reflectionProbeIndex : 8; + unsigned __int64 hasGfxEntIndex : 1; + unsigned __int64 customIndex : 5; // p0 >> 45 + unsigned __int64 materialSortedIndex : 14; // p0 >> 50 + unsigned __int64 tessellation : 2; // p1 >> 0 + unsigned __int64 prepass : 2; // p1 >> 2 + unsigned __int64 pad1 : 4; + unsigned __int64 useHeroLighting : 1; + unsigned __int64 sceneLightEnvIndex : 16; + unsigned __int64 viewModelRender : 1; + unsigned __int64 surfType : 4; + unsigned __int64 primarySortKey : 6; // p1 >> 30 + unsigned __int64 unused : 28; + }; + + union GfxDrawSurf + { + GfxDrawSurfFields fields; + Packed128 packed; + }; + + enum SurfaceTypeBits : std::uint64_t + { + SURFTYPE_BITS_DEFAULT = 0x0, + SURFTYPE_BITS_BARK = 0x1, + SURFTYPE_BITS_BRICK = 0x2, + SURFTYPE_BITS_CARPET = 0x4, + SURFTYPE_BITS_CLOTH = 0x8, + SURFTYPE_BITS_CONCRETE = 0x10, + SURFTYPE_BITS_DIRT = 0x20, + SURFTYPE_BITS_FLESH = 0x40, + SURFTYPE_BITS_FOLIAGE_DEBRIS = 0x80, + SURFTYPE_BITS_GLASS = 0x100, + SURFTYPE_BITS_GRASS = 0x200, + SURFTYPE_BITS_GRAVEL = 0x400, + SURFTYPE_BITS_ICE = 0x800, + SURFTYPE_BITS_METAL_SOLID = 0x1000, + SURFTYPE_BITS_METAL_GRATE = 0x2000, + SURFTYPE_BITS_MUD = 0x4000, + SURFTYPE_BITS_PAPER = 0x8000, + SURFTYPE_BITS_PLASTER = 0x10000, + SURFTYPE_BITS_ROCK = 0x20000, + SURFTYPE_BITS_SAND = 0x40000, + SURFTYPE_BITS_SNOW = 0x80000, + SURFTYPE_BITS_WATER_WAIST = 0x100000, + SURFTYPE_BITS_WOOD_SOLID = 0x200000, + SURFTYPE_BITS_ASPHALT = 0x400000, + SURFTYPE_BITS_CERAMIC = 0x800000, + SURFTYPE_BITS_PLASTIC_SOLID = 0x1000000, + SURFTYPE_BITS_RUBBER = 0x2000000, + SURFTYPE_BITS_FRUIT = 0x4000000, + SURFTYPE_BITS_PAINTEDMETAL = 0x8000000, + SURFTYPE_BITS_RIOTSHIELD = 0x10000000, + SURFTYPE_BITS_SLUSH = 0x20000000, + SURFTYPE_BITS_ASPHALT_WET = 0x40000000, + SURFTYPE_BITS_ASPHALT_DEBRIS = 0x80000000, + SURFTYPE_BITS_CONCRETE_WET = 0x100000000, + SURFTYPE_BITS_CONCRETE_DEBRIS = 0x200000000, + SURFTYPE_BITS_FOLIAGE_VEGETATION = 0x400000000, + SURFTYPE_BITS_FOLIAGE_LEAVES = 0x800000000, + SURFTYPE_BITS_GRASS_TALL = 0x1000000000, + SURFTYPE_BITS_METAL_HOLLOW = 0x2000000000, + SURFTYPE_BITS_METAL_VEHICLE = 0x4000000000, + SURFTYPE_BITS_METAL_THIN = 0x8000000000, + SURFTYPE_BITS_METAL_WET = 0x10000000000, + SURFTYPE_BITS_METAL_DEBRIS = 0x20000000000, + SURFTYPE_BITS_PLASTIC_HOLLOW = 0x40000000000, + SURFTYPE_BITS_PLASTIC_TARP = 0x80000000000, + SURFTYPE_BITS_ROCK_WET = 0x100000000000, + SURFTYPE_BITS_ROCK_DEBRIS = 0x200000000000, + SURFTYPE_BITS_WATER_ANKLE = 0x400000000000, + SURFTYPE_BITS_WATER_KNEE = 0x800000000000, + SURFTYPE_BITS_WOOD_HOLLOW = 0x1000000000000, + SURFTYPE_BITS_WOOD_WET = 0x2000000000000, + SURFTYPE_BITS_WOOD_DEBRIS = 0x4000000000000, + SURFTYPE_BITS_CUSHION = 0x8000000000000, + }; + + struct MaterialInfo { const char* name; - char __pad0[0x118]; - char textureCount; - char constantCount; - char stateBitsCount; - char stateFlags; - char cameraRegion; - char materialType; - char assetFlags; + unsigned char gameFlags; + unsigned char sortKey; + unsigned char textureAtlasRowCount; + unsigned char textureAtlasColumnCount; + unsigned char textureAtlasFrameBlend; + unsigned char textureAtlasAsArray; + unsigned char renderFlags; + GfxDrawSurf drawSurf; + void* surfaceTypeBits; + unsigned int hashIndex; + }; + + struct Material + { + union + { + const char* name; + MaterialInfo info; + }; + unsigned char stateBitsEntry[240]; + unsigned char textureCount; + unsigned char constantCount; + unsigned char stateBitsCount; + unsigned char stateFlags; + unsigned char cameraRegion; + unsigned char materialType; + unsigned char layerCount; + unsigned char assetFlags; MaterialTechniqueSet* techniqueSet; MaterialTextureDef* textureTable; void* constantTable; GfxStateBits* stateBitsTable; - char __pad2[0x108]; + unsigned char constantBufferIndex[240]; + void* constantBufferTable; + unsigned char constantBufferCount; + const char** subMaterials; }; static_assert(sizeof(Material) == 0x250); @@ -1382,18 +1542,16 @@ namespace game char data[1]; }; - union $3FA29451CE6F1FA138A5ABAB84BE9676 - { - ID3D11Texture1D* linemap; - ID3D11Texture2D* map; - ID3D11Texture3D* volmap; - ID3D11Texture2D* cubemap; - GfxImageLoadDef* loadDef; - }; - struct GfxTexture { - $3FA29451CE6F1FA138A5ABAB84BE9676 ___u0; + union + { + ID3D11Texture1D* linemap; + ID3D11Texture2D* map; + ID3D11Texture3D* volmap; + ID3D11Texture2D* cubemap; + GfxImageLoadDef* loadDef; + }; ID3D11ShaderResourceView* shaderView; ID3D11ShaderResourceView* shaderViewAlternate; }; @@ -1448,6 +1606,81 @@ namespace game const char* name; }; + struct DDLMember + { + const char* name; + int index; + void* parent; + int bitSize; + int limitSize; + int offset; + int type; + int externalIndex; + unsigned int rangeLimit; + unsigned int serverDelta; + unsigned int clientDelta; + int arraySize; + int enumIndex; + int permission; + }; + + struct DDLHash + { + unsigned int hash; + int index; + }; + + struct DDLHashTable + { + DDLHash* list; + int count; + int max; + }; + + struct DDLStruct + { + const char* name; + int bitSize; + int memberCount; + DDLMember* members; + DDLHashTable hashTableUpper; + DDLHashTable hashTableLower; + }; + + struct DDLEnum + { + const char* name; + int memberCount; + const char** members; + DDLHashTable hashTable; + }; + + struct DDLDef + { + char* name; + unsigned short version; + unsigned int checksum; + unsigned char flags; + int bitSize; + int byteSize; + DDLStruct* structList; + int structCount; + DDLEnum* enumList; + int enumCount; + DDLDef* next; + int headerBitSize; + int headerByteSize; + int reserveSize; + int userFlagsSize; + bool paddingUsed; + }; + + struct DDLRoot + { + const char* name; + DDLDef* ddlDef; + }; + union XAssetHeader { void* data; @@ -1510,6 +1743,16 @@ namespace game float halfSize[3]; }; + struct rectDef_s + { + float x; + float y; + float w; + float h; + int horzAlign; + int vertAlign; + }; + // made up struct client_state_t { @@ -1599,6 +1842,13 @@ namespace game int ingame_cursor_visible; }; + enum PMem_Direction + { + PHYS_ALLOC_LOW = 0x0, + PHYS_ALLOC_HIGH = 0x1, + PHYS_ALLOC_COUNT = 0x2, + }; + enum PMem_Source { PMEM_SOURCE_EXTERNAL = 0x0, @@ -1607,19 +1857,38 @@ namespace game PMEM_SOURCE_DEFAULT_HIGH = 0x3, PMEM_SOURCE_MOVIE = 0x4, PMEM_SOURCE_SCRIPT = 0x5, + PMEM_SOURCE_UNK5 = 0x5, + PMEM_SOURCE_UNK6 = 0x6, + PMEM_SOURCE_UNK7 = 0x7, + PMEM_SOURCE_UNK8 = 0x8, + PMEM_SOURCE_CUSTOMIZATION = 0x9, }; - struct physical_memory + struct PhysicalMemoryAllocation { - char __pad0[0x10]; - char* buf; - char __pad1[0x8]; - int unk1; - size_t size; - char __pad2[0x500]; - }; + const char* name; + char __pad0[16]; + unsigned __int64 pos; + char __pad1[8]; + }; static_assert(sizeof(PhysicalMemoryAllocation) == 40); - static_assert(sizeof(physical_memory) == 0x530); + struct PhysicalMemoryPrim + { + const char* name; + unsigned int allocListCount; + char __pad0[4]; + unsigned char* buf; + char __pad1[8]; + int unk1; + char __pad2[4]; + unsigned __int64 pos; + PhysicalMemoryAllocation allocList[32]; + }; static_assert(sizeof(PhysicalMemoryPrim) == 1328); + + struct PhysicalMemory + { + PhysicalMemoryPrim prim[2]; + }; static_assert(sizeof(PhysicalMemory) == 0xA60); namespace mp { @@ -1660,16 +1929,20 @@ namespace game { char __pad0[2]; char pm_type; // 2 - char __pad1[18573]; + char __pad1[297]; + float angles[3]; // 300 304 308 + char __pad2[18264]; sessionState_t sessionState; - char __pad2[220]; // 254 + char __pad3[220]; // 254 team_t team; - char __pad3[30]; + char __pad4[30]; char name[32]; // 18834 - char __pad4[622]; + char __pad5[622]; int flags; // 19488 }; // size = ? + static_assert(offsetof(gclient_s, angles) == 300); + static_assert(offsetof(gclient_s, sessionState) == 18576); static_assert(offsetof(gclient_s, team) == 18800); static_assert(offsetof(gclient_s, name) == 18834); static_assert(offsetof(gclient_s, flags) == 19488); @@ -1701,16 +1974,75 @@ namespace game static_assert(sizeof(gentity_s) == 736); + struct SprintState + { + int sprintButtonUpRequired; + int sprintDelay; + int lastSprintStart; + int lastSprintEnd; + int sprintStartMaxLength; + }; + struct playerState_s { - int clientNum; - char __pad0[116]; + char clientNum; + char __pad0[1]; + char pm_type; + char __pad1[44]; + int otherFlags; + char __pad2[28]; + int pm_time; + int pm_flags; + int eFlags; + int linkFlags; + char __pad3[24]; vec3_t origin; vec3_t velocity; - char __pad1[312]; - int sprintButtonUpRequired; + char __pad4[312]; + SprintState sprintState; + char __pad5[88]; + int weaponState0; + char __pad6[7040]; + int perks[2]; }; + static_assert(offsetof(playerState_s, pm_type) == 2); + //static_assert(offsetof(playerState_s, groundEntityNum) == 34); + static_assert(offsetof(playerState_s, otherFlags) == 48); + static_assert(offsetof(playerState_s, pm_time) == 80); + static_assert(offsetof(playerState_s, pm_flags) == 84); + static_assert(offsetof(playerState_s, eFlags) == 88); + static_assert(offsetof(playerState_s, linkFlags) == 92); + static_assert(offsetof(playerState_s, origin) == 120); + static_assert(offsetof(playerState_s, velocity) == 132); + static_assert(offsetof(playerState_s, sprintState) == 456); + static_assert(offsetof(playerState_s, weaponState0) == 564); + static_assert(offsetof(playerState_s, perks) == 7608); + + struct snapshot_s + { + playerState_s ps; + }; + + struct cg_s + { + char __pad0[18680]; + snapshot_s* nextSnap; + char __pad1[582400]; + int unk_601088; + int renderingThirdPerson; + char __pad2[378580]; + int unk_979676; + char __pad3[16]; + int unk_979696; + }; + + static_assert(offsetof(cg_s, nextSnap) == 18680); + static_assert(offsetof(cg_s, unk_601088) == 601088); + static_assert(offsetof(cg_s, renderingThirdPerson) == 601092); + static_assert(offsetof(cg_s, unk_979676) == 979676); + static_assert(offsetof(cg_s, unk_979696) == 979696); + struct pmove_t { playerState_s* ps; @@ -1752,6 +2084,15 @@ namespace game }; // size = 1011960 static_assert(sizeof(client_t) == 1011960); + + struct XZone + { + char __pad0[32]; + char name[64]; + char __pad1[408]; + }; + + static_assert(sizeof(XZone) == 504); } namespace sp @@ -1773,6 +2114,15 @@ namespace game struct playerState_s { }; + + struct XZone + { + char __pad0[32]; + char name[64]; + char __pad1[128]; + }; + + static_assert(sizeof(XZone) == 224); } union playerState_s @@ -1781,6 +2131,293 @@ namespace game mp::playerState_s* mp; }; + struct GfxWorldDpvsPlanes + { + int cellCount; + void* planes; + unsigned short* nodes; + unsigned int* sceneEntCellBits; + }; assert_sizeof(GfxWorldDpvsPlanes, 32); + + struct sunflare_t + { + bool hasValidData; + Material* spriteMaterial; + Material* flareMaterial; + float spriteSize; + float flareMinSize; + float flareMinDot; + float flareMaxSize; + float flareMaxDot; + float flareMaxAlpha; + int flareFadeInTime; + int flareFadeOutTime; + float blindMinDot; + float blindMaxDot; + float blindMaxDarken; + int blindFadeInTime; + int blindFadeOutTime; + float glareMinDot; + float glareMaxDot; + float glareMaxLighten; + int glareFadeInTime; + int glareFadeOutTime; + float sunFxPosition[3]; + }; assert_sizeof(sunflare_t, 112); + + typedef void* umbraTomePtr_t; + + struct GfxBuildInfo + { + const char* args0; + const char* args1; + const char* buildStartTime; + const char* buildEndTime; + }; assert_sizeof(GfxBuildInfo, 32); + + struct GfxStaticModelDrawInst + { + char __pad0[56]; + XModel* __ptr64 model; + unsigned short cullDist; + unsigned short flags; + unsigned short lightingHandle; + unsigned short staticModelId; + unsigned short primaryLightEnvIndex; + short unk0; + char unk1; // lod related + unsigned char reflectionProbeIndex; + unsigned char firstMtlSkinIndex; + unsigned char sunShadowFlags; + }; assert_sizeof(GfxStaticModelDrawInst, 80); + assert_offsetof(GfxStaticModelDrawInst, model, 56); + assert_offsetof(GfxStaticModelDrawInst, cullDist, 64); + assert_offsetof(GfxStaticModelDrawInst, flags, 66); + assert_offsetof(GfxStaticModelDrawInst, lightingHandle, 68); + assert_offsetof(GfxStaticModelDrawInst, primaryLightEnvIndex, 72); + assert_offsetof(GfxStaticModelDrawInst, reflectionProbeIndex, 77); // maybe wrong + assert_offsetof(GfxStaticModelDrawInst, firstMtlSkinIndex, 78); + + struct GfxStaticModelVertexLighting + { + unsigned char visibility[4]; + unsigned short ambientColorFloat16[4]; + unsigned short highlightColorFloat16[4]; + }; assert_sizeof(GfxStaticModelVertexLighting, 20); + + struct GfxStaticModelVertexLightingInfo + { + GfxStaticModelVertexLighting* lightingValues; + ID3D11Buffer* lightingValuesVb; + int numLightingValues; + }; + + struct GfxStaticModelLightmapInfo + { + float offset[2]; + float scale[2]; + unsigned int lightmapIndex; + }; + + struct GfxStaticModelGroundLightingInfo + { + unsigned short groundLighting[4]; // float16 + }; + + struct GfxStaticModelLightGridLightingInfo + { + unsigned short colorFloat16[4]; + int a; + float b; + char __pad1[8]; + }; + + union GfxStaticModelLighting + { + GfxStaticModelVertexLightingInfo vertexLightingInfo; + GfxStaticModelLightmapInfo modelLightmapInfo; + GfxStaticModelGroundLightingInfo modelGroundLightingInfo; + GfxStaticModelLightGridLightingInfo modelLightGridLightingInfo; + }; assert_sizeof(GfxStaticModelLighting, 24); + + struct GfxWorldDpvsStatic + { + unsigned int smodelCount; // 0 + unsigned int subdivVertexLightingInfoCount; // 4 + unsigned int staticSurfaceCount; // 8 + unsigned int litOpaqueSurfsBegin; // 12 + unsigned int litOpaqueSurfsEnd; // 16 + unsigned int unkSurfsBegin; + unsigned int unkSurfsEnd; + unsigned int litDecalSurfsBegin; // 28 + unsigned int litDecalSurfsEnd; // 32 + unsigned int litTransSurfsBegin; // 36 + unsigned int litTransSurfsEnd; // 40 + unsigned int shadowCasterSurfsBegin; // 44 + unsigned int shadowCasterSurfsEnd; // 48 + unsigned int emissiveSurfsBegin; // 52 + unsigned int emissiveSurfsEnd; // 56 + unsigned int smodelVisDataCount; // 60 + unsigned int surfaceVisDataCount; // 64 + unsigned int* smodelVisData[4]; // 72 80 88 96 + unsigned int* smodelUnknownVisData[27]; + unsigned int* surfaceVisData[4]; // 320 328 336 344 + unsigned int* surfaceUnknownVisData[27]; + unsigned int* smodelUmbraVisData[4]; // 568 576 584 592 + unsigned int* surfaceUmbraVisData[4]; // 600 608 616 624 + unsigned int* lodData; // 632 + unsigned int* tessellationCutoffVisData; // 640 + unsigned int* sortedSurfIndex; // 648 + void* smodelInsts; // 656 + void* surfaces; // 664 + void* surfacesBounds; // 672 + GfxStaticModelDrawInst* smodelDrawInsts; // 680 + unsigned int* unknownSModelVisData1; // 688 + unsigned int* unknownSModelVisData2; // 696 + GfxStaticModelLighting* smodelLighting; // 704 (array) + void* subdivVertexLighting; // 712 (array) + GfxDrawSurf* surfaceMaterials; // 720 + unsigned int* surfaceCastsSunShadow; // 728 + unsigned int sunShadowOptCount; // 736 + unsigned int sunSurfVisDataCount; // 740 + unsigned int* surfaceCastsSunShadowOpt; // 744 + void* surfaceDeptAndSurf; // 752 + void* constantBuffersLit; // 760 + void* constantBuffersAmbient; // 768 + int usageCount; // 776 + }; assert_sizeof(GfxWorldDpvsStatic, 784); + assert_offsetof(GfxWorldDpvsStatic, smodelVisData[0], 72); + assert_offsetof(GfxWorldDpvsStatic, surfaceVisData[0], 320); + assert_offsetof(GfxWorldDpvsStatic, smodelUmbraVisData[0], 568); + assert_offsetof(GfxWorldDpvsStatic, tessellationCutoffVisData, 640); + assert_offsetof(GfxWorldDpvsStatic, smodelDrawInsts, 680); + assert_offsetof(GfxWorldDpvsStatic, smodelLighting, 704); + assert_offsetof(GfxWorldDpvsStatic, sunSurfVisDataCount, 740); + assert_offsetof(GfxWorldDpvsStatic, constantBuffersAmbient, 768); + + struct GfxWorld + { + const char* name; // 0 + const char* baseName; // 8 + unsigned int bspVersion; // 16 + int planeCount; // 20 + int nodeCount; // 24 + unsigned int surfaceCount; // 28 + int skyCount; // 32 + void* skies; // 40 + unsigned int portalGroupCount; // 48 + unsigned int lastSunPrimaryLightIndex; // 52 + unsigned int primaryLightCount; // 56 + unsigned int primaryLightEnvCount; // 60 + unsigned int sortKeyLitDecal; // 64 + unsigned int sortKeyEffectDecal; // 68 + unsigned int sortKeyTopDecal; // 72 + unsigned int sortKeyEffectAuto; // 76 + unsigned int sortKeyDistortion; // 80 + unsigned int sortKeyUnknown; // 84 + unsigned int sortKeyUnknown2; // 88 + char __pad0[4]; // 92 + GfxWorldDpvsPlanes dpvsPlanes; // 96 + void* aabbTreeCounts; // 128 + void* aabbTrees; // 136 + void* cells; // 144 + void* portalGroup; // 152 + int unk_vec4_count_0; // 160 + char __pad1[4]; + vec4_t* unk_vec4_0; // 168 + + //GfxWorldDraw draw; // 176 + //GfxLightGrid lightGrid; // 432 + char __pad2[1336]; // 176 + + int modelCount; // 1512 + void* models; // 1520 + vec3_t mins1; + vec3_t maxs1; + vec3_t mins2; + vec3_t maxs2; + unsigned int checksum; + int materialMemoryCount; // 1580 + void* materialMemory; // 1584 + sunflare_t sun; // 1592 + float outdoorLookupMatrix[4][4]; + GfxImage* outdoorImage; // 1768 + unsigned int* cellCasterBits; // 1776 + unsigned int* cellHasSunLitSurfsBits; // 1784 + void* sceneDynModel; // 1792 + void* sceneDynBrush; // 1800 + unsigned int* primaryLightEntityShadowVis; // 1808 + unsigned int* primaryLightDynEntShadowVis[2]; // 1816 1824 + unsigned short* nonSunPrimaryLightForModelDynEnt; // 1832 + void* shadowGeom; // 1840 + void* shadowGeomOptimized; // 1848 + void* lightRegion; // 1856 + + GfxWorldDpvsStatic dpvs; // 1864 + //GfxWorldDpvsDynamic dpvsDyn; // 2648 + char __pad3[96]; // 2648 + + unsigned int mapVtxChecksum; // 2744 + unsigned int heroOnlyLightCount; // 2748 + void* heroOnlyLights; // 2752 + unsigned char fogTypesAllowed; // 2760 + unsigned int umbraTomeSize; // 2764 + char* umbraTomeData; // 2768 + umbraTomePtr_t umbraTomePtr; // 2776 + unsigned int mdaoVolumesCount; // 2784 + void* mdaoVolumes; // 2792 + char __pad4[32]; + GfxBuildInfo buildInfo; // 2832 + }; assert_sizeof(GfxWorld, 0xB30); + assert_offsetof(GfxWorld, skyCount, 32); + assert_offsetof(GfxWorld, skies, 40); + assert_offsetof(GfxWorld, dpvsPlanes, 96); + assert_offsetof(GfxWorld, aabbTreeCounts, 128); + assert_offsetof(GfxWorld, cells, 144); + assert_offsetof(GfxWorld, portalGroup, 152); + assert_offsetof(GfxWorld, unk_vec4_count_0, 160); + assert_offsetof(GfxWorld, unk_vec4_0, 168); + assert_offsetof(GfxWorld, __pad2, 176); + //assert_offsetof(GfxWorld, lightGrid, 432); + assert_offsetof(GfxWorld, modelCount, 1512); + assert_offsetof(GfxWorld, models, 1520); + assert_offsetof(GfxWorld, materialMemoryCount, 1580); + assert_offsetof(GfxWorld, materialMemory, 1584); + assert_offsetof(GfxWorld, sun, 1592); + assert_offsetof(GfxWorld, outdoorImage, 1768); + assert_offsetof(GfxWorld, cellCasterBits, 1776); + assert_offsetof(GfxWorld, cellHasSunLitSurfsBits, 1784); + assert_offsetof(GfxWorld, dpvs, 1864); + assert_offsetof(GfxWorld, __pad3, 2648); + assert_offsetof(GfxWorld, heroOnlyLightCount, 2748); + assert_offsetof(GfxWorld, heroOnlyLights, 2752); + assert_offsetof(GfxWorld, umbraTomeSize, 2764); + assert_offsetof(GfxWorld, umbraTomeData, 2768); + assert_offsetof(GfxWorld, umbraTomePtr, 2776); + assert_offsetof(GfxWorld, mdaoVolumesCount, 2784); + assert_offsetof(GfxWorld, mdaoVolumes, 2792); + assert_offsetof(GfxWorld, mdaoVolumes, 2792); + assert_offsetof(GfxWorld, buildInfo, 2832); + + struct DB_AuthSignature + { + unsigned char bytes[256]; + }; + + struct DB_AuthHash + { + unsigned char bytes[32]; + }; + + struct XPakHeader + { + char header[8]; + std::int32_t version; + unsigned char unknown[16]; + DB_AuthHash hash; + DB_AuthSignature signature; + }; + namespace hks { struct lua_State; diff --git a/src/client/game/symbols.hpp b/src/client/game/symbols.hpp index 8d48b504..3c77e591 100644 --- a/src/client/game/symbols.hpp +++ b/src/client/game/symbols.hpp @@ -47,7 +47,9 @@ namespace game WEAK symbol Com_SetSlowMotion{0x0, 0x17E5F0}; WEAK symbol Com_Error{0x384820, 0x159860}; WEAK symbol Com_Quit_f{0x0, 0x1F9280}; - WEAK symbol Com_Shutdown{0x3A6A50, 0x0}; + WEAK symbol Com_Shutdown{0x3A6A50, 0x157E10}; + WEAK symbol Com_IsAddonMap{0x40AED0, 0x17C100}; + WEAK symbol Com_sprintf{0x429200, 0x5AF0F0}; WEAK symbol Quit{0x3A5A20, 0x17CF50}; @@ -94,6 +96,7 @@ namespace game WEAK symbol FS_Startup{0x40D890, 0x0}; WEAK symbol FS_AddLocalizedGameDirectory{0x40B1E0, 0x1878F0}; + WEAK symbol GetObjectType{0x3C3680, 0x50A810}; WEAK symbol GetVariable{0x3C3740, 0x50A8D0}; WEAK symbol GetNewVariable{0x3C3360, 0x50A4F0}; WEAK symbol GetNewArrayVariable{0x3C31E0, 0x50A370}; @@ -122,6 +125,8 @@ namespace game WEAK symbol I_CleanStr{0x4293E0, 0x5AF2E0}; WEAK symbol Key_KeynumToString{0x1AC410, 0x199990}; + WEAK symbol Key_GetBindingForCmd{0x377280, 0x1572B0}; + WEAK symbol Key_SetBinding{0x1AC570, 0x199AE0}; WEAK symbol Live_SyncOnlineDataFlags{0x0, 0x1A5C10}; @@ -175,6 +180,7 @@ namespace game WEAK symbol Scr_GetFloat{0x3C87D0, 0x50F870}; WEAK symbol Scr_GetString{0x3C8CC0, 0x50FCB0}; WEAK symbol Scr_GetNumParam{0x3C89E0, 0x50F9D0}; + WEAK symbol Scr_CastString{0x3C4450, 0x50B4B0}; WEAK symbol Scr_ClearOutParams{0x3C7EF0, 0x50F070}; WEAK symbol Scr_GetEntityIdRef{0x3C6760, 0x50D8E0}; WEAK symbol Scr_GetEntityId{0x3C66B0, 0x50D830}; @@ -193,7 +199,7 @@ namespace game WEAK symbol DB_EnumXAssets_Internal{0x1F0BF0, 0x394C60}; WEAK symbol DB_GetXAssetName{0x1BF890, 0x366140}; - WEAK symbol DB_GetXAssetTypeSize{0x0, 0x0}; + WEAK symbol DB_GetXAssetTypeSize{0x0, 0x366180}; WEAK symbol DB_FindXAssetHeader{0x1F1120, 0x3950C0}; WEAK symbol UI_AddMenuList{0x0, 0x1D9960}; WEAK symbol UI_SafeTranslateString{0x3840A0, 0x4E8BC0}; + WEAK symbol UI_DrawWrappedText{0x406CD0, 0x1DCE30}; WEAK symbol UI_SetActiveMenu{0x0, 0x1E4D80}; @@ -296,25 +304,34 @@ namespace game WEAK symbol connectionState{0x0, 0x2EC82C8}; - // TODO: move to dvars.cpp when done - WEAK symbol fs_gameDirVal{0x0, 0x2EC86B8}; - - WEAK symbol g_poolSize{0x0, 0x10B3C80}; + WEAK symbol g_poolSize{0xEC97D0, 0x10B3C80}; WEAK symbol g_compressor{0x2574804, 0x3962804}; WEAK symbol scr_VarGlob{0xBD80E00, 0xB138180}; WEAK symbol scr_VmPub{0xC3F4E20, 0xB7AE3C0}; WEAK symbol scr_function_stack{0xC4015C0, 0xB7B8940}; - WEAK symbol g_scriptmem{0xD5F3140, 0xC92EC40}; + WEAK game::symbol pmem_size{0xD5F26D8, 0xC92E1D8}; + WEAK game::symbol pmem_buffer{0xD5F26D0, 0xC92E1D0}; + + WEAK game::symbol g_mem{0xD5F26E0, 0xC92E1E0}; + WEAK game::symbol g_scriptmem{0xD5F3140, 0xC92EC40}; + WEAK game::symbol g_physmem{0xD5F3BA0, 0xC92F6A0}; + + WEAK game::symbol stream_size{0x1DAD810, 0x258AA10}; + WEAK game::symbol stream_buffer{0x1DAD808, 0x258AA08}; WEAK symbol gfxDrawMethod{0xF7530B0, 0xE9213F0}; WEAK symbol dvarCount{0xC90E550, 0x2999C34}; WEAK symbol dvarPool{0xC90E560, 0x344DF20}; - WEAK symbol DB_XAssetPool{0xEC9FB0, 0x10B4460}; + WEAK symbol g_assetPool{0xEC9FB0, 0x10B4460}; WEAK symbol g_assetNames{0x991BA0, 0x10B30D0}; + WEAK symbol g_zoneInfo{0x0, 0x5F5A370}; + WEAK symbol g_zoneIndex{0x0, 0x3D1008C}; + + WEAK symbol db_fs{0x25C1168, 0x1566C08}; WEAK symbol keyCatchers{0x252AF70, 0x2EC82C4}; WEAK symbol playerKeys{0x2395B0C, 0x2999E1C}; @@ -332,6 +349,8 @@ namespace game WEAK symbol maps{0x7CE5A0, 0x926C80}; + WEAK symbol d3d11_device{0x1163B98, 0x12DFBF8}; + namespace mp { WEAK symbol g_entities{0x0, 0x71F19E0}; @@ -345,11 +364,15 @@ namespace game WEAK symbol client_state{0x0, 0x2EC84F0}; WEAK symbol connect_state{0x0, 0x2EC8510}; + + WEAK symbol g_zones{0x0, 0x5F292B0}; } namespace sp { WEAK symbol g_entities{0x56E74D0, 0x0}; + + WEAK symbol g_zones{0x45FE990, 0x0}; } namespace hks diff --git a/src/client/launcher/html/html_frame.cpp b/src/client/launcher/html/html_frame.cpp index d844ffec..0c5c992e 100644 --- a/src/client/launcher/html/html_frame.cpp +++ b/src/client/launcher/html/html_frame.cpp @@ -1,40 +1,117 @@ #include #include "html_frame.hpp" - -#include +#include "utils/nt.hpp" +#include "utils/io.hpp" +#include "utils/hook.hpp" std::atomic html_frame::frame_count_ = 0; +namespace +{ + void* original_func{}; + GUID browser_emulation_guid{ 0xac969931, 0x3566, 0x4b50, {0xae, 0x48, 0x71, 0xb9, 0x6a, 0x75, 0xc8, 0x79} }; + + int WINAPI co_internet_feature_value_internal_stub(const GUID* guid, uint32_t* result) + { + const auto res = static_cast(original_func)(guid, result); + + if (IsEqualGUID(*guid, browser_emulation_guid)) + { + *result = 11000; + return 0; + } + + return res; + } + + void patch_cached_browser_emulator(const utils::nt::library& urlmon) + { + std::string data{}; + if (!utils::io::read_file(urlmon.get_path().generic_string(), &data)) + { + return; + } + + const utils::nt::library file_lib(reinterpret_cast(data.data())); + + auto translate_file_offset_to_rva = [&](const size_t file_offset) -> size_t + { + const auto sections = file_lib.get_section_headers(); + for (const auto* section : sections) + { + if (section->PointerToRawData <= file_offset && section->PointerToRawData + section->SizeOfRawData > file_offset) + { + const auto section_va = file_offset - section->PointerToRawData; + return section_va + section->VirtualAddress; + } + } + + return 0; + }; + + const auto guid_pos = data.find(std::string(reinterpret_cast(&browser_emulation_guid), sizeof(browser_emulation_guid))); + if (guid_pos == std::string::npos) + { + return; + } + + const auto guid_rva = translate_file_offset_to_rva(guid_pos); + const auto guid_va = reinterpret_cast(urlmon.get_ptr() + guid_rva); + + if (!IsEqualGUID(*guid_va, browser_emulation_guid)) + { + return; + } + + const size_t unrelocated_guid_va = file_lib.get_optional_header()->ImageBase + guid_rva; + const auto guid_ptr_pos = data.find(std::string(reinterpret_cast(&unrelocated_guid_va), sizeof(unrelocated_guid_va))); + if (guid_ptr_pos == std::string::npos) + { + return; + } + + const auto guid_ptr_rva = translate_file_offset_to_rva(guid_ptr_pos); + *reinterpret_cast(urlmon.get_ptr() + guid_ptr_rva) = guid_va; + } + + void setup_ie_hooks() + { + static const auto _ = [] + { + const auto urlmon = utils::nt::library::load("urlmon.dll"s); + const auto target = urlmon.get_iat_entry("iertutil.dll", MAKEINTRESOURCEA(700)); + + original_func = *target; + utils::hook::set(target, co_internet_feature_value_internal_stub); + + patch_cached_browser_emulator(urlmon); + + return 0; + }(); + (void)_; + } +} + html_frame::callback_params::callback_params(DISPPARAMS* params, VARIANT* res) : result(res) { for (auto i = params->cArgs; i > 0; --i) { - auto* param = ¶ms->rgvarg[i - 1]; + auto param = ¶ms->rgvarg[i - 1]; this->arguments.emplace_back(param); } } -html_frame::html_frame() - : in_place_frame_(this) - , in_place_site_(this) - , ui_handler_(this) - , client_site_(this) - , html_dispatch_(this) +html_frame::html_frame() : in_place_frame_(this), in_place_site_(this), ui_handler_(this), client_site_(this), +html_dispatch_(this) { + setup_ie_hooks(); if (frame_count_++ == 0 && OleInitialize(nullptr) != S_OK) { throw std::runtime_error("Unable to initialize the OLE library"); } - auto needs_restart = false; - needs_restart |= set_browser_feature("FEATURE_BROWSER_EMULATION", 11000); - needs_restart |= set_browser_feature("FEATURE_GPU_RENDERING", 1); - - if (needs_restart) - { - utils::nt::relaunch_self(); - utils::nt::terminate(0); - } + set_browser_feature("FEATURE_BROWSER_EMULATION", 11000); + set_browser_feature("FEATURE_GPU_RENDERING", 1); } html_frame::~html_frame() diff --git a/src/client/launcher/window.cpp b/src/client/launcher/window.cpp index b9390612..eceaf84b 100644 --- a/src/client/launcher/window.cpp +++ b/src/client/launcher/window.cpp @@ -3,6 +3,10 @@ #include +#ifndef DWMWA_USE_IMMERSIVE_DARK_MODE +#define DWMWA_USE_IMMERSIVE_DARK_MODE 20 +#endif + std::mutex window::mutex_; std::vector window::windows_; @@ -37,6 +41,10 @@ void window::create(const std::string& title, const int width, const int height, this->handle_ = CreateWindowExA(NULL, this->wc_.lpszClassName, title.data(), flags, x, y, width, height, nullptr, nullptr, this->wc_.hInstance, this); + BOOL value = TRUE; + DwmSetWindowAttribute(this->handle_, + DWMWA_USE_IMMERSIVE_DARK_MODE, &value, sizeof(value)); + SendMessageA(this->handle_, WM_DPICHANGED, 0, 0); } @@ -50,7 +58,7 @@ void window::close() { if (!this->handle_) return; - SendMessageA(this->handle_, WM_KILL_WINDOW, NULL, NULL); + DestroyWindow(this->handle_); this->handle_ = nullptr; } diff --git a/src/client/launcher/window.hpp b/src/client/launcher/window.hpp index 4dea45d6..ddb4fd88 100644 --- a/src/client/launcher/window.hpp +++ b/src/client/launcher/window.hpp @@ -1,4 +1,5 @@ #pragma once +#pragma comment (lib, "dwmapi.lib") #define WM_KILL_WINDOW (WM_USER+0) diff --git a/src/client/main.cpp b/src/client/main.cpp index c61a57cc..97e4d7e1 100644 --- a/src/client/main.cpp +++ b/src/client/main.cpp @@ -164,12 +164,41 @@ void remove_crash_file() void enable_dpi_awareness() { const utils::nt::library user32{"user32.dll"}; - const auto set_dpi = user32 - ? user32.get_proc("SetProcessDpiAwarenessContext") - : nullptr; - if (set_dpi) + { - set_dpi(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2); + const auto set_dpi = user32 + ? user32.get_proc( + "SetProcessDpiAwarenessContext") + : nullptr; + if (set_dpi) + { + set_dpi(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2); + return; + } + } + + { + const utils::nt::library shcore{"shcore.dll"}; + const auto set_dpi = shcore + ? shcore.get_proc( + "SetProcessDpiAwareness") + : nullptr; + if (set_dpi) + { + set_dpi(PROCESS_PER_MONITOR_DPI_AWARE); + return; + } + } + + { + const auto set_dpi = user32 + ? user32.get_proc( + "SetProcessDPIAware") + : nullptr; + if (set_dpi) + { + set_dpi(); + } } } diff --git a/src/client/resource.hpp b/src/client/resource.hpp index ceb43a33..9636a45c 100644 --- a/src/client/resource.hpp +++ b/src/client/resource.hpp @@ -20,3 +20,5 @@ #define LUI_UPDATER 310 #define LUA_JSON 311 + +#define DVAR_LIST 312 diff --git a/src/client/resource.rc b/src/client/resource.rc index b833631f..fd0c2fb1 100644 --- a/src/client/resource.rc +++ b/src/client/resource.rc @@ -123,6 +123,8 @@ LUI_UPDATER RCDATA "resources/ui_scripts/updater.lua" LUA_JSON RCDATA "resources/json.lua" +DVAR_LIST RCDATA "resources/dvar_list.json" + #endif // English (United States) resources ///////////////////////////////////////////////////////////////////////////// diff --git a/src/client/resources/dvar_list.json b/src/client/resources/dvar_list.json new file mode 100644 index 00000000..8ed37cf4 --- /dev/null +++ b/src/client/resources/dvar_list.json @@ -0,0 +1,9286 @@ +[ + [ + "ai_count", + "Sets AI count." + ], + [ + "accessToSubscriberContent", + "Whether to display the subscriber maps." + ], + [ + "aci", + "" + ], + [ + "actionSlotsHide", + "Hide the actionslots." + ], + [ + "ai_grenadeReturn_approachMinDot", + "Minimal dot product between the approach and throw vectors to perform a grenade return" + ], + [ + "ai_grenadeReturn_debug", + "Turns on debug info for AI grenade returns" + ], + [ + "ai_grenadeReturn_extraFuseTime", + "The amount of time (in ms) to add to a grenade fuse when trying to return grenade that's below minFuseTime" + ], + [ + "ai_grenadeReturn_minDistSqr", + "Minimal distance to a grenade to consider it for a return so that transition anims will play" + ], + [ + "ai_grenadeReturn_minFuseTime", + "If the fuse time drops below this value when an ally is attempting to return a grenade, add extra fuse time" + ], + [ + "ai_grenadeReturn_stationary", + "If set, AI will attempt to return grenades that they are within pickup distance - regardless of min dist" + ], + [ + "ai_grenadeReturn_traceToGrenade", + "If set, AI will only attempt to return grenades when they have a clear sight trace to the grenade" + ], + [ + "ai_threatUpdateInterval", + "AI target threat update interval in milliseconds" + ], + [ + "aim_autoaim_enabled", + "" + ], + [ + "aim_target_sentient_radius", + "The radius used to calculate target bounds for a sentient(actor or player)" + ], + [ + "ammoCounterHide", + "Hide the Ammo Counter" + ], + [ + "armory_contentpacks_enabled", + "Allowed armory content packs. 0: none , 1: first armory content pack enabled, 2: first and second armory content pack enabled" + ], + [ + "badHost_detectMinServerTime", + "Time in MS before the bad host dection system kicks in after match start" + ], + [ + "badhost_maxDoISuckFrames", + "Max lagged frames need to end match" + ], + [ + "band_12players", + "12 player bandwidth req'd" + ], + [ + "band_18players", + "18 player bandwidth req'd" + ], + [ + "band_2players", + "2 player bandwidth req'd" + ], + [ + "band_4players", + "4 player bandwidth req'd" + ], + [ + "band_8players", + "8 player bandwidth req'd" + ], + [ + "bg_allowScuffFootsteps", + "If true, scuff sounds will be played when the player rotates in place." + ], + [ + "bg_bulletExplDmgFactor", + "Weapon damage multiplier that will be applied at the center of the slash damage area." + ], + [ + "bg_bulletExplRadius", + "The radius of the bullet splash damage, where the damage gradually falls off to 0." + ], + [ + "bg_compassShowEnemies", + "Whether enemies are visible on the compass at all times" + ], + [ + "bg_idleSwingSpeed", + "The rate at which the player's legs swing around when idle (multi-player only)" + ], + [ + "bg_shieldHitEncodeHeightVM", + "The decoding range, in height, of a client's viewmodel shield." + ], + [ + "bg_shieldHitEncodeHeightWorld", + "The encoding range, in height, of a client's world shield. A hit in this range is encoded into one of 8 rows." + ], + [ + "bg_shieldHitEncodeWidthVM", + "The decoding range, in width, of a client's viewmodel shield." + ], + [ + "bg_shieldHitEncodeWidthWorld", + "The encoding range, in width, of a client's world shield. A hit in this range is encoded into one of 16 collumns." + ], + [ + "bg_shock_fadeOverride", + "Override the time for the shellshock kick effect to fade in MP" + ], + [ + "bg_shock_lookControl", + "Alter player control during shellshock" + ], + [ + "bg_shock_lookControl_fadeTime", + "The time for the shellshock player control to fade in seconds" + ], + [ + "bg_shock_lookControl_maxpitchspeed", + "Maximum pitch movement rate while shellshocked in degrees per second" + ], + [ + "bg_shock_lookControl_maxyawspeed", + "Maximum yaw movement rate while shell shocked in degrees per second" + ], + [ + "bg_shock_lookControl_mousesensitivityscale", + "Sensitivity scale to apply to a shellshocked player" + ], + [ + "bg_shock_movement", + "Affect player's movement speed duringi shellshock" + ], + [ + "bg_shock_screenBlurBlendFadeTime", + "The amount of time in seconds for the shellshock effect to fade" + ], + [ + "bg_shock_screenBlurBlendTime", + "The amount of time in seconds for the shellshock effect to fade" + ], + [ + "bg_shock_screenFlashShotFadeTime", + "In seconds, how soon from the end of the effect to start blending out the whiteout layer." + ], + [ + "bg_shock_screenFlashWhiteFadeTime", + "In seconds, how soon from the end of the effect to start blending out the whiteout layer." + ], + [ + "bg_shock_screenType", + "Shell shock screen effect type" + ], + [ + "bg_shock_sound", + "Play shell shock sound" + ], + [ + "bg_shock_soundDryLevel", + "Shell shock sound dry level" + ], + [ + "bg_shock_soundEnd", + "Shellshock end sound alias" + ], + [ + "bg_shock_soundEndAbort", + "Shellshock aborted end sound alias" + ], + [ + "bg_shock_soundFadeInTime", + "Shell shock sound fade in time in seconds" + ], + [ + "bg_shock_soundFadeOutTime", + "Shell shock sound fade out time in seconds" + ], + [ + "bg_shock_soundLoop", + "Shellshock loop alias" + ], + [ + "bg_shock_soundLoopEndDelay", + "Sound loop end offset time from the end of the shellshock in seconds" + ], + [ + "bg_shock_soundLoopFadeTime", + "Shell shock sound loop fade time in seconds" + ], + [ + "bg_shock_soundLoopSilent", + "The sound that gets blended with the shellshock loop alias" + ], + [ + "bg_shock_soundModEndDelay", + "The delay from the end of the shell shock to the end of the sound modification" + ], + [ + "bg_shock_soundRoomType", + "Shell shock sound reverb room type" + ], + [ + "bg_shock_soundSubmix", + "Shell shock submix to apply" + ], + [ + "bg_shock_soundWetLevel", + "Shell shock sound wet level" + ], + [ + "bg_shock_viewKickFadeTime", + "The time for the shellshock kick effect to fade" + ], + [ + "bg_shock_viewKickPeriod", + "The period of the shellshock view kick effect" + ], + [ + "bg_shock_viewKickRadius", + "Shell shock kick radius" + ], + [ + "bg_swingSpeed", + "The rate at which the player's legs swing around when idle (multi-player only)" + ], + [ + "bg_torsoSwingSpeed", + "The rate at which the player's torso swings around when strafing (multi-player only)" + ], + [ + "boostcheatHeadshotsTotalCoef", + "" + ], + [ + "boostcheatHeadshotsTotalMean", + "" + ], + [ + "boostcheatHeadshotsTotalStddev", + "" + ], + [ + "boostcheatIntercept", + "" + ], + [ + "boostcheatKillerXAnomalyCoef", + "" + ], + [ + "boostcheatKillerXAnomalyMean", + "" + ], + [ + "boostcheatKillerXAnomalyStddev", + "" + ], + [ + "boostcheatKillerYAnomalyCoef", + "" + ], + [ + "boostcheatKillerYAnomalyMean", + "" + ], + [ + "boostcheatKillerYAnomalyStddev", + "" + ], + [ + "boostcheatMeanDistanceMostKilledPlayerTraveledMean", + "" + ], + [ + "boostcheatMeanDistanceVictimTraveledCoef", + "" + ], + [ + "boostcheatMeanDistanceVictimTraveledMean", + "" + ], + [ + "boostcheatMeanDistanceVictimTraveledStddev", + "" + ], + [ + "boostcheatMeanMostKilledPlayerLifetimeMillisecondsMean", + "" + ], + [ + "boostcheatMostKilledPlayerHKRatioCoef", + "" + ], + [ + "boostcheatMostKilledPlayerHKRatioMean", + "" + ], + [ + "boostcheatMostKilledPlayerHKRatioStddev", + "" + ], + [ + "boostcheatMostKilledPlayerKillsRatioCoef", + "" + ], + [ + "boostcheatMostKilledPlayerKillsRatioMean", + "" + ], + [ + "boostcheatMostKilledPlayerKillsRatioStddev", + "" + ], + [ + "boostcheatMostKilledPlayerKillsTotalCoef", + "" + ], + [ + "boostcheatMostKilledPlayerKillsTotalMean", + "" + ], + [ + "boostcheatMostKilledPlayerKillsTotalStddev", + "" + ], + [ + "boostcheatMostKilledPlayerKillTimestampsAnomalyMean", + "" + ], + [ + "boostcheatVictimXAnomalyCoef", + "" + ], + [ + "boostcheatVictimXAnomalyMean", + "" + ], + [ + "boostcheatVictimXAnomalyStddev", + "" + ], + [ + "boostcheatVictimYAnomalyCoef", + "" + ], + [ + "boostcheatVictimYAnomalyMean", + "" + ], + [ + "boostcheatVictimYAnomalyStddev", + "" + ], + [ + "bot_DifficultyDefault", + "default difficulty level of bots" + ], + [ + "ca_auto_signin", + "CoD Anywhere start sign-in task automatically on startup or first party sign-in" + ], + [ + "ca_do_mlc", + "CoD Anywhere Do Multi Login check" + ], + [ + "ca_intra_only", + "CoD Anywhere Intra Network Only" + ], + [ + "ca_require_signin", + "CoD Anywhere require sign in to enter MP" + ], + [ + "ca_show_signup_request", + "CoD Anywhere should you show new users a popup requesting they create a CoD Account?" + ], + [ + "camera_thirdPerson", + "Use third person view globally" + ], + [ + "cameraShakeRemoteHelo_Angles", + "Remote helicopter gunner cam, range to shake the view." + ], + [ + "cameraShakeRemoteHelo_Freqs", + "Remote helicopter gunner cam, how fast to shake." + ], + [ + "cameraShakeRemoteHelo_SpeedRange", + "Remote helicopter gunner cam, range of missile speed to scale the shaking." + ], + [ + "cameraShakeRemoteMissile_Angles", + "Remote missile-cam, range to shake the view." + ], + [ + "cameraShakeRemoteMissile_Freqs", + "Remote missile-cam, how fast to shake." + ], + [ + "cameraShakeRemoteMissile_SpeedRange", + "Remote missile-cam, range of missile speed to scale the shaking." + ], + [ + "cg_airstrikeCamFstop", + "Airstrike kill camera aperture. Lower f-stop yields a shallower depth of field. Typical values range from 1 to 22" + ], + [ + "cg_airstrikeKillCamFarBlur", + "" + ], + [ + "cg_airstrikeKillCamFarBlurDist", + "" + ], + [ + "cg_airstrikeKillCamFarBlurStart", + "" + ], + [ + "cg_airstrikeKillCamFov", + "Airstrike kill camera field of view." + ], + [ + "cg_airstrikeKillCamNearBlur", + "" + ], + [ + "cg_airstrikeKillCamNearBlurEnd", + "" + ], + [ + "cg_airstrikeKillCamNearBlurStart", + "" + ], + [ + "cg_blood", + "Show Blood" + ], + [ + "cg_bloodThickColor", + "Color of the blood overlay's thick blood splatter" + ], + [ + "cg_bloodThinColor", + "Color of the blood overlay's thin blood splatter" + ], + [ + "cg_brass", + "Weapons eject brass" + ], + [ + "cg_centertime", + "The time for a center printed message to fade" + ], + [ + "cg_chatHeight", + "The font height of a chat message" + ], + [ + "cg_chatTime", + "The amount of time that a chat message is visible" + ], + [ + "cg_ColorBlind_EnemyTeam", + "Enemy team color for color blind people" + ], + [ + "cg_ColorBlind_MyParty", + "Player party color for color blind people" + ], + [ + "cg_ColorBlind_MyTeam", + "Player team color for color blind people" + ], + [ + "cg_connectionIconSize", + "Size of the connection icon" + ], + [ + "cg_constantSizeHeadIcons", + "Head icons are the same size regardless of distance from the player" + ], + [ + "cg_crosshairAlpha", + "The alpha value of the crosshair" + ], + [ + "cg_crosshairAlphaMin", + "The minimum alpha value of the crosshair when it fades in" + ], + [ + "cg_crosshairDynamic", + "Crosshair is Dynamic" + ], + [ + "cg_crosshairEnemyColor", + "The crosshair color when over an enemy" + ], + [ + "cg_crosshairVerticalOffset", + "Amount to vertically offset the crosshair from the center." + ], + [ + "cg_cullBulletAngle", + "Cull bullet trajectories that don't fall within this fov" + ], + [ + "cg_cullBullets", + "Whether to cull bullet fire prediction if trajectory doesn't pass your view or anywhere near you" + ], + [ + "cg_cursorHints", + "Draw cursor hints where:\n 0: no hints" + ], + [ + "cg_deadChatWithDead", + "If true, dead players can all chat together, regardless of team" + ], + [ + "cg_deadChatWithTeam", + "If true, dead players can talk to living players on their team" + ], + [ + "cg_deadHearAllLiving", + "If true, dead players can hear all living players talk" + ], + [ + "cg_deadHearTeamLiving", + "If true, dead players can hear living players on their team talk" + ], + [ + "cg_descriptiveText", + "Draw descriptive spectator messages" + ], + [ + "cg_draw2D", + "Draw 2D screen elements" + ], + [ + "cg_drawBreathHint", + "Draw a 'hold breath to steady' hint" + ], + [ + "cg_drawBuildName", + "Draw build name" + ], + [ + "cg_drawCrosshair", + "Turn on weapon crosshair" + ], + [ + "cg_drawCrosshairNames", + "Draw the name of an enemy under the crosshair" + ], + [ + "cg_drawCrosshairNamesPosX", + "" + ], + [ + "cg_drawCrosshairNamesPosY", + "" + ], + [ + "cg_drawDamageDirection", + "Draw hit direction arrow." + ], + [ + "cg_drawDamageFlash", + "Draw flash when hit." + ], + [ + "cg_drawDoubleTapDetonateHint", + "Draw a 'double tap to detonate grenade' hint" + ], + [ + "cg_drawEffectNum", + "Draw counts of effects and elements" + ], + [ + "cg_drawFPS", + "Draw frames per second" + ], + [ + "cg_drawFPSLabels", + "Draw FPS Info Labels" + ], + [ + "cg_drawFriendlyHUDGrenades", + "Draw grenade warning indicators for friendly grenades (should be true if friendly-fire is enabled)" + ], + [ + "cg_drawFriendlyNames", + "Whether to show friendly names in game" + ], + [ + "cg_drawFriendlyNamesAlways", + "Whether to always show friendly names in game (for certain gametypes)" + ], + [ + "cg_drawGun", + "Draw the view model" + ], + [ + "cg_drawHealth", + "Draw health bar" + ], + [ + "cg_drawMantleHint", + "Draw a 'press key to mantle' hint" + ], + [ + "cg_drawMaterial", + "Draw debugging information for materials" + ], + [ + "cg_drawpaused", + "Draw paused screen" + ], + [ + "cg_drawScriptUsage", + "Draw debugging information for scripts" + ], + [ + "cg_drawSnapshot", + "Draw debugging information for snapshots" + ], + [ + "cg_drawStatsSource", + "Draw stats source" + ], + [ + "cg_drawTalk", + "Controls which icons CG_TALKER ownerdraw draws" + ], + [ + "cg_drawTurretCrosshair", + "Draw a cross hair when using a turret" + ], + [ + "cg_drawViewpos", + "Draw viewpos" + ], + [ + "cg_e3TrailerHacks", + "Tweaks for trailer recording" + ], + [ + "cg_equipmentSounds", + "Play equipment sounds" + ], + [ + "cg_errordecay", + "Decay for predicted error" + ], + [ + "cg_everyoneHearsEveryone", + "If true, all players can all chat together, regardless of team or death" + ], + [ + "cg_explosiveKillCamBackDist", + "Explosive kill camera: distance of camera backwards from explosive." + ], + [ + "cg_explosiveKillCamGroundBackDist", + "Explosive kill camera when stuck to ground: distance of camera backwards from explosive." + ], + [ + "cg_explosiveKillCamGroundUpDist", + "Explosive kill camera when stuck to ground: distance of camera backwards from explosive." + ], + [ + "cg_explosiveKillCamStopDecelDist", + "Rocket and Grenade Launcher kill camera: distance from player to begin coming to rest" + ], + [ + "cg_explosiveKillCamStopDist", + "Rocket and Grenade Launcher kill camera: distance from player to begin coming to rest" + ], + [ + "cg_explosiveKillCamUpDist", + "Explosive kill camera: distance of camera backwards from explosive." + ], + [ + "cg_explosiveKillCamWallOutDist", + "Explosive kill camera when stuck to wall: distance of camera out from wall." + ], + [ + "cg_explosiveKillCamWallSideDist", + "Explosive kill camera when stuck to wall: distance of camera out from wall." + ], + [ + "cg_flashbangNameFadeIn", + "Time in milliseconds to fade in friendly names" + ], + [ + "cg_flashbangNameFadeOut", + "Time in milliseconds to fade out friendly names" + ], + [ + "cg_foliagesnd_alias", + "The sound that plays when an actor or player enters a foliage clip brush." + ], + [ + "cg_footsteps", + "Play footstep sounds that are NOT sprint" + ], + [ + "cg_footstepsSprint", + "Play sprint footstep sounds" + ], + [ + "cg_fov", + "The field of view angle in degrees" + ], + [ + "cg_fovMin", + "The minimum possible field of view" + ], + [ + "cg_fovScale", + "Scale applied to the field of view" + ], + [ + "cg_friendlyNameFadeIn", + "Time in milliseconds to fade in friendly names" + ], + [ + "cg_friendlyNameFadeOut", + "Time in milliseconds to fade out friendly names" + ], + [ + "cg_gameBoldMessageWidth", + "The maximum character width of the bold game messages" + ], + [ + "cg_gameMessageWidth", + "The maximum character width of the game messages" + ], + [ + "cg_gun_x", + "Forward position of the viewmodel" + ], + [ + "cg_gun_y", + "Right position of the viewmodel" + ], + [ + "cg_gun_z", + "Up position of the viewmodel" + ], + [ + "cg_headIconMinScreenRadius", + "The minumum radius of a head icon on the screen" + ], + [ + "cg_hearKillerTime", + "Duration (in milliseconds) to hear the person you just killed" + ], + [ + "cg_hearVictimEnabled", + "If true, you can hear the person you just killed" + ], + [ + "cg_hearVictimTime", + "Duration (in milliseconds) to hear the person you just killed" + ], + [ + "cg_heliKillCamFarBlur", + "" + ], + [ + "cg_heliKillCamFarBlurDist", + "" + ], + [ + "cg_heliKillCamFarBlurStart", + "" + ], + [ + "cg_heliKillCamFov", + "Helicopter kill camera field of view." + ], + [ + "cg_heliKillCamFstop", + "Helicopter kill camera aperture. Lower f-stop yields a shallower depth of field. Typical values range from 1 to 22" + ], + [ + "cg_heliKillCamNearBlur", + "" + ], + [ + "cg_heliKillCamNearBlurEnd", + "" + ], + [ + "cg_heliKillCamNearBlurStart", + "" + ], + [ + "cg_hintFadeTime", + "Time in milliseconds for the cursor hint to fade" + ], + [ + "cg_hudChatIntermissionPosition", + "Position of the HUD chat box during intermission" + ], + [ + "cg_hudChatPosition", + "Position of the HUD chat box" + ], + [ + "cg_hudDamageIconHeight", + "The height of the damage icon" + ], + [ + "cg_hudDamageIconInScope", + "Draw damage icons when aiming down the sight of a scoped weapon" + ], + [ + "cg_hudDamageIconOffset", + "The offset from the center of the damage icon" + ], + [ + "cg_hudDamageIconOverlayTime", + "The amount of time (in ms) for the overlay portion of the damage icon to stay on screen" + ], + [ + "cg_hudDamageIconStartFadeTime", + "The amount of time (in ms) before the damage icon begins to fade" + ], + [ + "cg_hudDamageIconTime", + "The amount of time for the damage icon to stay on screen after damage is taken" + ], + [ + "cg_hudDamageIconWidth", + "The width of the damage icon" + ], + [ + "cg_hudGrenadeIconEnabledFlash", + "Show the grenade indicator for flash grenades" + ], + [ + "cg_hudGrenadeIconHeight", + "The height of the grenade indicator icon" + ], + [ + "cg_hudGrenadeIconInScope", + "Show the grenade indicator when aiming down the sight of a scoped weapon" + ], + [ + "cg_hudGrenadeIconMaxRangeFlash", + "The minimum distance that a flashbang has to be from a player in order to be shown on the grenade indicator" + ], + [ + "cg_hudGrenadeIconMaxRangeFrag", + "The minimum distance that a grenade has to be from a player in order to be shown on the grenade indicator" + ], + [ + "cg_hudGrenadeIconOffset", + "The offset from the center of the screen for a grenade icon" + ], + [ + "cg_hudGrenadeIconWidth", + "The width of the grenade indicator icon" + ], + [ + "cg_hudGrenadePointerHeight", + "The height of the grenade indicator pointer" + ], + [ + "cg_hudGrenadePointerPivot", + "The pivot point of th grenade indicator pointer" + ], + [ + "cg_hudGrenadePointerPulseFreq", + "The number of times per second that the grenade indicator flashes in Hertz" + ], + [ + "cg_hudGrenadePointerPulseMax", + "The maximum alpha of the grenade indicator pulse. Values higher than 1 will cause the indicator to remain at full brightness for longer" + ], + [ + "cg_hudGrenadePointerPulseMin", + "The minimum alpha of the grenade indicator pulse. Values lower than 0 will cause the indicator to remain at full transparency for longer" + ], + [ + "cg_hudGrenadePointerWidth", + "The width of the grenade indicator pointer" + ], + [ + "cg_hudLegacySplitscreenScale", + "Screen scale for hud elements in splitscreen" + ], + [ + "cg_hudLighting_basic_additiveLumOffset", + "[basic] Offset applied to additive light color." + ], + [ + "cg_hudLighting_basic_additiveLumScale", + "[basic] Scale applied to additive light color." + ], + [ + "cg_hudLighting_basic_additiveOffset", + "" + ], + [ + "cg_hudLighting_basic_additiveScale", + "" + ], + [ + "cg_hudLighting_basic_ambientLumOffset", + "[basic] Offset applied to ambient light color." + ], + [ + "cg_hudLighting_basic_ambientLumScale", + "[basic] Scale applied to ambient light color." + ], + [ + "cg_hudLighting_basic_ambientOffset", + "" + ], + [ + "cg_hudLighting_basic_ambientScale", + "" + ], + [ + "cg_hudLighting_basic_diffuseLumOffset", + "[basic] Offset applied to diffuse light color." + ], + [ + "cg_hudLighting_basic_diffuseLumScale", + "[basic] Scale applied to diffuse light color." + ], + [ + "cg_hudLighting_basic_diffuseOffset", + "" + ], + [ + "cg_hudLighting_basic_diffuseScale", + "" + ], + [ + "cg_hudLighting_basic_specExponent", + "[basic] Specular exponent. Higher values result in sharper highlights." + ], + [ + "cg_hudLighting_basic_specLumOffset", + "[basic] Offset applied to spec light luminance." + ], + [ + "cg_hudLighting_basic_specLumScale", + "[basic] Scale applied to spec light luminance." + ], + [ + "cg_hudLighting_basic_specOffset", + "" + ], + [ + "cg_hudLighting_basic_specScale", + "" + ], + [ + "cg_hudLighting_blood_additiveLumOffset", + "[blood] Offset applied to additive light color." + ], + [ + "cg_hudLighting_blood_additiveLumScale", + "[blood] Scale applied to additive light color." + ], + [ + "cg_hudLighting_blood_additiveOffset", + "" + ], + [ + "cg_hudLighting_blood_additiveScale", + "" + ], + [ + "cg_hudLighting_blood_ambientLumOffset", + "[blood] Offset applied to ambient light color." + ], + [ + "cg_hudLighting_blood_ambientLumScale", + "[blood] Scale applied to ambient light color." + ], + [ + "cg_hudLighting_blood_ambientOffset", + "" + ], + [ + "cg_hudLighting_blood_ambientScale", + "" + ], + [ + "cg_hudLighting_blood_diffuseLumOffset", + "[blood] Offset applied to diffuse light color." + ], + [ + "cg_hudLighting_blood_diffuseLumScale", + "[blood] Scale applied to diffuse light color." + ], + [ + "cg_hudLighting_blood_diffuseOffset", + "" + ], + [ + "cg_hudLighting_blood_diffuseScale", + "" + ], + [ + "cg_hudLighting_blood_specExponent", + "[blood] Specular exponent. Higher values result in sharper highlights." + ], + [ + "cg_hudLighting_blood_specLumOffset", + "[blood] Offset applied to spec light luminance." + ], + [ + "cg_hudLighting_blood_specLumScale", + "[blood] Scale applied to spec light luminance." + ], + [ + "cg_hudLighting_blood_specOffset", + "" + ], + [ + "cg_hudLighting_blood_specScale", + "" + ], + [ + "cg_hudLighting_fadeSharpness", + "This controls how sharp the lines are when fading using the mask alpha. Higher values are sharper." + ], + [ + "cg_hudMapBorderWidth", + "The size of the full map's border, filled by the CG_PLAYER_FULLMAP_BORDER ownerdraw" + ], + [ + "cg_hudMapFriendlyHeight", + "" + ], + [ + "cg_hudMapFriendlyWidth", + "" + ], + [ + "cg_hudMapPlayerHeight", + "" + ], + [ + "cg_hudMapPlayerWidth", + "" + ], + [ + "cg_hudMapRadarLineThickness", + "Thickness, relative to the map width, of the radar texture that sweeps across the full screen map" + ], + [ + "cg_hudObjectiveTextScale", + "" + ], + [ + "cg_hudProneY", + "Virtual screen y coordinate of the prone blocked message" + ], + [ + "cg_hudSayPosition", + "Position of the HUD say box" + ], + [ + "cg_hudSplitscreenCompassElementScale", + "Scale value to apply to compass elements in splitscreen" + ], + [ + "cg_hudSplitscreenCompassScale", + "Scale value to apply to the compass in splitscreen" + ], + [ + "cg_hudSplitscreenStanceScale", + "Scale value to apply to the stance HUD element in splitscreen" + ], + [ + "cg_hudStanceFlash", + "The background color of the flash when the stance changes" + ], + [ + "cg_hudVotePosition", + "Position of the HUD vote box" + ], + [ + "cg_invalidCmdHintBlinkInterval", + "Blink rate of an invalid command hint" + ], + [ + "cg_invalidCmdHintDuration", + "Duration of an invalid command hint" + ], + [ + "cg_javelinKillCamCloseZDist", + "Javelin kill camera: closest distance above the target." + ], + [ + "cg_javelinKillCamDownDist", + "Javelin kill camera: distance to follow during ascent." + ], + [ + "cg_javelinKillCamFov", + "Javelin kill camera: fov" + ], + [ + "cg_javelinKillCamLookLerpDist", + "Javelin kill camera: distance over which to lerp to look at player during descent. A value of zero means don't lerp at all." + ], + [ + "cg_javelinKillCamPassDist", + "Javelin kill camera: distance away when passing." + ], + [ + "cg_javelinKillCamPassTime", + "Javelin kill camera: time in seconds to pass javelin on the way up" + ], + [ + "cg_javelinKillCamUpDist", + "Javelin kill camera: distance to follow during ascent." + ], + [ + "cg_killCamDefaultLerpTime", + "Default time used to lerp between killcam entities." + ], + [ + "cg_killCamTurretLerpTime", + "Time used to lerp to a killcam entity of the TURRET type." + ], + [ + "cg_landingSounds", + "Play landing on surface sounds" + ], + [ + "cg_largeExplosiveKillCamBackDist", + "Large Explosive kill camera: distance of camera backwards from explosive." + ], + [ + "cg_largeExplosiveKillCamUpDist", + "Large Explosive kill camera: distance of camera backwards from explosive." + ], + [ + "cg_mapLocationSelectionCursorSpeed", + "Speed of the cursor when selecting a location on the map" + ], + [ + "cg_marks_ents_player_only", + "Marks on entities from players' bullets only." + ], + [ + "cg_minCullBulletDist", + "Don't cull bullet trajectories that are within this distance to you." + ], + [ + "cg_objectiveText", + "" + ], + [ + "cg_overheadIconSize", + "The maximum size to show overhead icons like 'rank'" + ], + [ + "cg_overheadNamesFarDist", + "The far distance at which name sizes are scaled by cg_overheadNamesFarScale" + ], + [ + "cg_overheadNamesFarScale", + "The amount to scale overhead name sizes at cg_overheadNamesFarDist" + ], + [ + "cg_overheadNamesFont", + "Font for overhead names ( see menudefinition.h )" + ], + [ + "cg_overheadNamesGlow", + "Glow color for overhead names" + ], + [ + "cg_overheadNamesMaxDist", + "The maximum distance for showing friendly player names" + ], + [ + "cg_overheadNamesNearDist", + "The near distance at which names are full size" + ], + [ + "cg_overheadNamesSize", + "The maximum size to show overhead names" + ], + [ + "cg_overheadRankSize", + "The size to show rank text" + ], + [ + "cg_remoteMissileKillCamBackDist", + "Remote missile kill camera: distance of camera backwards from rocket." + ], + [ + "cg_remoteMissileKillCamUpDist", + "Remote missile kill camera: distance of camera backwards from rocket." + ], + [ + "cg_rocketKillCamBackDist", + "Rocket kill camera: distance of camera backwards from rocket." + ], + [ + "cg_rocketKillCamUpDist", + "Rocket kill camera: distance of camera backwards from rocket." + ], + [ + "cg_scriptIconSize", + "Size of Icons defined by script" + ], + [ + "cg_showmiss", + "Show prediction errors" + ], + [ + "cg_sprintMeterDisabledColor", + "The color of the sprint meter when the sprint meter is full" + ], + [ + "cg_sprintMeterEmptyColor", + "The color of the sprint meter when the sprint meter is full" + ], + [ + "cg_sprintMeterFullColor", + "The color of the sprint meter when the sprint meter is full" + ], + [ + "cg_subtitleMinTime", + "The minimum time that the subtitles are displayed on screen in seconds" + ], + [ + "cg_subtitleWidthStandard", + "The width of the subtitles on a non wide-screen" + ], + [ + "cg_subtitleWidthWidescreen", + "The width of the subtitle on a wide-screen" + ], + [ + "cg_teamChatsOnly", + "Allow chatting only on the same team" + ], + [ + "cg_TeamColor_Allies", + "Allies team color" + ], + [ + "cg_TeamColor_Axis", + "Axis team color" + ], + [ + "cg_TeamColor_EnemyTeam", + "Enemy team color" + ], + [ + "cg_TeamColor_Free", + "Free Team color" + ], + [ + "cg_TeamColor_MyParty", + "Player team color when in the same party" + ], + [ + "cg_TeamColor_MyTeam", + "Player team color" + ], + [ + "cg_TeamColor_Spectator", + "Spectator team color" + ], + [ + "cg_turretKillCamBackDist", + "Turret kill camera: distance of camera backwards from Turret." + ], + [ + "cg_turretKillCamFov", + "Turret kill camera field of view." + ], + [ + "cg_turretKillCamUpDist", + "Turret kill camera: distance of camera backwards from Turret." + ], + [ + "cg_turretRemoteKillCamBackDist", + "Remote Turret kill camera: distance of camera backwards from Turret." + ], + [ + "cg_turretRemoteKillCamFov", + "Remote Turret kill camera field of view." + ], + [ + "cg_turretRemoteKillCamUpDist", + "Remote Turret kill camera: distance of camera backwards from Turret." + ], + [ + "cg_vectorFieldsForceUniform", + "Forces all vector field assets to represent a single, uniform direction" + ], + [ + "cg_viewVehicleInfluence", + "The influence on the view angles from being in a vehicle" + ], + [ + "cg_viewZSmoothingMax", + "Threshhold for the maximum smoothing distance we'll do" + ], + [ + "cg_viewZSmoothingMin", + "Threshhold for the minimum smoothing distance it must move to smooth" + ], + [ + "cg_viewZSmoothingTime", + "Amount of time to spread the smoothing over" + ], + [ + "cg_voiceIconSize", + "Size of the 'voice' icon" + ], + [ + "cg_waterSheeting_distortionScaleFactor", + "Distortion uv scales (Default to 1)" + ], + [ + "cg_waterSheeting_magnitude", + "Distortion magnitude" + ], + [ + "cg_waterSheeting_radius", + "Tweak dev var; Glow radius in pixels at 640x480" + ], + [ + "cg_weapHitCullAngle", + "Angle of cone within which to cull back facing weapon hit effects" + ], + [ + "cg_weapHitCullEnable", + "When true, cull back facing weapon hit fx." + ], + [ + "cg_weaponCycleDelay", + "The delay after cycling to a new weapon to prevent holding down the cycle weapon button from cycling too fast" + ], + [ + "cg_weaponHintsCoD1Style", + "Draw weapon hints in CoD1 style: with the weapon name, and with the icon below" + ], + [ + "cg_weaponVisInterval", + "Do weapon vis checks once per this many frames, per centity" + ], + [ + "cg_youInKillCamSize", + "Size of the 'you' Icon in the kill cam" + ], + [ + "cl_anglespeedkey", + "Multiplier for max angle speed for game pad and keyboard" + ], + [ + "cl_bypassMouseInput", + "Bypass UI mouse input and send directly to the game" + ], + [ + "cl_connectionAttempts", + "Maximum number of connection attempts before aborting" + ], + [ + "cl_connectTimeout", + "Timeout time in seconds while connecting to a server" + ], + [ + "cl_demo_uploadfb", + "Should we upload to FB" + ], + [ + "cl_dirSelConvergenceTime", + "Time to converge to the new direction when selecting a direction on the map." + ], + [ + "cl_force_paused", + "Force the client to be paused. Can't be overridden by LUA scripts, the start button, etc." + ], + [ + "cl_freelook", + "Enable looking with mouse" + ], + [ + "cl_hudDrawsBehindUI", + "Should the HUD draw when the UI is up?" + ], + [ + "cl_ingame", + "True if the game is active" + ], + [ + "cl_inhibit_stats_upload", + "Inhibit upload of stats during demo playback" + ], + [ + "cl_lessprint", + "Print less to the console by filtering out certain spammy channels" + ], + [ + "cl_maxpackets", + "Maximum number of packets sent per frame" + ], + [ + "cl_maxPing", + "Maximum ping for the client" + ], + [ + "cl_migrationTimeout", + "Seconds to wait to hear from new host during host migration before timeout occurs" + ], + [ + "cl_modifiedDebugPlacement", + "Modify the location of debug output (outside of safe area)" + ], + [ + "cl_motdString", + "" + ], + [ + "cl_mouseAccel", + "Mouse acceleration" + ], + [ + "cl_noprint", + "Print nothing to the console" + ], + [ + "cl_packetdup", + "Enable packet duplication" + ], + [ + "cl_pauseAudioZoneEnabled", + "Enable the paused audio zone when the menus are up" + ], + [ + "cl_paused", + "Pause the game" + ], + [ + "cl_pitchspeed", + "Max pitch speed in degrees for game pad" + ], + [ + "cl_pranks", + "pranks" + ], + [ + "cl_pushToTalk", + "Do we have to press a button to talk" + ], + [ + "cl_serverStatusResendTime", + "Time in milliseconds to resend a server status message" + ], + [ + "cl_showmouserate", + "Print mouse rate debugging information to the console" + ], + [ + "cl_textChatEnabled", + "Do we want to use text chat" + ], + [ + "cl_timeout", + "Seconds with no received packets until a timeout occurs" + ], + [ + "cl_voice", + "Use voice communications" + ], + [ + "cl_yawspeed", + "Max yaw speed in degrees for game pad and keyboard" + ], + [ + "clientSideEffects", + "Enable loading _fx.gsc files on the client" + ], + [ + "cod_anywhere_errorMessage", + "CoD Anywhere error message" + ], + [ + "cod_anywhere_showPopup", + "Temp Development: Should we show the CoD Anywhere popup" + ], + [ + "cod_anywhere_single_task_popup_text", + "CoD Anywhere success message" + ], + [ + "com_animCheck", + "Check anim tree" + ], + [ + "com_cinematicEndInWhite", + "Set by script. True if cinematic ends with a white screen." + ], + [ + "com_completionResolveCommand", + "Command to run when the message box successfully closes" + ], + [ + "com_errorMessage", + "Most recent error message" + ], + [ + "com_errorResolveCommand", + "Command to run when they close the error box" + ], + [ + "com_filter_output", + "Use console filters for filtering output." + ], + [ + "com_maxfps", + "Cap frames per second" + ], + [ + "com_maxFrameTime", + "Time slows down if a frame takes longer than this many milliseconds" + ], + [ + "com_playerProfile", + "Set to the name of the profile" + ], + [ + "com_recommendedSet", + "" + ], + [ + "commerce_dl_retry_step", + "Step in m/s for the commerce download retry" + ], + [ + "commerce_manifest_file_max_retry_time", + "Max time that the commerce manifest can retry" + ], + [ + "commerce_manifest_file_retry_step", + "Step in m/s for the commerce manifest retry" + ], + [ + "commerce_max_dl_retry_time", + "Max time that the commerce download can retry" + ], + [ + "commerce_max_retry_time", + "Max time that the commerce upload can retry" + ], + [ + "commerce_retry_step", + "Step in m/s for the commerce upload retry" + ], + [ + "compass", + "Display Compass" + ], + [ + "compassClampIcons", + "If true, friendlies and enemy pings clamp to the edge of the radar. If false, they disappear off the edge." + ], + [ + "compassCoords", + "x = North-South coord base value, \ny = East-West coord base value, \nz = scale (game units per coord unit)" + ], + [ + "compassECoordCutoff", + "Left cutoff for the scrolling east-west coords" + ], + [ + "compassFriendlyHeight", + "" + ], + [ + "compassFriendlyWidth", + "" + ], + [ + "compassHideSansObjectivePointer", + "Hide the compass, but leave the obective pointer visible." + ], + [ + "compassHideVehicles", + "When enabled, disables the CG_PLAYER_COMPASS_VEHICLES ownerdraw." + ], + [ + "compassMaxRange", + "The maximum range from the player in world space that objects will be shown on the compass" + ], + [ + "compassMinRadius", + "The minimum radius from the center of the compass that objects will appear." + ], + [ + "compassMinRange", + "The minimum range from the player in world space that objects will appear on the compass" + ], + [ + "compassObjectiveArrowHeight", + "" + ], + [ + "compassObjectiveArrowOffset", + "The offset of the objective arrow inward from the edge of the compass map" + ], + [ + "compassObjectiveArrowRotateDist", + "Distance from the corner of the compass map at which the objective arrow rotates to 45 degrees" + ], + [ + "compassObjectiveArrowWidth", + "" + ], + [ + "compassObjectiveDetailDist", + "When an objective is closer than this distance (in meters), the icon will not be drawn on the tickertape." + ], + [ + "compassObjectiveDrawLines", + "Draw horizontal and vertical lines to the active target, if it is within the minimap boundries" + ], + [ + "compassObjectiveHeight", + "" + ], + [ + "compassObjectiveIconHeight", + "" + ], + [ + "compassObjectiveIconWidth", + "" + ], + [ + "compassObjectiveMaxHeight", + "The maximum height that an objective is considered to be on this level" + ], + [ + "compassObjectiveMaxRange", + "The maximum range at which an objective is visible on the compass" + ], + [ + "compassObjectiveMinAlpha", + "The minimum alpha for an objective at the edge of the compass" + ], + [ + "compassObjectiveMinDistRange", + "The distance that objective transition effects play over, centered on compassObjectiveNearbyDist." + ], + [ + "compassObjectiveMinHeight", + "The minimum height that an objective is considered to be on this level" + ], + [ + "compassObjectiveNearbyDist", + "When an objective is closer than this distance (in meters), the icon will not be drawn on the tickertape." + ], + [ + "compassObjectiveNumRings", + "The number of rings when a new objective appears" + ], + [ + "compassObjectiveRingSize", + "The maximum objective ring sige when a new objective appears on the compass" + ], + [ + "compassObjectiveRingTime", + "The amount of time between each ring when an objective appears" + ], + [ + "compassObjectiveTextHeight", + "Objective text height" + ], + [ + "compassObjectiveTextScale", + "Scale to apply to hud objectives" + ], + [ + "compassObjectiveWidth", + "" + ], + [ + "compassObjectiveWraparoundTime", + "How long it takes for the objective to wrap around the compass from one edge to the other" + ], + [ + "compassPlayerHeight", + "" + ], + [ + "compassPlayerWidth", + "" + ], + [ + "compassRadarLineThickness", + "Thickness, relative to the compass size, of the radar texture that sweeps across the map" + ], + [ + "compassRadarPingFadeTime", + "How long an enemy is visible on the compass after it is detected by radar" + ], + [ + "compassRotation", + "Style of compass" + ], + [ + "compassSize", + "Scale the compass" + ], + [ + "compassSoundPingFadeTime", + "The time in seconds for the sound overlay on the compass to fade" + ], + [ + "compassTickertapeStretch", + "How far the tickertape should stretch from its center." + ], + [ + "comscore_active", + "Are we allowed to enable ComScore tracking or not" + ], + [ + "con_gameMsgWindow0FadeInTime", + "" + ], + [ + "con_gameMsgWindow0FadeOutTime", + "" + ], + [ + "con_gameMsgWindow0Filter", + "" + ], + [ + "con_gameMsgWindow0LineCount", + "" + ], + [ + "con_gameMsgWindow0MsgTime", + "" + ], + [ + "con_gameMsgWindow0ScrollTime", + "" + ], + [ + "con_gameMsgWindow1FadeInTime", + "" + ], + [ + "con_gameMsgWindow1FadeOutTime", + "" + ], + [ + "con_gameMsgWindow1Filter", + "" + ], + [ + "con_gameMsgWindow1LineCount", + "" + ], + [ + "con_gameMsgWindow1MsgTime", + "" + ], + [ + "con_gameMsgWindow1ScrollTime", + "" + ], + [ + "con_gameMsgWindow2FadeInTime", + "" + ], + [ + "con_gameMsgWindow2FadeOutTime", + "" + ], + [ + "con_gameMsgWindow2Filter", + "" + ], + [ + "con_gameMsgWindow2LineCount", + "" + ], + [ + "con_gameMsgWindow2MsgTime", + "" + ], + [ + "con_gameMsgWindow2ScrollTime", + "" + ], + [ + "con_gameMsgWindow3FadeInTime", + "" + ], + [ + "con_gameMsgWindow3FadeOutTime", + "" + ], + [ + "con_gameMsgWindow3Filter", + "" + ], + [ + "con_gameMsgWindow3LineCount", + "" + ], + [ + "con_gameMsgWindow3MsgTime", + "" + ], + [ + "con_gameMsgWindow3ScrollTime", + "" + ], + [ + "con_subtitleLeading", + "Leading for subtitles, calculated as a percentage of the font height" + ], + [ + "con_typewriterColorGlowCheckpoint", + "" + ], + [ + "con_typewriterColorGlowCompleted", + "" + ], + [ + "con_typewriterColorGlowFailed", + "" + ], + [ + "con_typewriterColorGlowUpdated", + "" + ], + [ + "con_typewriterColorInteriorCheckpoint", + "" + ], + [ + "con_typewriterColorInteriorCompleted", + "" + ], + [ + "con_typewriterColorInteriorFailed", + "" + ], + [ + "con_typewriterColorInteriorUpdated", + "" + ], + [ + "con_typewriterDecayDuration", + "Time (in milliseconds) to spend disolving the line away." + ], + [ + "con_typewriterDecayStartTime", + "Time (in milliseconds) to spend between the build and disolve phases." + ], + [ + "con_typewriterPrintSpeed", + "Time (in milliseconds) to print each letter in the line." + ], + [ + "counterDownloadInterval", + "Number of minutes before all the global counters are uploaded" + ], + [ + "counterUploadInterval", + "Number of minutes before all the global counters are uploaded" + ], + [ + "cpu_speed_12players", + "12 player sys_configureGHz req'd" + ], + [ + "cpu_speed_18players", + "18 player sys_configureGHz req'd" + ], + [ + "cpu_speed_8players", + "8 player sys_configureGHz req'd" + ], + [ + "cSplineDebugRender", + "Debug Render the csplines." + ], + [ + "cSplineDebugRenderCorridor", + "Debug Render the cspline corridor." + ], + [ + "cSplineDebugRenderData", + "Debug Render the cspline data." + ], + [ + "cSplineDebugRenderSplineId", + "Select a cspline - 0 for all." + ], + [ + "dailychallenge_killswitch", + "daily challenge killswitch - int with bits used to flag individual daily challenges as enabled" + ], + [ + "dailychallenge_killswitch2", + "daily challenge killswitch2 - int with bits used to flag 2nd set of individual daily challenges as enabled" + ], + [ + "dailychallenge_period", + "daily challenge period - utc value for a day" + ], + [ + "data_validation_allow_drop", + "" + ], + [ + "dc_lobbymerge", + "Allows lobby merging across data centres" + ], + [ + "dcacheSimulateNoHDD", + "When turned on, simulate no HDD for caching." + ], + [ + "dcacheThrottleEnabled", + "Enable or disable dcache upload throttling." + ], + [ + "dcacheThrottleKBytesPerSec", + "Dcache upload throttle limit in K Bytes per second." + ], + [ + "dedicated_dhclient", + "True if we're a client playing on a DH server" + ], + [ + "demonwareConsideredConnectedTime", + "Number of milliseconds after being disconnected from demonware before considering shutting down." + ], + [ + "developer", + "Enable development options" + ], + [ + "didyouknow", + "" + ], + [ + "discard_playerstats_on_suspend", + "Forces stats discard on suspend" + ], + [ + "drawEntityCount", + "Enable entity count drawing" + ], + [ + "drawEntityCountPos", + "Where to draw the entity count graph" + ], + [ + "drawEntityCountSize", + "undefined" + ], + [ + "drawKillcamData", + "Enable drawing server killcam data" + ], + [ + "drawKillcamDataPos", + "Where to draw the server killcam graph" + ], + [ + "drawKillcamDataSize", + "How big to draw the killcam data graph" + ], + [ + "drawServerBandwidth", + "Enable drawing server bandwidth" + ], + [ + "drawServerBandwidthPos", + "Where to draw the server bandwidth graph" + ], + [ + "ds_dcid", + "optional datacenter id - from playlist" + ], + [ + "ds_dcid_override", + "force datacenter id" + ], + [ + "ds_info", + "ds info string" + ], + [ + "ds_info_enable", + "Enable ds info string" + ], + [ + "ds_introRequestTimeout", + "ds intro request timeout (ms)" + ], + [ + "ds_keepaliveInterval", + "ds keepalive interval (ms)" + ], + [ + "ds_keepaliveTimeout", + "ds keepalive timeout (ms)" + ], + [ + "ds_pingclient_max_reping_distance", + "don't re-ping a datacenter if it's further away than this (miles)" + ], + [ + "ds_pingclient_max_repings", + "max # of times to re-ping a datacenter" + ], + [ + "ds_pingclient_maxpings", + "max pings to send per datacenter" + ], + [ + "ds_pingclient_maxpings_per_tick", + "max new pings each tick" + ], + [ + "ds_pingclient_min_reping_delay", + "min msec delay between re-pings" + ], + [ + "ds_pingclient_min_reping_latency", + "don't re-ping a datacenter if latency is less than this" + ], + [ + "ds_pingclient_minpings", + "min responses required per datacenter" + ], + [ + "ds_pingclient_odsf", + "does dsping set odsf flag" + ], + [ + "dsping_dc_0", + "" + ], + [ + "dsping_dc_1", + "" + ], + [ + "dsping_dc_10", + "" + ], + [ + "dsping_dc_11", + "" + ], + [ + "dsping_dc_12", + "" + ], + [ + "dsping_dc_13", + "" + ], + [ + "dsping_dc_14", + "" + ], + [ + "dsping_dc_15", + "" + ], + [ + "dsping_dc_16", + "" + ], + [ + "dsping_dc_17", + "" + ], + [ + "dsping_dc_18", + "" + ], + [ + "dsping_dc_19", + "" + ], + [ + "dsping_dc_2", + "" + ], + [ + "dsping_dc_20", + "" + ], + [ + "dsping_dc_21", + "" + ], + [ + "dsping_dc_22", + "" + ], + [ + "dsping_dc_23", + "" + ], + [ + "dsping_dc_24", + "" + ], + [ + "dsping_dc_25", + "" + ], + [ + "dsping_dc_26", + "" + ], + [ + "dsping_dc_27", + "" + ], + [ + "dsping_dc_28", + "" + ], + [ + "dsping_dc_29", + "" + ], + [ + "dsping_dc_3", + "" + ], + [ + "dsping_dc_30", + "" + ], + [ + "dsping_dc_31", + "" + ], + [ + "dsping_dc_32", + "" + ], + [ + "dsping_dc_33", + "" + ], + [ + "dsping_dc_34", + "" + ], + [ + "dsping_dc_35", + "" + ], + [ + "dsping_dc_36", + "" + ], + [ + "dsping_dc_37", + "" + ], + [ + "dsping_dc_38", + "" + ], + [ + "dsping_dc_39", + "" + ], + [ + "dsping_dc_4", + "" + ], + [ + "dsping_dc_40", + "" + ], + [ + "dsping_dc_41", + "" + ], + [ + "dsping_dc_42", + "" + ], + [ + "dsping_dc_43", + "" + ], + [ + "dsping_dc_44", + "" + ], + [ + "dsping_dc_45", + "" + ], + [ + "dsping_dc_46", + "" + ], + [ + "dsping_dc_47", + "" + ], + [ + "dsping_dc_48", + "" + ], + [ + "dsping_dc_49", + "" + ], + [ + "dsping_dc_5", + "" + ], + [ + "dsping_dc_50", + "" + ], + [ + "dsping_dc_51", + "" + ], + [ + "dsping_dc_52", + "" + ], + [ + "dsping_dc_53", + "" + ], + [ + "dsping_dc_54", + "" + ], + [ + "dsping_dc_55", + "" + ], + [ + "dsping_dc_56", + "" + ], + [ + "dsping_dc_57", + "" + ], + [ + "dsping_dc_58", + "" + ], + [ + "dsping_dc_59", + "" + ], + [ + "dsping_dc_6", + "" + ], + [ + "dsping_dc_60", + "" + ], + [ + "dsping_dc_61", + "" + ], + [ + "dsping_dc_62", + "" + ], + [ + "dsping_dc_63", + "" + ], + [ + "dsping_dc_7", + "" + ], + [ + "dsping_dc_8", + "" + ], + [ + "dsping_dc_9", + "" + ], + [ + "dvl", + "Enables the data validation system. Only available in non-retail builds." + ], + [ + "dw_addrHandleTimeout", + "Delay before destroying an addrHandle after the connection is lost\n" + ], + [ + "dw_leaderboard_write_active", + "Are leaderboard writes enabled" + ], + [ + "dw_presence_active", + "Is the demonware presence system enabled" + ], + [ + "dw_presence_coop_join_active", + "Do we allow players to join on presence for private coop matches (post session to demonware" + ], + [ + "dw_presence_get_delay", + "Number of milliseconds to wait after booting the game to fetch demonware presence" + ], + [ + "dw_presence_get_rate", + "Number of milliseconds to wait between sending presence state to demonware" + ], + [ + "dw_presence_put_delay", + "Number of milliseconds to wait in a presence state before sending to demonware" + ], + [ + "dw_presence_put_rate", + "Number of milliseconds to wait between sending presence state to demonware" + ], + [ + "dw_region_lookup_timeout", + "Timeout (in MS) after which we will accept not having found a region code and use the default" + ], + [ + "dw_shared_presence_active", + "Is the demonware shared presence system enabled" + ], + [ + "dw_shared_presence_get_delay", + "Number of milliseconds to wait after booting the game to fetch demonware presence" + ], + [ + "dw_shared_presence_get_rate", + "Number of milliseconds to wait between sending presence state to demonware" + ], + [ + "dw_shared_presence_put_delay", + "Number of milliseconds to wait in a shared presence state before sending to demonware" + ], + [ + "dw_shared_presence_put_rate", + "Number of milliseconds to wait between sending presence state to demonware" + ], + [ + "dwBandwidthTestTaskTimeout", + "default timeout for the bandwidth test task (in ms). 0 means no timeout" + ], + [ + "dynEnt_active", + "Disable/enable dynent reactions" + ], + [ + "dynEnt_playerWakeUpRadius", + "Determines threshold distance from player within which all dynents are woken up." + ], + [ + "dynEnt_playerWakeUpZOffset", + "Determines vertical distance from player's feet from which wake up sphere is centered." + ], + [ + "elite_clan_active", + "Are we allowed to show Elite Clans or not" + ], + [ + "elite_clan_cool_off_time", + "Cool off time between calls to fetch the elite clan" + ], + [ + "elite_clan_delay", + "Delay before the bdTeams calls start to Demonware. -1 means On-Demand and it will wait until the 'starteliteclan' menu call" + ], + [ + "elite_clan_division_icon_active", + "Are we allowed to show Elite Clan division icon or not" + ], + [ + "elite_clan_get_blob_profile_max_retry_time", + "Max time that the Elite Clan get private profile can retry" + ], + [ + "elite_clan_get_blob_profile_retry_step", + "Step in m/s for the Elite Clan get private profile retry" + ], + [ + "elite_clan_get_clan_max_retry_time", + "Max time that the Elite Clan get clan can retry" + ], + [ + "elite_clan_get_clan_retry_step", + "Step in m/s for the Elite Clan get clan retry" + ], + [ + "elite_clan_get_members_max_retry_time", + "Max time that the Elite Clan get members can retry" + ], + [ + "elite_clan_get_members_retry_step", + "Step in m/s for the Elite Clan get members retry" + ], + [ + "elite_clan_get_private_member_profile_max_retry_time", + "Max time that the Elite Clan get private profile can retry" + ], + [ + "elite_clan_get_private_member_profile_retry_step", + "Step in m/s for the Elite Clan get private profile retry" + ], + [ + "elite_clan_get_public_profile_max_retry_time", + "Max time that the Elite Clan get public profile can retry" + ], + [ + "elite_clan_get_public_profile_retry_step", + "Step in m/s for the Elite Clan get public profile retry" + ], + [ + "elite_clan_get_team_stats_max_retry_time", + "Max time that the Elite Clan get team stats can retry" + ], + [ + "elite_clan_get_team_stats_retry_step", + "Step in m/s for the Elite Clan get team stats retry" + ], + [ + "elite_clan_motd_throttle_time", + "Throttle time between motd update calls" + ], + [ + "elite_clan_remote_view_active", + "Are we allowed to view the clans for remote players" + ], + [ + "elite_clan_remote_view_max_retry_time", + "Max time that the Elite Clan remote viewing can retry" + ], + [ + "elite_clan_remote_view_retry_step", + "Step in m/s for the retry for viewing a remote Elite Clan" + ], + [ + "elite_clan_send_message_to_members_max_retry_time", + "Max time that the Elite Clan send message to members can retry" + ], + [ + "elite_clan_send_message_to_members_retry_step", + "Step in m/s for the Elite Clan send message to members retry" + ], + [ + "elite_clan_set_private_member_profile_max_retry_time", + "Max time that the Elite Clan set private member profile can retry" + ], + [ + "elite_clan_set_private_member_profile_retry_step", + "Step in m/s for the Elite Clan set private member profile retry" + ], + [ + "elite_clan_single_task_popup_text", + "String to be displayed on popup when a single task is being performed" + ], + [ + "elite_clan_using_title", + "Stores whether the Elite Clan title is in use by the user" + ], + [ + "emblems_active", + "Are we allowed to enable Emblems or not" + ], + [ + "enable_recordRecentActivity", + "records the timestamp of when the player was recently active to the tracker leaderboards" + ], + [ + "enableReportingRegisteredParties", + "If true then party membership data and host status will be reported in matchdata blob." + ], + [ + "entitlements_active", + "Are we allowed to show Entitlements or not" + ], + [ + "entitlements_config_file_max_retry_time", + "Max time that the Entitlements config file read can retry" + ], + [ + "entitlements_config_file_retry_step", + "Step in m/s for the Entitlements config file read retry" + ], + [ + "entitlements_cool_off_time", + "Cool off time between calls to fetch the elite clan" + ], + [ + "entitlements_delay", + "Delay before the entitlement calls start to Demonware. -1 means On-Demand and it will wait until the 'startentitlements' menu call" + ], + [ + "entitlements_key_archive_max_retry_time", + "Max time that the Entitlements key archive read can retry" + ], + [ + "entitlements_key_archive_retry_step", + "Step in m/s for the Entitlements key archive read retry" + ], + [ + "entitlementSystemOk", + "Set by the game to inform that the entitlement system is initialised" + ], + [ + "facebook_active", + "Are we allowed to show Facebook or not" + ], + [ + "facebook_delay", + "Delay before the Facebook calls start to Demonware. -1 means On-Demand and it will wait until the 'startfacebook' menu call" + ], + [ + "facebook_friends_active", + "Are we allowed to show Facebook Friends or not" + ], + [ + "facebook_friends_max_retry_time", + "Max time that the Facebook friends read can retry" + ], + [ + "facebook_friends_refresh_time", + "Time in seconds between Facebook friend refreshes" + ], + [ + "facebook_friends_retry_step", + "Step in m/s for the Facebook friends read retry" + ], + [ + "facebook_friends_showing_count", + "Contains how many facebook friends are being shown in the UI." + ], + [ + "facebook_friends_throttle_time", + "Throttle time between Facebook friend pages" + ], + [ + "facebook_max_retry_time", + "Max time that the Facebook authentication can retry" + ], + [ + "facebook_password", + "Facebook Password" + ], + [ + "facebook_password_asterisk", + "Facebook Password (Asterisk Version)" + ], + [ + "facebook_popup_text", + "Facebook Popup Text" + ], + [ + "facebook_retry_step", + "Step in m/s for the Facebook authentication retry" + ], + [ + "facebook_upload_photo_active", + "Are we allowed to Upload Photos to Facebook or not" + ], + [ + "facebook_upload_video_active", + "Are we allowed to Upload Videos to Facebook or not" + ], + [ + "facebook_username", + "Facebook Username" + ], + [ + "fixedtime", + "Use a fixed time rate for each frame" + ], + [ + "FoFIconMaxSize", + "Maximum size a Friend-or-Foe icon should ever grow to." + ], + [ + "FoFIconMinSize", + "Minimum size a Friend-or-Foe icon should ever shrink to." + ], + [ + "FoFIconScale", + "Base scale of Friend-or-Foe icons." + ], + [ + "FoFIconSpawnTimeDelay", + "How long to wait, after spawning, before showing the Friend-or-Foe icon on a player." + ], + [ + "FoFIconSpawnTimeFade", + "Length of the Friend-or-Foe icons' fade-ins." + ], + [ + "friendsCacheSteamFriends", + "Use cache of steam friends before querying steam api" + ], + [ + "friendsMaxSteamLookupsPerFrame", + "Number of steam friends to query steam status per frame when doing a refresh.\n" + ], + [ + "friendsWidgetMinimumRefreshTimer", + "Minimum delay before refreshing friends data if you aren't on the friends screen\n" + ], + [ + "fs_basegame", + "Base game name" + ], + [ + "fs_basepath", + "Base game path" + ], + [ + "fs_basepath_output", + "Base game path" + ], + [ + "fs_cdpath", + "CD path" + ], + [ + "fs_copyfiles", + "Copy all used files to another location" + ], + [ + "fs_debug", + "Enable file system debugging information" + ], + [ + "fs_game", + "Game data directory. Must be \"\" or a sub directory of 'mods/'. Use \"Mods\" menu to change." + ], + [ + "fs_homepath", + "Game home path" + ], + [ + "fs_ignoreLocalized", + "Ignore localized assets" + ], + [ + "fx_alphaThreshold", + "Don't draw billboard sprites, oriented sprites or tails with alpha below this threshold (0-256)." + ], + [ + "fx_cast_shadow", + "Enable transparency shadow mapping from script" + ], + [ + "fx_count", + "Debug effects count" + ], + [ + "fx_cull_elem_draw", + "Culls effect elems for drawing" + ], + [ + "fx_cull_elem_draw_flicker", + "Flicker DPVS culled effect elems" + ], + [ + "fx_cull_elem_spawn", + "Culls effect elems for spawning" + ], + [ + "fx_debugBolt", + "Debug effects bolt" + ], + [ + "fx_deferelem", + "Toggles deferred processing of elements instead of effects" + ], + [ + "fx_dpvs_cull_elem_draw", + "Culls effect elems for drawing using DPVS(2: ignore per-effect portal culling flag)" + ], + [ + "fx_draw", + "" + ], + [ + "fx_draw_omniLight", + "" + ], + [ + "fx_draw_simd", + "Draw effects using SIMD / Vector code." + ], + [ + "fx_draw_spotLight", + "" + ], + [ + "fx_drawClouds", + "Toggles the drawing of particle clouds" + ], + [ + "fx_enable", + "Toggles all effects processing" + ], + [ + "fx_flare", + "Toggles fx flare" + ], + [ + "fx_freeze", + "Freeze effects" + ], + [ + "fx_killEffectOnRewind", + "Causes effects that have been marked for a soft kill (fade out) to be killed immediately on a rewind." + ], + [ + "fx_lightGridSampleOffset", + "the length of effect sample's offset along X Axis" + ], + [ + "fx_mark_profile", + "Turn on FX profiling for marks (specify which local client, with '1' being the first.)" + ], + [ + "fx_marks", + "Toggles whether bullet hits leave marks" + ], + [ + "fx_marks_ents", + "Toggles whether bullet hits leave marks" + ], + [ + "fx_marks_nearlimit", + "Sets limit of number of decals that can exist at the same location (0 for unlimited)" + ], + [ + "fx_marks_smodels", + "Toggles whether bullet hits leave marks" + ], + [ + "fx_physicsImpactVelocityThreshold", + "Set the min normal velocity threshold in order for model physics fx to generate child impact effects." + ], + [ + "fx_profile", + "Turn on FX profiling (specify which local client, with '1' being the first.)" + ], + [ + "fx_profileFilter", + "Only show effects with this as a substring in FX profile" + ], + [ + "fx_profileFilterElemCountZero", + "Do not include FX that have a zero element count" + ], + [ + "fx_profileSkip", + "Skip the first n lines in FX profile (to see ones off bottom of screen)" + ], + [ + "fx_profileSort", + "Choose sort criteria for FX profiling" + ], + [ + "fx_showLightGridSampleOffset", + "show light grid sample offset in CreateFX mode" + ], + [ + "fx_visMinTraceDist", + "Minimum visibility trace size" + ], + [ + "g_ai", + "Enable AI" + ], + [ + "g_allowVote", + "Enable voting on this server" + ], + [ + "g_atmosFogDistanceScaleReadOnly", + "scale applied to scene distance used for atmospheric fog calculation" + ], + [ + "g_atmosFogEnabledReadOnly", + "use atmospheric fog" + ], + [ + "g_atmosFogExtinctionStrengthReadOnly", + "scale out scatter contribution of atmospheric fog" + ], + [ + "g_atmosFogHalfPlaneDistanceReadOnly", + "distance at which atmospheric fog contributes half the pixels color" + ], + [ + "g_atmosFogHazeSpreadReadOnly", + "directionality of haze (1ReadOnly = all forward scatter, 0ReadOnly = all back scatter)" + ], + [ + "g_atmosFogHazeStrengthReadOnly", + "portion of atmospheric fog density that is haze (0ReadOnly = all fog, 1ReadOnly = all haze)" + ], + [ + "g_atmosFogHeightFogBaseHeightReadOnly", + "height fog is full density at this world height and below" + ], + [ + "g_atmosFogHeightFogEnabledReadOnly", + "use height for atmospheric fog" + ], + [ + "g_atmosFogHeightFogHalfPlaneDistanceReadOnly", + "at this distance above g_atmosFogHeightFogBaseHeight, height fog density is half" + ], + [ + "g_atmosFogInScatterStrengthReadOnly", + "scale in scatter contribution of atmospheric fog" + ], + [ + "g_atmosFogSkyAngularFalloffEnabledReadOnly", + "use angular sky falloff for atmospheric fog" + ], + [ + "g_atmosFogSkyDistanceReadOnly", + "distance used for sky box when applying atmospheric fog" + ], + [ + "g_atmosFogSkyFalloffAngleRangeReadOnly", + "sky fog angular falloff angle range sky fog falls off over this range from the start angle" + ], + [ + "g_atmosFogSkyFalloffStartAngleReadOnly", + "sky fog angular falloff start angle (full strength fog at this angle)" + ], + [ + "g_atmosFogStartDistanceReadOnly", + "distance from camera at which fog contribution begins" + ], + [ + "g_atmosFogSunDirectionReadOnly", + "sun direction used when calculating atmospheric fog" + ], + [ + "g_banIPs", + "IP addresses to ban from playing" + ], + [ + "g_clonePlayerMaxVelocity", + "Maximum velocity in each axis of a cloned player\n(for death animations)" + ], + [ + "g_deadChat", + "Allow dead players to chat with living players" + ], + [ + "g_dropForwardSpeed", + "Forward speed of a dropped item" + ], + [ + "g_dropHorzSpeedRand", + "Random component of the initial horizontal speed of a dropped item" + ], + [ + "g_dropUpSpeedBase", + "Base component of the initial vertical speed of a dropped item" + ], + [ + "g_dropUpSpeedRand", + "Random component of the initial vertical speed of a dropped item" + ], + [ + "g_earthquakeEnable", + "Enable camera shake" + ], + [ + "g_fogColorIntensityReadOnly", + "HDR fog color intensity that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_fogColorReadOnly", + "Fog color that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_fogHalfDistReadOnly", + "" + ], + [ + "g_fogMaxOpacityReadOnly", + "Fog max opacity that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_fogStartDistReadOnly", + "" + ], + [ + "g_friendlyfireDist", + "Maximum range for disabling fire at a friendly" + ], + [ + "g_friendlyNameDist", + "Maximum range for seeing a friendly's name" + ], + [ + "g_gametype", + "The current game mode" + ], + [ + "g_giveAll", + "Give all weapons" + ], + [ + "g_hardcore", + "Hardcore?" + ], + [ + "g_heightFogBaseHeightReadOnly", + "height fog is full density at this world height and below" + ], + [ + "g_heightFogEnabledReadOnly", + "use height for normal/sun fog, set in the most recent call to \"setexpfog\"" + ], + [ + "g_heightFogHalfPlaneDistanceReadOnly", + "at this distance above g_heightFogBaseHeight, height fog density is half, set in the most recent call to \"setexpfog\"" + ], + [ + "g_inactivity", + "Time delay before player is kicked for inactivity" + ], + [ + "g_keyboarduseholdtime", + "The time to hold down the 'use' button to activate a 'use' command on a keyboard" + ], + [ + "g_knockback", + "Maximum knockback" + ], + [ + "g_lagged_damage_threshold", + "Threshold (ms) beyond which we will report a damaged lagged client to the tracker leaderboards." + ], + [ + "g_listEntity", + "List the entities" + ], + [ + "g_mantleBlockTimeBuffer", + "Time that the client think is delayed after mantling" + ], + [ + "g_maxDroppedWeapons", + "Maximum number of dropped weapons" + ], + [ + "g_minGrenadeDamageSpeed", + "Minimum speed at which getting hit be a grenade will do damage (not the grenade explosion damage)" + ], + [ + "g_oldschool", + "Oldschool?" + ], + [ + "g_password", + "Password" + ], + [ + "g_playerCollisionEjectSpeed", + "Speed at which to push intersecting players away from each other" + ], + [ + "g_ScoresColor_Allies", + "Allies team color on scoreboard" + ], + [ + "g_ScoresColor_Axis", + "Axis team color on scoreboard" + ], + [ + "g_ScoresColor_EnemyTeam", + "Enemy team color on scoreboard" + ], + [ + "g_ScoresColor_Free", + "Free Team color on scoreboard" + ], + [ + "g_ScoresColor_MyParty", + "Player team color on scoreboard when in the same party" + ], + [ + "g_ScoresColor_MyTeam", + "Player team color on scoreboard" + ], + [ + "g_ScoresColor_Spectator", + "Spectator team color on scoreboard" + ], + [ + "g_scriptMainMenu", + "" + ], + [ + "g_sunFogBeginFadeAngleReadOnly", + "Angle from the sun direction to start fade away from the sun fog color that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogColorIntensityReadOnly", + "HDR sun fog color intensity that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogColorReadOnly", + "Sun fog color that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogDirReadOnly", + "Sun fog direction that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogEnabledReadOnly", + "Sun fog was enabled in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogEndFadeAngleReadOnly", + "Angle from the sun direction to end fade away from the sun fog color that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_sunFogScaleReadOnly", + "Distance scale in the sun fog direction that was set in the most recent call to \"setexpfog\"" + ], + [ + "g_TeamIcon_Allies", + "Icon name for the allied scores banner" + ], + [ + "g_TeamIcon_Axis", + "Icon name for the axis scores banner" + ], + [ + "g_TeamIcon_EnemyAllies", + "Icon name for the allied scores banner" + ], + [ + "g_TeamIcon_EnemyAxis", + "Icon name for the axis scores banner when you're on axis." + ], + [ + "g_TeamIcon_Free", + "Icon name for the scores of players with no team" + ], + [ + "g_TeamIcon_MyAllies", + "Icon name for the allied scores banner" + ], + [ + "g_TeamIcon_MyAxis", + "Icon name for the axis scores banner when you're on axis." + ], + [ + "g_TeamIcon_Spectator", + "Icon name for the scores of players who are spectators" + ], + [ + "g_TeamName_Allies", + "Allied team name" + ], + [ + "g_TeamName_Axis", + "Axis team name" + ], + [ + "g_TeamTitleColor_EnemyTeam", + "Enemy team color for titles" + ], + [ + "g_TeamTitleColor_MyTeam", + "Player team color for titles" + ], + [ + "g_TeamTitleColor_Spectator", + "Spectator team color for titles" + ], + [ + "g_useholdspawndelay", + "Time in milliseconds that the player is unable to 'use' after spawning" + ], + [ + "g_useholdtime", + "Time to hold the 'use' button to activate use on a gamepad" + ], + [ + "g_voiceChatTalkingDuration", + "Time after the last talk packet was received that the player is considered by the\nserver to still be talking in milliseconds" + ], + [ + "g_gravity", + "Game gravity in inches per second squared" + ], + [ + "g_speed", + "changes the speed of the player" + ], + [ + "gamedate", + "The date compiled" + ], + [ + "gamedvr_active", + "Are we allowed to enable GameDVR or not" + ], + [ + "gameMode", + "Current gameMode" + ], + [ + "gamename", + "The name of the game" + ], + [ + "glass_angular_vel", + "Sets the range of angular velocities used by new glass pieces" + ], + [ + "glass_beamDamage", + "The amount of damage beam attacks do to glass" + ], + [ + "glass_break", + "Toggle whether or not glass breaks when shot" + ], + [ + "glass_crack_pattern_scale", + "The scale applied to the radius used for the crack pattern" + ], + [ + "glass_damageToDestroy", + "The amount of damage a piece of glass must take to look damaged" + ], + [ + "glass_damageToWeaken", + "The amount of damage a piece of glass must take to look damaged" + ], + [ + "glass_edge_angle", + "Sets the range of angle deflections used by new glass pieces on a supported edge" + ], + [ + "glass_fall_delay", + "Sets how long a heavy piece supported by a single edge waits before falling, based on glass_fall_ratio" + ], + [ + "glass_fall_gravity", + "Gravity for falling pieces of glass" + ], + [ + "glass_fall_ratio", + "Ratio of piece area to supporting edge length squared. Below the min, the piece never falls." + ], + [ + "glass_fringe_maxcoverage", + "The maximum portion of the original piece of glass that is allowed to remain after the glass shatters" + ], + [ + "glass_fringe_maxsize", + "The maximum area for an edge piece of glass when shattering. Pieces larger than this will be broken into smaller ones" + ], + [ + "glass_fx_chance", + "Chance to play an effect on a small piece of glass when it hits the ground" + ], + [ + "glass_hinge_friction", + "Friction used by moving glass pieces when joined like a hinge to a frame" + ], + [ + "glass_linear_vel", + "Sets the range of linear velocities used by new glass pieces" + ], + [ + "glass_max_pieces_per_frame", + "Maximum number of pieces to create in one frame. This is a guideline and not a hard limit." + ], + [ + "glass_max_shatter_fx_per_frame", + "Maximum number of shatter effects to play in one frame This is a guideline and not a hard limit." + ], + [ + "glass_meleeDamage", + "The amount of damage melee attacks do to glass" + ], + [ + "glass_physics_chance", + "The chance for a given shard of glass to use physics" + ], + [ + "glass_physics_maxdist", + "The maximum distance of a glass piece from the player to do physics" + ], + [ + "glass_radiusDamageMultiplier", + "The amount to scale damage to glass from grenades and other explosions" + ], + [ + "glass_shard_maxsize", + "The maximum area for a flying piece of glass when shattering. Pieces larger than this will be broken into smaller ones" + ], + [ + "glass_shattered_scale", + "The scale of the shattered glass material" + ], + [ + "glass_trace_interval", + "The length of time, in milliseconds, between glass piece traces" + ], + [ + "gpad_button_deadzone", + "Game pad button deadzone threshhold" + ], + [ + "gpad_dpadDebounceTime", + "" + ], + [ + "gpad_menu_scroll_delay_first", + "Menu scroll key-repeat delay, for the first repeat, in milliseconds" + ], + [ + "gpad_menu_scroll_delay_rest_accel", + "Menu scroll key-repeat delay acceleration from start to end, for repeats after the first, in milliseconds per repeat" + ], + [ + "gpad_menu_scroll_delay_rest_end", + "Menu scroll key-repeat delay end, for repeats after the first, in milliseconds" + ], + [ + "gpad_menu_scroll_delay_rest_start", + "Menu scroll key-repeat delay start, for repeats after the first, in milliseconds" + ], + [ + "gpad_stick_deadzone_max", + "Game pad maximum stick deadzone" + ], + [ + "gpad_stick_deadzone_min", + "Game pad minimum stick deadzone" + ], + [ + "gpad_stick_pressed", + "Game pad stick pressed threshhold" + ], + [ + "gpad_stick_pressed_hysteresis", + "Game pad stick pressed no-change-zone around gpad_stick_pressed to prevent bouncing" + ], + [ + "groupDownloadInterval", + "Minimum interval to wait before getting new group counts" + ], + [ + "groupUploadInterval", + "Minimum interval to wait before setting new group counts" + ], + [ + "heli_barrelMaxVelocity", + "" + ], + [ + "heli_barrelRotation", + "How much to rotate the turret barrel when a helicopter fires" + ], + [ + "heli_barrelSlowdown", + "" + ], + [ + "hiDef", + "True if the game video is running in high-def." + ], + [ + "httpnetfs", + "Stream fastfiles from the specified http server" + ], + [ + "hud_bloodOverlayLerpRate", + "Rate at which blood overlay fades out" + ], + [ + "hud_deathQuoteFadeTime", + "The time for the death quote to fade" + ], + [ + "hud_drawHud", + "" + ], + [ + "hud_enable", + "Enable hud elements" + ], + [ + "hud_fade_ammodisplay", + "The time for the ammo display to fade in seconds" + ], + [ + "hud_fade_compass", + "The time for the compass to fade in seconds" + ], + [ + "hud_fade_healthbar", + "The time for the health bar to fade in seconds" + ], + [ + "hud_fade_offhand", + "The time for the offhand weapons to fade in seconds" + ], + [ + "hud_fade_sprint", + "The time for the sprint meter to fade in seconds" + ], + [ + "hud_flash_period_offhand", + "Offhand weapons flash period on changing weapon" + ], + [ + "hud_flash_time_offhand", + "Offhand weapons flash duration on changing weapon" + ], + [ + "hud_health_pulserate_critical", + "The pulse rate of the 'critical' pulse effect" + ], + [ + "hud_health_pulserate_injured", + "The pulse rate of the 'injured' pulse effect" + ], + [ + "hud_health_startpulse_critical", + "The health level at which to start the 'injured' pulse effect" + ], + [ + "hud_health_startpulse_injured", + "The health level at which to start the 'injured' pulse effect" + ], + [ + "hudElemPausedBrightness", + "Brightness of the hudelems when the game is paused." + ], + [ + "hudOutlineDuringADS", + "Turn on the HUD outline (green for friendly, red for enemy) when you are pointing at a player while in ADS." + ], + [ + "igs_config_dw_filename", + "Name of the configuration file on DW Publisher storage." + ], + [ + "igs_sosp", + "Show Original Season Pass" + ], + [ + "igs_td", + "Show Trial DLC" + ], + [ + "igs_version", + "Version id for the In-game store. Set version number to 0, to disable update." + ], + [ + "in_mouse", + "Initialize the mouse drivers" + ], + [ + "inpubliclobby", + "Currently in a public lobby" + ], + [ + "intro", + "Intro movie should play" + ], + [ + "inventory_addEntitlementsToLocalInventory", + "bypass the exchange and directly add entitlements to the local cached player inventory." + ], + [ + "inventory_enabled", + "enable/disable the inventory feature" + ], + [ + "inventory_enableEntitlementDLCScanning", + "enable scanning of entitlement DLC." + ], + [ + "inventory_enableRevoke", + "Enable revoke on purchases you no longer have rights to." + ], + [ + "inventory_exchangeEnabled", + "enable/disable the 1st party exchange feature" + ], + [ + "inventory_exchangeMaxConsumablesPerBoot", + "The maximum number of the same consumable that can be added per boot." + ], + [ + "inventory_exchangeRetryBaseMS", + "The amount to delay with each subsequent retry as base value to be multiplied by an exponential factor 1000 = (1000, 2000, 4000, 8000 etc.)" + ], + [ + "inventory_exchangeRetryByRound", + "enable/disable retry with exponential delay one round of exchanges at a time (1, 2, 3, 1, 2, 3, 1, 2, 3 etc.), vs exponential delay per exchange (1, 1, 1, 2, 2, 2, 3, 3, 3 etc.)" + ], + [ + "inventory_exchangeRetryMax", + "The number of times to retry for each exchange." + ], + [ + "inventory_excludeEntitlementDLCScanning", + "exclude scanning of entitlement DLC (comma separated list of ids to exclude)." + ], + [ + "inventory_ignoreDWPushNotification_claimAchievement", + "ignore incoming push notifications from DW to signal item update" + ], + [ + "inventory_ignoreDWPushNotification_itemUpdate", + "ignore incoming push notifications from DW to signal item update" + ], + [ + "inventory_taskDefaultTimeout", + "default timeout for inventory tasks (in seconds)" + ], + [ + "inventory_taskExchangeTimeout", + "default timeout for inventory exchange tasks (in seconds)" + ], + [ + "inventory_taskGetTimeout", + "default timeout for inventory GET tasks (in seconds)" + ], + [ + "inventory_triggerExchangeOnContentMount", + "trigger an exchange after mounting new content packs" + ], + [ + "inventory_triggerExchangeOnStoreExit", + "trigger an exchange when exiting the store" + ], + [ + "iotd_active", + "Is the IOTD system enabled" + ], + [ + "iotd_retry", + "Can the IOTD system retry fetching data from Demonware" + ], + [ + "jump_slowdownEnable", + "Slow player movement after jumping" + ], + [ + "jump_height", + "The maximum height of a player\\'s jump" + ], + [ + "laserDebug", + "Enables the display of various debug info." + ], + [ + "laserLightRadius", + "undefined" + ], + [ + "laserRadius", + "undefined" + ], + [ + "lb_filter", + "Filter applied to the leaderboard display: ('none','friends','facebook_friends')" + ], + [ + "lb_group", + "GroupID applied to the leaderboard display" + ], + [ + "lb_maxrows", + "Maximum number of rows to fetch" + ], + [ + "lb_minrefresh", + "Minimum time (in seconds) between leaderboard fetches" + ], + [ + "lb_readDelay", + "Delay time between reads(in milliseconds) between leaderboard fetches" + ], + [ + "lb_throttle_time", + "Lobby throttling amount" + ], + [ + "lb_times_in_window", + "Lobby throttling window amount" + ], + [ + "lb_window", + "Lobby throttling window" + ], + [ + "live_qosec_firstupdatems", + "MS to wait before deciding to early out qos" + ], + [ + "live_qosec_lastupdatems", + "MS since last update required to early out qos" + ], + [ + "live_qosec_minpercent", + "Minimum percentage of probe results required before early outing qos" + ], + [ + "live_qosec_minprobes", + "Minimum probe results required before early outing qos" + ], + [ + "liveanticheatunknowndvar", + "Live Anti Cheat Unknown Dvar" + ], + [ + "livestreaming_active", + "Are we allowed to enable LiveStreaming or not" + ], + [ + "loading_sre_fatal", + "Loading errors prevent level from loading." + ], + [ + "lobby_animationSpeed", + "How long each frame of the animation should draw, in milliseconds" + ], + [ + "lobby_animationTilesHigh", + "How many animation tiles high is the searching_for_player texture" + ], + [ + "lobby_animationTilesWide", + "How many animation tiles wide is the searching_for_player texture" + ], + [ + "lobby_numAnimationFrames", + "How many animation tiles are in the searching_for_player texture" + ], + [ + "lobby_searchingPartyColor", + "The color to show that we're searching for that slot when shown in lobbies" + ], + [ + "loc_language", + "Language" + ], + [ + "loc_translate", + "Enable translations" + ], + [ + "log_host_migration_chance", + "The % chance of host migration results telemetry" + ], + [ + "log_party_state", + "Log party state updates to Black Box system" + ], + [ + "lowAmmoWarningColor1", + "Color 1 of 2 to oscilate between" + ], + [ + "lowAmmoWarningColor2", + "Color 2 of 2 to oscilate between" + ], + [ + "lowAmmoWarningNoAmmoColor1", + "Like lowAmmoWarningColor1, but when no ammo." + ], + [ + "lowAmmoWarningNoAmmoColor2", + "lowAmmoWarningColor2, but when no ammo." + ], + [ + "lowAmmoWarningNoReloadColor1", + "Like lowAmmoWarningColor1, but when no ammo." + ], + [ + "lowAmmoWarningNoReloadColor2", + "lowAmmoWarningColor2, but when no ammo." + ], + [ + "lowAmmoWarningPulseFreq", + "Frequency of the pulse (oscilation between the 2 colors)" + ], + [ + "lowAmmoWarningPulseMax", + "Min of oscilation range: 0 is color1 and 1.0 is color2. Can be < 0, and the wave will clip at 0." + ], + [ + "lowAmmoWarningPulseMin", + "Max of oscilation range: 0 is color1 and 1.0 is color2. Can be > 1.0, and the wave will clip at 1.0." + ], + [ + "lsp_enumertion_max_retry_time", + "Max time that the LSP enumeration can retry" + ], + [ + "lsp_enumertion_retry_step", + "Step in m/s for the LSP enumeration retry" + ], + [ + "lui_demoMode", + "Check if the game is in demo mode." + ], + [ + "lui_FFotDSupportEnabled", + "Enables lui to update itself via the ffotd" + ], + [ + "lui_hud_motion_angle_ease_speed", + "Hud motion ease percentage of degrees per second" + ], + [ + "lui_hud_motion_bob_scale", + "Hud motion bob scale" + ], + [ + "lui_hud_motion_enabled", + "Enable hud motion" + ], + [ + "lui_hud_motion_perspective", + "value for hud motion perspective transform in pixels" + ], + [ + "lui_hud_motion_rotation_max", + "Hud motion rotation max" + ], + [ + "lui_hud_motion_rotation_scale", + "Hud motion rotation scale" + ], + [ + "lui_hud_motion_trans_ease_speed", + "Hud motion ease percentage of pixels per second" + ], + [ + "lui_hud_motion_translation_max", + "Hud motion translation max" + ], + [ + "lui_hud_motion_translation_scale", + "Hud motion translation scale" + ], + [ + "lui_loot_duplicateredemption", + "Whether a user can redeem duplicate loot items in the Armory" + ], + [ + "LUI_MemErrorsFatal", + "Out of memory errors cause drops when true, reinits the UI system if false" + ], + [ + "lui_menuFlowEnabled", + "Enables LUI menu flow" + ], + [ + "lui_mlg_rules_unlocked", + "Whether MLG rules are unlocked" + ], + [ + "lui_priv_lobby_team", + "Team selected in private match lobby" + ], + [ + "lui_splitscreensignin_menu", + "Enables the LUI splitscreensignin menu" + ], + [ + "lui_splitscreenupscaling", + "Force splitscreen upscaling off/on (-1 off, 1 on) -- requires map change" + ], + [ + "lui_systemlink_menu", + "Enables the LUI systemlink menu" + ], + [ + "lui_waitingforgavelmessagesconfirmed", + "" + ], + [ + "lui_waitingfornetworktype", + "value is LuiWaitingForNetworkType enum" + ], + [ + "lui_waitingforonlinedatafetch_controller", + "the controller index that is fetching the online stats data" + ], + [ + "LUI_WorkerCmdGC", + "Dev-only flag to enable/disable LUI workerCmd GC thread" + ], + [ + "lui_xboxlive_menu", + "Enables the LUI xboxlive menu" + ], + [ + "m_filter", + "Allow mouse movement smoothing" + ], + [ + "m_forward", + "Forward speed in units per second" + ], + [ + "m_pitch", + "Default pitch" + ], + [ + "m_side", + "Sideways motion in units per second" + ], + [ + "m_yaw", + "Default yaw" + ], + [ + "manifestfs", + "Use a manifest file to read segmented fastfiles" + ], + [ + "mapname", + "The current map name" + ], + [ + "mapPackMPGroupFourFlags", + "Map pack flags that comprise MP ala carte map pack 1" + ], + [ + "mapPackMPGroupFreeFlags", + "Map pack flags that comprise the free MP ala carte map pack" + ], + [ + "mapPackMPGroupOneFlags", + "Map pack flags that comprise MP ala carte map pack 1" + ], + [ + "mapPackMPGroupThreeFlags", + "Map pack flags that comprise MP ala carte map pack 1" + ], + [ + "mapPackMPGroupTwoFlags", + "Map pack flags that comprise MP ala carte map pack 1" + ], + [ + "marketing_active", + "Enable/Disable Marketing Comms" + ], + [ + "marketing_refresh_time", + "Time in seconds to wait before refreshing marketing messages from demonware" + ], + [ + "marketing_motd_once_per_day", + "Display motd once per day" + ], + [ + "matchdata_active", + "Are match data uploads enabled" + ], + [ + "matchdata_maxcompressionbuffer", + "Max SP match data compression buffer to use (in bytes)" + ], + [ + "matchmaking_debug", + "Enable matchmaking debugging information" + ], + [ + "max_ping_threshold_good", + "max ping value to be considered as good" + ], + [ + "max_ping_threshold_medium", + "max ping value to be considered as medium" + ], + [ + "max_xp_per_match", + "" + ], + [ + "maxPrestigeOverride", + "Overrides the maximum prestige level, disabled if 0." + ], + [ + "maxVoicePacketsPerSec", + "" + ], + [ + "maxVoicePacketsPerSecForServer", + "" + ], + [ + "mdsd", + "enable match data stat delta logging?" + ], + [ + "melee_debug", + "Turn on debug lines for melee traces" + ], + [ + "mis_cheat", + "Set when level unlock cheat is performed" + ], + [ + "migration_dvarErrors", + "Whether to check for illegal script dvar changes." + ], + [ + "min_wait_for_players", + "" + ], + [ + "missileRemoteFOV", + "Remote missile-cam, FOV to use." + ], + [ + "missileRemoteSteerPitchRange", + "Remote-controlled missile allowed up/down range. To keep players from steering missiles above the horizon." + ], + [ + "missileRemoteSteerPitchRate", + "Remote-controlled missile up/down steering speed." + ], + [ + "missileRemoteSteerYawRate", + "Remote-controlled missile left/right steering speed." + ], + [ + "mm_aw_onboarding_rank", + "If a player is at or above this rank in AW, she is not considered onboarding" + ], + [ + "mm_blops2_onboarding_skill", + "Used to determine onboarding status for Ghosts" + ], + [ + "mm_bucket_option", + "if using bucketing, describes what pools can join with each other" + ], + [ + "mm_country_code", + "country code" + ], + [ + "mm_ghosts_onboarding_skill", + "Used to determine onboarding status for Ghosts" + ], + [ + "mm_past_title_stats_source", + "what type of information do we use from the past titles (rank vs kdr, etc)" + ], + [ + "mm_skill_calculation_type", + "" + ], + [ + "mm_skill_enforcement", + "" + ], + [ + "mm_skill_lower_bucket", + "lower mm skill bucket" + ], + [ + "mm_skill_param_delta", + "Delta parameter for Johnson SU distribution curve" + ], + [ + "mm_skill_param_gamma", + "Gamma parameter for Johnson SU distribution curve" + ], + [ + "mm_skill_param_lambda", + "Lambda parameter for Johnson SU distribution curve" + ], + [ + "mm_skill_param_xi", + "Xi parameter for Johnson SU distribution curve" + ], + [ + "mm_skill_strict_enforcement", + "" + ], + [ + "mm_skill_type", + "mm skill type" + ], + [ + "mm_skill_upper_bucket", + "upper mm skill bucket" + ], + [ + "mm_sph_1", + "" + ], + [ + "mm_sph_10", + "" + ], + [ + "mm_sph_11", + "" + ], + [ + "mm_sph_12", + "" + ], + [ + "mm_sph_13", + "" + ], + [ + "mm_sph_14", + "" + ], + [ + "mm_sph_15", + "" + ], + [ + "mm_sph_16", + "" + ], + [ + "mm_sph_17", + "" + ], + [ + "mm_sph_18", + "" + ], + [ + "mm_sph_2", + "" + ], + [ + "mm_sph_3", + "" + ], + [ + "mm_sph_4", + "" + ], + [ + "mm_sph_5", + "" + ], + [ + "mm_sph_6", + "" + ], + [ + "mm_sph_7", + "" + ], + [ + "mm_sph_8", + "" + ], + [ + "mm_sph_9", + "" + ], + [ + "mm_split_population", + "" + ], + [ + "mm_test_type", + "mm test type" + ], + [ + "mm_use_onboarding_skill", + "If set, we will for the player's skill to be the lowest available" + ], + [ + "monkeytoy", + "Restrict console access" + ], + [ + "motd", + "" + ], + [ + "motd_store_link", + "Add a link to the in-game store in the MOTD popup" + ], + [ + "motionTrackerBlurDuration", + "The motion blur duration for motion tracker dots" + ], + [ + "motionTrackerCenterX", + "" + ], + [ + "motionTrackerCenterY", + "" + ], + [ + "motionTrackerPingFadeTime", + "How long an enemy is visible on the motion tracker after being detected" + ], + [ + "motionTrackerPingPitchAddPerEnemy", + "The added percentage of pitch for each additional enemy that is detected (final pitch = base pitch * (1 + enemy count * this))" + ], + [ + "motionTrackerPingPitchBase", + "The pitch of the motion tracker sound for a nearby enemy" + ], + [ + "motionTrackerPingPitchNearby", + "The pitch of the motion tracker sound for a nearby enemy" + ], + [ + "motionTrackerPingSize", + "The width and height of the motion tracker's enemy indicators as a percentage of motion tracker scale" + ], + [ + "msg_field_delta2", + "enable the delta2 serialization." + ], + [ + "name", + "Player name" + ], + [ + "net_authPort", + "UDP port for Steam authentication" + ], + [ + "net_ip", + "Network IP Address" + ], + [ + "net_masterServerPort", + "UDP port for Steam server browser" + ], + [ + "net_noudp", + "Disable UDP" + ], + [ + "net_port", + "Network port" + ], + [ + "net_socksEnabled", + "Enable network sockets" + ], + [ + "net_socksPassword", + "Network socket password" + ], + [ + "net_socksPort", + "Network socket port" + ], + [ + "net_socksServer", + "Network socket server" + ], + [ + "net_socksUsername", + "Network socket username" + ], + [ + "nextmap", + "Next map to play" + ], + [ + "nightVisionDisableEffects", + "" + ], + [ + "nightVisionFadeInOutTime", + "How long the fade to/from black lasts when putting on or removing night vision goggles." + ], + [ + "nightVisionPowerOnTime", + "How long the black-to-nightvision fade lasts when turning on the goggles." + ], + [ + "num_available_map_packs", + "Number of map packs available for this platform" + ], + [ + "objectiveFontSize", + "Onscreen Objective Pointer - Fontsize of the icon's text." + ], + [ + "objectiveTextOffsetY", + "Onscreen Objective Pointer - Offset of the icon's text." + ], + [ + "onlinegame", + "Current game is an online game with stats, custom classes, unlocks" + ], + [ + "overrideNVGModelWithKnife", + "When true, nightvision animations will attach the weapDef's knife model instead of the night vision goggles." + ], + [ + "overtimeTimeLimit", + "" + ], + [ + "p2pAuth_allow_steam_p2p", + "Determines if Steam based P2P is allowed if direct connectivity is not possible." + ], + [ + "p2pAuth_disable", + "" + ], + [ + "paintExplosionRadius", + "The radius of the paint grenade explosion" + ], + [ + "painVisionLerpOutRate", + "Rate at which pain vision effect lerps out" + ], + [ + "painVisionTriggerHealth", + "Health (0 to 1) that will trigger the pain vision effect" + ], + [ + "party_alternateMapVoteStatus", + "Alternate map vote progress" + ], + [ + "party_dlc_only_intended_mappacks", + "When selecting next map for rotation, should any maps not in the intended search be excluded, even if available?" + ], + [ + "party_firstSubpartyIndex", + "Determines sort order and coloring of parties in lobbies. Randomly set by code. Dvar provided for debugging." + ], + [ + "party_followPartyHostOutOfGames", + "Whether we should follow our party host out of a game in progress." + ], + [ + "party_gamesize", + "Current maximum game size" + ], + [ + "party_gametype", + "Current gametype" + ], + [ + "party_inactiveHeartbeatPeriod", + "How often to send inactive party keep alive packets in milliseconds." + ], + [ + "party_initial_dlc_search_timer", + "Time until DLC enabled search should show an error dialog suggesting the user consider going to non dlc search" + ], + [ + "party_IsLocalClientSelected", + "True if selected player is a local client. Only valid when used with party feeders." + ], + [ + "party_kickplayerquestion", + "String to store the question about kicking the selected player" + ], + [ + "party_listFocus", + "True when an item in the party list has focus." + ], + [ + "party_lobbyPlayerCount", + "Number of players currently in the party/lobby" + ], + [ + "party_mapname", + "Current map name" + ], + [ + "party_mapvote_entrya_mapname", + "Primary map vote entry name" + ], + [ + "party_mapvote_entryb_mapname", + "Alternate map vote entry name" + ], + [ + "party_matchedPlayerCount", + "Number of matched players before revealing their true names" + ], + [ + "party_maxplayers", + "Maximum number of players in a party" + ], + [ + "party_maxPrivatePartyPlayers", + "Max number of players allowed in a private party." + ], + [ + "party_maxTeamDiff", + "Maximum difference allowed between teams before starting a match" + ], + [ + "party_membersMissingMapPack", + "Whether any party member is missing one of the enabled map packs. Only valid after running partyUpdateMissingMapPackDvar" + ], + [ + "party_minLobbyTime", + "Minimum time (in seconds) for a lobby to be open before auto starting a match" + ], + [ + "party_minplayers", + "Minimum number of players in a party" + ], + [ + "party_nextMapVoteStatus", + "Next map vote progress" + ], + [ + "party_partyPlayerCount", + "Number of players currently in the party/lobby" + ], + [ + "party_partyPlayerCountNum", + "Number of players currently in the party/lobby" + ], + [ + "party_playersCoop", + "True if the current playlist places all players on the allies team" + ], + [ + "party_playervisible", + "Whether selected player in party is showing true info or not. Only valid when used with party feeders." + ], + [ + "party_randomMapVoteStatus", + "Random map vote progress" + ], + [ + "party_resume_dlc_search_timer", + "Time until DLC enabled search should show an error dialog suggesting the user consider going to non dlc search" + ], + [ + "party_search_for_dlc_content", + "search for DLC enabled games else standard maps only will be used" + ], + [ + "party_selectedIndex", + "Current selected player index in the feeder." + ], + [ + "party_selectedIndexChangedTime", + "Time stamp in milliseconds when the selected index last changed." + ], + [ + "party_statusString", + "Party Status (localized )" + ], + [ + "party_teambased", + "True if the current playlist is team based" + ], + [ + "party_teamsVisible", + "teams are visible in UI" + ], + [ + "party_timer", + "Time until game begins in seconds, for UI display" + ], + [ + "partyChatDisallowed", + "Whether to disallow ps4 Live Party Chat" + ], + [ + "partymigrate_broadcast_interval", + "Time between telling people who the new host is when migrating lobby" + ], + [ + "partymigrate_cpuBonusPing", + "The ping rewarded to a CPU meeting the bonus threshold when scoring hosts." + ], + [ + "partymigrate_cpuBonusThreshold", + "The required excess %CPU speed to get a bonus when scoring hosts." + ], + [ + "partymigrate_logResults", + "Instrumentation - Whether to log the best host calculation results. 0 is disabled, 1 for games, 2 for parties, 3 for both." + ], + [ + "partymigrate_makeHostTimeout", + "Time before giving up on makeHost requests" + ], + [ + "partymigrate_pingtest_active", + "Whether to do a ping test when lobby migrating" + ], + [ + "partymigrate_pingtest_filterThreshold", + "Acceptable ping difference from best ping host for host selection (ms)" + ], + [ + "partymigrate_pingtest_minThreshold", + "Minimum meaningful ping delta for host selection (ms)" + ], + [ + "partymigrate_pingtest_retry", + "Time between ping tests when migrating lobby" + ], + [ + "partymigrate_pingtest_timeout", + "Time to give up on ping tests when migrating lobby" + ], + [ + "partymigrate_preferSameHost", + "When possible, prefer keeping the same host on migrations" + ], + [ + "partymigrate_selectiontime", + "Time before requiring a host selection when migrating lobby" + ], + [ + "partymigrate_timeout", + "Time before giving up on lobby migration if we hear nothing" + ], + [ + "partymigrate_timeoutmax", + "Time before giving up on lobby migration if we hear nothing" + ], + [ + "partymigrate_uploadtest_minThreshold", + "Minimum meaningful upload bandwidth delta for host selection (bps)" + ], + [ + "password", + "" + ], + [ + "perk_armorPiercingDamage", + "" + ], + [ + "perk_blastShieldScale", + "" + ], + [ + "perk_blastShieldScale_HC", + "" + ], + [ + "perk_bulletPenetrationMultiplier", + "Multiplier for extra bullet penetration" + ], + [ + "perk_extendedMagsMGAmmo", + "Number of extra bullets per clip for machine gun weapons with the extended mags perk." + ], + [ + "perk_extendedMagsPistolAmmo", + "Number of extra bullets per clip for pistol weapons with the extended mags perk." + ], + [ + "perk_extendedMagsRifleAmmo", + "Number of extra bullets per clip for rifle weapons with the extended mags perk." + ], + [ + "perk_extendedMagsSMGAmmo", + "Number of extra bullets per clip for sub machine gun weapons with the extended mags perk." + ], + [ + "perk_extendedMagsSpreadAmmo", + "Number of extra bullets per clip for spread weapons with the extended mags perk." + ], + [ + "perk_extraBreath", + "Number of extra seconds a player can hold his breath" + ], + [ + "perk_fastClimb", + "Scales the ladder climb time" + ], + [ + "perk_fastRegenRate", + "" + ], + [ + "perk_fastRegenWaitMS", + "" + ], + [ + "perk_fastSnipeScale", + "Scales the recovery speed of the view kick when using a sniper." + ], + [ + "perk_footstepVolumeAlly", + "" + ], + [ + "perk_footstepVolumeEnemy", + "" + ], + [ + "perk_footstepVolumePlayer", + "" + ], + [ + "perk_footstepVolumeSelectiveHearingMin", + "" + ], + [ + "perk_improvedExtraBreath", + "Number of extra seconds a player can hold his breath" + ], + [ + "perk_lightWeightViewBobScale", + "Scale for first person view movement while lightweight." + ], + [ + "perk_numExtraLethal", + "Number of extra lethal grenades" + ], + [ + "perk_numExtraTactical", + "Number of extra tactical grenades" + ], + [ + "perk_parabolicAngle", + "Eavesdrop perk's effective FOV angle" + ], + [ + "perk_parabolicIcon", + "Eavesdrop icon to use when displaying eavesdropped voice chats" + ], + [ + "perk_parabolicRadius", + "Eavesdrop perk's effective radius" + ], + [ + "perk_quickDrawSpeedScale", + "Scales the 'Hip to ADS' transition speed." + ], + [ + "perk_quickDrawSpeedScaleSniper", + "Scales the 'Hip to ADS' transition speed." + ], + [ + "perk_scavengerMode", + "" + ], + [ + "perk_sprintMultiplier", + "Multiplier for player_sprinttime" + ], + [ + "perk_sprintRecoveryMultiplierActual", + "" + ], + [ + "perk_sprintRecoveryMultiplierVisual", + "" + ], + [ + "perk_weapRateMultiplier", + "Percentage of weapon firing rate to use" + ], + [ + "perk_weapReloadMultiplier", + "Percentage of weapon reload time to use" + ], + [ + "perk_weapSpreadMultiplier", + "Percentage of weapon spread to use" + ], + [ + "phys_autoDisableLinear", + "A body must have linear velocity less than this to be considered idle." + ], + [ + "phys_autoDisableTime", + "The amount of time a body must be idle for it to go to sleep." + ], + [ + "phys_bulletSpinScale", + "Scale of the effective offset from the center of mass for the bullet impacts." + ], + [ + "phys_bulletUpBias", + "Up Bias for the direction of the bullet impact." + ], + [ + "phys_dragAngular", + "The amount of angular drag, applied globally" + ], + [ + "phys_dragLinear", + "The amount of linear drag, applied globally" + ], + [ + "phys_gravity", + "Physics gravity in units/sec^2." + ], + [ + "phys_gravity_ragdoll", + "Physics gravity used by ragdolls in units/sec^2." + ], + [ + "phys_gravityChangeWakeupRadius", + "The radius around the player within which objects get awakened when gravity changes" + ], + [ + "phys_jitterMaxMass", + "Maximum mass to jitter - jitter will fall off up to this mass" + ], + [ + "physVeh_explodeForce", + "The force applied to physics vehicles due to explosions" + ], + [ + "physVeh_explodeSpinScale", + "The max (random) offset from the center of mass at which splash damage applies its force" + ], + [ + "physVeh_jump", + "Set to 1 to make a vehicle jump." + ], + [ + "physVeh_minContactImpulse", + "The minimum impulse needed to register a contact notification" + ], + [ + "physVeh_minImpactMomentum", + "The minimum collision momentum needed to register an impact" + ], + [ + "physVeh_StepsPerFrame", + "The number of physics timesteps that the server frame will be divided into." + ], + [ + "pickupPrints", + "Print a message to the game window when picking up ammo, etc." + ], + [ + "player_breath_snd_delay", + "The delay before playing the breathe in sound" + ], + [ + "player_breath_snd_lerp", + "The interpolation rate for the player hold breath sound" + ], + [ + "player_current_floor", + "" + ], + [ + "player_MGUseRadius", + "The radius within which a player can mount a machine gun" + ], + [ + "player_stunWhiteFlash", + "If enabled, player's screens will flash white when they are stunned." + ], + [ + "player_throwbackInnerRadius", + "The radius to a live grenade player must be within initially to do a throwback" + ], + [ + "player_throwbackOuterRadius", + "The radius player is allow to throwback a grenade once the player has been in the inner radius" + ], + [ + "player_useRadius", + "The radius within which a player can use things" + ], + [ + "playercard_cache_download_max_retry_time", + "Max time that the player cache download can retry" + ], + [ + "playercard_cache_download_retry_step", + "Step in m/s for the player cache download retry" + ], + [ + "playercard_cache_upload_max_retry_time", + "Max time that the player cache upload can retry" + ], + [ + "playercard_cache_upload_retry_step", + "Step in m/s for the player cache upload retry" + ], + [ + "playercard_cache_validity_life", + "The life of a cached gamercard (it can be re-downloaded after this)" + ], + [ + "playerPositionRecordSampleTime", + "How often to sample player positions and save them into match data." + ], + [ + "playlist", + "The playlist number" + ], + [ + "playlistAggrFilename", + "Aggregated playlist filename" + ], + [ + "playlistFilename", + "Playlist filename" + ], + [ + "playListUpdateCheckMinutes", + "Minutes to wait between checking for updated playlist." + ], + [ + "prestige_shop_active", + "Are we allowed to show the Prestige Shop or not" + ], + [ + "prestige30EasterEggEnabled", + "Enables the easter egg for prestige 30 if 1, disabled if 0." + ], + [ + "privateMatch_joinPassword", + "" + ], + [ + "privateMatch_serverPassword", + "" + ], + [ + "profileMenuOption_blacklevel", + "" + ], + [ + "profileMenuOption_offensiveContentMode", + "Mode of the offensive content warning at startup - 0, skip and turn on; 1, skip and turn off; 2, ask user" + ], + [ + "profileMenuOption_safeAreaHorz", + "" + ], + [ + "profileMenuOption_safeAreaVert", + "" + ], + [ + "profileMenuOption_volume", + "" + ], + [ + "protocol", + "Protocol version" + ], + [ + "pt_AliensReadyUpPrivateInUse", + "Do we use the co-op Ready Up feature in public lobbies?" + ], + [ + "pt_AliensReadyUpPublicInUse", + "Do we use the co-op Ready Up feature in public lobbies?" + ], + [ + "pt_AliensReadyUpPublicStartTimerLength", + "co-op Ready Up start timer length in seconds" + ], + [ + "pt_allMembersDoQoS", + "Whether to send search results to all party/lobby members to get QoS data" + ], + [ + "pt_backoutOnClientPresence", + "Whether the host should backout the party on client presence. 0=fully disabled, 1=out of game only, 2=in-game also" + ], + [ + "pt_connectAttempts", + "Connect timeout when joining another game/party, per attempt" + ], + [ + "pt_connectTimeout", + "Connect timeout when joining another game/party, per attempt" + ], + [ + "pt_gameStartTimerLength", + "Time in seconds before a game start once enough party members are ready" + ], + [ + "pt_logHostSelectionChance", + "Sets the probability that we log our host selection results" + ], + [ + "pt_memberTimeout", + "Client timeout time in the general case" + ], + [ + "pt_migrateBeforeAdvertise", + "Whether lobbies made by private parties should migrate host before publishing" + ], + [ + "pt_migrationBandwidthBonusPing", + "The ping rewarded to the bonus bandwidth threshold when scoring hosts." + ], + [ + "pt_migrationBandwidthBonusThreshold", + "The required excess % upload bandwidth to get a bonus when scoring hosts." + ], + [ + "pt_migrationCPUWeight", + "How important CPU speed is when selecting a new host" + ], + [ + "pt_migrationNotInstalledWeight", + "How important not being done installing is when selecting a new host" + ], + [ + "pt_migrationPingBad", + "" + ], + [ + "pt_migrationPingWeight", + "" + ], + [ + "pt_migrationQuitsBad", + "" + ], + [ + "pt_migrationQuitsWeight", + "" + ], + [ + "pt_migrationRAMWeight", + "How important it is to have the minimum amount of RAM when selecting a new host" + ], + [ + "pt_migrationThreshold", + "Minimum amount which another client must be better than the current host to trigger a migration" + ], + [ + "pt_migrationUploadBad", + "" + ], + [ + "pt_migrationUploadWeight", + "" + ], + [ + "pt_migrationWifiPenalty", + "How important Wifi is when selecting a new host" + ], + [ + "pt_pregameStartTimerLength", + "Time in seconds before showing and starting the game start timer" + ], + [ + "pt_rejoin", + "Enable defensive rejoin command" + ], + [ + "pt_reservedAnonymousSlotTime", + "Time in milliseconds that ANONYMOUS slots will be reserved." + ], + [ + "pt_reservedCommittedSlotTime", + "Time in milliseconds that COMMITTED slots will be reserved" + ], + [ + "pt_reservedJoiningSlotTime", + "Time in milliseconds that JOINING slots will be reserved" + ], + [ + "pt_searchConnectAttempts", + "Connect timeout when joining another game/party, per attempt" + ], + [ + "pt_searchPauseTime", + "Minimum amount of time to pause between searches" + ], + [ + "pt_searchRandomDelay", + "Time period over which the search timers will get randomly delayed." + ], + [ + "pt_searchResultsLifetime", + "Time at which we consider the search results stale" + ], + [ + "pt_searchResultsMin", + "Minimum amount of time that has to pass before we'll search again for matches" + ], + [ + "pt_stillConnectingWaitTime", + "Amount of time to wait for someone to finish connecting before searching for lobbies to merge with" + ], + [ + "pt_useMigrationWeights", + "Killswitch to turn on or off the host selection by weights" + ], + [ + "publisherFileFetchTimeout", + "default timeout for publisher files FETCH tasks (in seconds)" + ], + [ + "r_adaptiveSubdiv", + "Enables screen space Catmull Clark adaptive tessellation. If disabled, models tessellate to their designed subdivision level." + ], + [ + "r_adaptiveSubdivBaseFactor", + "Screen space Catmull Clark adaptive tessellation factor for the base model. Smaller values mean more tessellation." + ], + [ + "r_adaptiveSubdivPatchFactor", + "Screen space Catmull Clark adaptive tessellation factor for the base model. Smaller values mean more tessellation." + ], + [ + "r_allCells", + "Draw all cells. Most useful for seeing if portals or cells are hiding things they should not.." + ], + [ + "r_amdGPU", + "At least on AMD GPU used for rendering." + ], + [ + "r_aoBlurSharpness", + "Controls the tolerance for depth discontinuities during the bilateral blur step. Larger values reduce the depth tolerance and effectively sharpen more edges." + ], + [ + "r_aoBlurStep", + "Step scale applied to sample offsets during the bilateral blur. A value of 1 results in a normal gaussian blur. Increasing it to 2 or 3 makes the filter larger but causes fine dithering patterns." + ], + [ + "r_aoDiminish", + "Decrease the effect of occlusion on brighter colors" + ], + [ + "r_aoPower", + "Power curve applied to AO factor" + ], + [ + "r_aoStrength", + "Strength of Ambient Occlusion effect" + ], + [ + "r_aoUseTweaks", + "Use r_ao* dvars instead of the current light set values for AO common params" + ], + [ + "r_artUseTweaks", + "Tells the game that art tweaks is enabled and script is in control (as opposed to ColorEd)." + ], + [ + "r_aspectRatio", + "Screen aspect ratio. Most widescreen monitors are 16:10 instead of 16:9." + ], + [ + "r_asyncCompute", + "Enables scheduling GPU compute shader work prior to the graphics frame, improving overlap." + ], + [ + "r_atlasAnimFPS", + "Speed to animate atlased 2d materials" + ], + [ + "r_autopriority", + "Automatically set the priority of the windows process when the game is minimized" + ], + [ + "r_balanceLightmapOpaqueLists", + "Split lightmap opaque into multiple draw lists." + ], + [ + "r_blacklevel", + "Black level (negative brightens output)" + ], + [ + "r_blur", + "Dev tweak to blur the screen" + ], + [ + "r_blurdstGaussianBlurLevel", + "MIP level to start gaussian blur at" + ], + [ + "r_blurdstGaussianBlurRadius", + "Amount to gaussian blur blur distortion render target" + ], + [ + "r_brightness", + "Brightness adjustment" + ], + [ + "r_cacheModelLighting", + "Speed up model lighting by caching previous results" + ], + [ + "r_cacheSModelLighting", + "Speed up static model lighting by caching previous results" + ], + [ + "r_charLightAmbient", + "" + ], + [ + "r_clampLodScale", + "Clamps the amount that the engine can adjust the LOD distance. 0 the engine can fully adjust. 1 the engine cannot adjust it at all. 0.5 the maximum the engine can adjust the LOD distance is 50% or the default." + ], + [ + "r_clear", + "Controls how the color buffer is cleared" + ], + [ + "r_clearColor", + "Color to clear the screen to when clearing the frame buffer" + ], + [ + "r_clearColor2", + "Color to clear every second frame to (for use during development)" + ], + [ + "r_clutCompositeVisionSet", + "Composite clut with vision set." + ], + [ + "r_cmdbuf_worker", + "Process command buffer in a separate thread" + ], + [ + "r_colorGradingEnable", + "Enable color grading." + ], + [ + "r_colorMap", + "Replace all color maps with pure black or pure white" + ], + [ + "r_colorScaleUseTweaks", + "Override color scale LightSet settings with tweak dvar values. (MP)" + ], + [ + "r_combinePostOpaqueFx", + "" + ], + [ + "r_contrast", + "Contrast adjustment" + ], + [ + "r_darkBlur", + "Apply blur (decrease of visual acuity) when dark" + ], + [ + "r_darkBlurPower", + "Power curve of blurring (decrease of visual acuity) when dark" + ], + [ + "r_darkBlurRadius", + "Radius of blurring (decrease of visual acuity) when dark" + ], + [ + "r_darkColor", + "Reduce color sensitivity when dark" + ], + [ + "r_darkColorPower", + "Power curve of color sensitivity when dark" + ], + [ + "r_debugLineWidth", + "Width of server side debug lines" + ], + [ + "r_defaultPatchCount", + "Patches per thread group for all other surfaces." + ], + [ + "r_depthPrepass", + "Enable depth prepass for various geometries" + ], + [ + "r_depthSortEnable", + "Enable sorting of transparent surfaces." + ], + [ + "r_depthSortRange", + "Range to consider depth sort," + ], + [ + "r_desaturation", + "Desaturation adjustment" + ], + [ + "r_detailMap", + "Replace all detail maps with an image that effectively disables them" + ], + [ + "r_diffuseColorScale", + "Globally scale the diffuse color of all point lights" + ], + [ + "r_displacementMap", + "Replace all displacement maps with an image that effectively disables them" + ], + [ + "r_displacementPatchCount", + "Patches per thread group for displacement surfaces." + ], + [ + "r_distortion", + "Enable distortion" + ], + [ + "r_distortion_script_force_off", + "Force distortion off in script" + ], + [ + "r_dlightLimit", + "Maximum number of dynamic lights drawn simultaneously" + ], + [ + "r_dof_bias", + "Depth of field bias as a power function (like gamma); less than 1 is sharper" + ], + [ + "r_dof_enable", + "Enable the depth of field effect" + ], + [ + "r_dof_farBlur", + "" + ], + [ + "r_dof_farEnd", + "Depth of field far end distance, in inches" + ], + [ + "r_dof_farStart", + "Depth of field far start distance, in inches" + ], + [ + "r_dof_nearBlur", + "" + ], + [ + "r_dof_nearEnd", + "Depth of field near end distance, in inches" + ], + [ + "r_dof_nearStart", + "Depth of field near start distance, in inches" + ], + [ + "r_dof_physical_adsFocusSpeed", + "ADS focus speed (focus dist. far to near, focus dist. near to far, aperture opening, aperture closing)" + ], + [ + "r_dof_physical_adsMaxFstop", + "ADS maximum f-stop (optimal aperture and focus distance are automatically calculated for this mode)" + ], + [ + "r_dof_physical_adsMinFstop", + "ADS minimum f-stop (optimal aperture and focus distance are automatically calculated for this mode)" + ], + [ + "r_dof_physical_bokehEnable", + "Enable the bokeh depth of field effect" + ], + [ + "r_dof_physical_bokehPreset", + "Changes dof sampling quality" + ], + [ + "r_dof_physical_bokehRotation", + "Bokeh shape rotation in degrees (hexagonal and octogonal only)" + ], + [ + "r_dof_physical_bokehShape", + "Changes the bokeh shape" + ], + [ + "r_dof_physical_bokehSharpness", + "Bokeh shape sharpness, trades sharpness for stability (circular only)" + ], + [ + "r_dof_physical_enable", + "enable physical camera controls (using aperture priority)" + ], + [ + "r_dof_physical_filmDiagonal", + "Diagonal size of the film/sensor (mm). The bigger the sensor size, the bigger the circle of confusion (which means stronger blurring at all distances). Defaults to full-frame 35mm" + ], + [ + "r_dof_physical_focusDistance", + "Distance to the plane in focus for the scene" + ], + [ + "r_dof_physical_fstop", + "Aperture of the camera for the scene. Lower f-stop yields a shallower depth of field. Typical values range from f/1 to f/22. Rare extremes are f/0.75 and f/32" + ], + [ + "r_dof_physical_hipEnable", + "Enable hyperfocal mode" + ], + [ + "r_dof_physical_hipFocusSpeed", + "Hyperfocal mode focus speed (focus dist. far to near, focus dist. near to far, aperture opening, aperture closing)" + ], + [ + "r_dof_physical_hipFstop", + "Aperture of the camera for the scene in the hyperfocal mode" + ], + [ + "r_dof_physical_hipSharpCocDiameter", + "Defines what circle of confusion can be considered sharp (mm). Defaults to 0.03mm, generally accepted value for 35mm" + ], + [ + "r_dof_physical_maxCocDiameter", + "Maximum circle of confusion diameter (virtual units, might be clamped for bokeh dof)" + ], + [ + "r_dof_physical_minFocusDistance", + "Minimum focus distance (inches)" + ], + [ + "r_dof_physical_viewModelFocusDistance", + "Distance to the plane in focus for the scene" + ], + [ + "r_dof_physical_viewModelFstop", + "Aperture of the camera for the view model. Lower f-stop yields a shallower depth of field. Typical values range from f/1 to f/22. Rare extremes are f/0.75 and f/32" + ], + [ + "r_dof_tweak", + "Use dvars to set the depth of field effect; overrides r_dof_enable" + ], + [ + "r_dof_viewModelEnd", + "Depth of field viewmodel end distance, in inches" + ], + [ + "r_dof_viewModelStart", + "Depth of field viewmodel start distance, in inches" + ], + [ + "r_drawSun", + "Enable sun effects" + ], + [ + "r_drawWater", + "Enable water animation" + ], + [ + "r_dynamicOPL", + "Enable drawing vfx lights as overlapping primary light for saving gpu performance." + ], + [ + "r_dynamicSpotLightShadows", + "Enable shadows for dynamic/VFX spot lights, you should set this dvar then spawn the new light." + ], + [ + "r_elevatedPriority", + "Utilize priority elevation for process." + ], + [ + "r_emblemBrightnessScale", + "Modifier that scales the brightness of the emblem on model materials" + ], + [ + "r_emissiveMap", + "Replace all emissive maps with pure black or pure white" + ], + [ + "r_enableNoTessBuckets", + "Enables placing triangles that don't need tessellation into additional draw calls using non-tessellated shaders." + ], + [ + "r_envBrdfLutMap", + "Replace environment BRDF lookup table with pure black (no secondary specular) or pure white (maximum secondary specular)" + ], + [ + "r_envMapExponent", + "Reflection exponent." + ], + [ + "r_envMapMaxIntensity", + "Max reflection intensity based on glancing angle." + ], + [ + "r_envMapMinIntensity", + "" + ], + [ + "r_envMapOverride", + "" + ], + [ + "r_envMapSunIntensity", + "Max sun specular intensity intensity with env map materials." + ], + [ + "r_eyePupil", + " Change eye's pupil Size." + ], + [ + "r_eyeRedness", + " Change eye's redness." + ], + [ + "r_eyeWetness", + " Change eye's wetness." + ], + [ + "r_fastModelPrimaryLightCheck", + "Reduce R_GetNonSunPrimaryLightForSphere/R_GetNonSunPrimaryLightForBox function calls" + ], + [ + "r_fastModelPrimaryLightLink", + "Speed up R_LinkSphereEntityToPrimaryLights and R_LinkBoxEntityToPrimaryLights." + ], + [ + "r_filmAltShader", + "Use alternate shader (with middle tint and dark desat) for film color." + ], + [ + "r_filmTweakBrightness", + "Tweak dev var; film color brightness" + ], + [ + "r_filmTweakContrast", + "Tweak dev var; film color contrast" + ], + [ + "r_filmTweakDarkTint", + "" + ], + [ + "r_filmTweakDesaturation", + "Tweak dev var; Desaturation applied after all 3D drawing to light areas" + ], + [ + "r_filmTweakDesaturationDark", + "Tweak dev var; Additional desaturation applied after all 3D drawing to dark areas" + ], + [ + "r_filmTweakEnable", + "Tweak dev var; enable film color effects" + ], + [ + "r_filmTweakInvert", + "Tweak dev var; enable inverted video" + ], + [ + "r_filmTweakLightTint", + "" + ], + [ + "r_filmTweakMediumTint", + "" + ], + [ + "r_filmUseTweaks", + "Overide film effects with tweak dvar values." + ], + [ + "r_flushAfterExecute", + "Whether to Flush after ExecuteCommandList." + ], + [ + "r_fog", + "Set to 0 to disable fog" + ], + [ + "r_fog_depthhack_scale", + "Fog scale for depth hack surfaces" + ], + [ + "r_fog_ev_adjust", + "Fog color ev adjustment (+2 means fog color is 2 stops brighter)" + ], + [ + "r_font_cache_debug_display", + "Display the current fontcache texture on the HUD for debug purposes" + ], + [ + "r_forceLod", + "Force all level of detail to this level" + ], + [ + "r_fullbright", + "Toggles rendering without lighting" + ], + [ + "r_fxaaSubpixel", + "FXAA sub-pixel amount, lower values have more aliasing and less blur" + ], + [ + "r_FXAverageColorFunc", + "How to compute FX system average color? 0 = use IWrad equation, 1 = legacy equation, 2 = spherical harmonics 1 coefficient." + ], + [ + "r_globalGenericMaterialScale", + "Hack global generic material constants" + ], + [ + "r_glow_allowed", + "Allow glow." + ], + [ + "r_glow_allowed_script_forced", + "Force 'allow glow' to be treated as true, by script." + ], + [ + "r_gunSightColorEntityScale", + "Scale the gun sight color when over an entity." + ], + [ + "r_gunSightColorNoneScale", + "Scale the gun sight color when not over an entity." + ], + [ + "r_hbaoBias", + "HBAO bias" + ], + [ + "r_hbaoBlurEnable", + "HBAO blur enabled" + ], + [ + "r_hbaoBlurSharpness", + "HBAO blur sharpness" + ], + [ + "r_hbaoCoarseAO", + "HBAO coarse AO" + ], + [ + "r_hbaoDebug", + "Debug render HBAO occlusion" + ], + [ + "r_hbaoDetailAO", + "HBAO detail AO" + ], + [ + "r_hbaoPowerExponent", + "HBAO power exponent" + ], + [ + "r_hbaoRadius", + "HBAO radius" + ], + [ + "r_hbaoSceneScale", + "HBAO scene scale" + ], + [ + "r_hbaoStrengthBlend", + "Blend factor between the artist-tuned proportional strength r_hbaoStrengthScale*artStrength, and the fixed strength r_hbaoStrengthFixed. A value of 0.0 is fully the proportional value, and a value of 1.0 is fully the fixed value." + ], + [ + "r_hbaoStrengthFixed", + "Fixed HBAO strength. Only used if r_hbaoStrengthBlend > 0.0." + ], + [ + "r_hbaoStrengthScale", + "Scale factor to convert SSAO strength to HBAO strength. Only used if r_hbaoStrengthBlend < 1.0." + ], + [ + "r_hbaoUseScriptScale", + "Enable/disable script-controlled strength scale while HBAO is active." + ], + [ + "r_hemiAoBlurTolerance", + "Hemi SSAO Blur Tolerance (log10)" + ], + [ + "r_hemiAoCombineResolutionsBeforeBlur", + "The higher quality modes blend wide and narrow sampling patterns. The wide pattern is due to deinterleaving and requires blurring. The narrow pattern is not on a deinterleaved buffer, but it only samples every other pixel. The blur on it is optional. If you combine the two before blurring, the narrow will get blurred as well. This creates a softer effect but can remove any visible noise from having 50% sample coverage." + ], + [ + "r_hemiAoCombineResolutionsWithMul", + "When combining the wide and narrow patterns, a mul() operation can be used or a min() operation. Multiplication exaggerates the result creating even darker creases. This is an artistic choice. I think it looks less natural, but often art teams prefer more exaggerated contrast. For me, it's more about having the right AO falloff so that it's a smooth gradient rather than falling off precipitously and forming overly dark recesses." + ], + [ + "r_hemiAoDepthSquash", + "Hemi SSAO depth squash. Value is rcp." + ], + [ + "r_hemiAoEnable", + "Enable Hemi SSAO" + ], + [ + "r_hemiAoHierarchyDepth", + "Hemi SSAO recursion depth (filter width)" + ], + [ + "r_hemiAoMaxDepthDownsample", + "Use max depth value when downsampling, instead of pseudo-randomly picking a depth sample? Leaving this at the default false may produce more stable results." + ], + [ + "r_hemiAoNoiseFilterTolerance", + "This is necessary to filter out pixel shimmer due to bilateral upsampling with too much lost resolution. High frequency detail can sometimes not be reconstructed, and the noise filter fills in the missing pixels with the result of the higher resolution SSAO. Value is log10." + ], + [ + "r_hemiAoPower", + "Power curve applied to Hemi SSAO factor, not applied in game yet" + ], + [ + "r_hemiAoQualityLevel", + "Hemi SSAO quality setting" + ], + [ + "r_hemiAoRejectionFalloff", + "Controls how aggressive to fade off samples that occlude spheres but by so much as to be unreliable. This is what gives objects a dark halo around them when placed in front of a wall. If you want to fade off the halo, boost your rejection falloff. The tradeoff is that it reduces overall AO. Value is rcp." + ], + [ + "r_hemiAoStrength", + "Strength of Hemi Screen Space Ambient Occlusion effect" + ], + [ + "r_hemiAoUpsampleTolerance", + "Hemi SSAO Upsample Tolerance (log10)" + ], + [ + "r_heroLighting", + "Enable hero-only lighting" + ], + [ + "r_highLodDist", + "Distance for high level of detail" + ], + [ + "r_hudFx", + "Draw HUD Effects" + ], + [ + "r_hudOutlineEnable", + "Enables wireframe outlines to be drawn around DObjs (as a post process)." + ], + [ + "r_hudOutlinePostMode", + "hud outline apply mode" + ], + [ + "r_hudOutlineWidth", + "Set the width of the Hud Outline" + ], + [ + "r_ignore", + "" + ], + [ + "r_ignoref", + "" + ], + [ + "r_imageQuality", + "Image quality" + ], + [ + "r_inGameVideo", + "Allow in game cinematics" + ], + [ + "r_lateAllocParamCacheAllowed", + "Enable late allocation of parameter cache for VS stage." + ], + [ + "r_lateAllocParamCacheDefault", + "Late allocation of parameter cache value for sub-div materials." + ], + [ + "r_lateAllocParamCacheDisplacement", + "Late allocation of parameter cache value for sub-div materials." + ], + [ + "r_lateAllocParamCacheSubdiv", + "Late allocation of parameter cache value for sub-div materials." + ], + [ + "r_lightCacheLessFrequentMaxDistance", + "Adjust the distance fx models (and models tagged as less-frequently-lit by script) move before immediately being relit" + ], + [ + "r_lightCacheLessFrequentPeriod", + "Adjust how frequently fx models (and models tagged as less-frequently-lit by script) get relit on average (1 is every frame, 8 is every 8th frame)" + ], + [ + "r_lightGridAvgApplyPrimaryLight", + "apply primary light color onto r_showLightGridAvgProbes boxes" + ], + [ + "r_lightGridAvgFollowCamera", + "allow the r_showLightGridAvgProbes boxes following current camera position" + ], + [ + "r_lightGridAvgProbeCount", + "how many light grid avg color probes will show up)" + ], + [ + "r_lightGridAvgTraceGround", + " lock boxes to ground " + ], + [ + "r_lightGridContrast", + "Adjust the contrast of light color from the light grid" + ], + [ + "r_lightGridDefaultFXLightingLookup", + "Default FX lighting lookup location\n" + ], + [ + "r_lightGridDefaultModelLightingLookup", + "Default model lighting lookup location" + ], + [ + "r_lightGridEnableTweaks", + "Enable tweaks of the light color from the light grid" + ], + [ + "r_lightGridIntensity", + "Adjust the intensity of light color from the light grid" + ], + [ + "r_lightGridSHBands", + "Spherical harmonics bands being used for evaluating current-gen light grids colors. 0 = default, 1 = 1 band, 2 = 2 bands, 3 = 3 bands.\n" + ], + [ + "r_lightGridUseTweakedValues", + "Use tweaked values instead of default" + ], + [ + "r_lightMap", + "Replace all lightmaps with pure black or pure white" + ], + [ + "r_litSurfaceHDRScalar", + "Vision set based scalar applied to lit surfaces" + ], + [ + "r_loadForRenderer", + "Set to false to disable dx allocations (for dedicated server mode)" + ], + [ + "r_lockPvs", + "Lock the viewpoint used for determining what is visible to the current position and direction" + ], + [ + "r_lod4Dist", + "Distance for lowest level of detail 4" + ], + [ + "r_lod5Dist", + "Distance for lowest level of detail 5" + ], + [ + "r_lodBiasRigid", + "" + ], + [ + "r_lodBiasSkinned", + "" + ], + [ + "r_lodScaleRigid", + "" + ], + [ + "r_lodScaleSkinned", + "" + ], + [ + "r_lowestLodDist", + "Distance for lowest level of detail" + ], + [ + "r_lowLodDist", + "Distance for low level of detail" + ], + [ + "r_mbEnable", + "Set of objects which will be enabled for motion blur" + ], + [ + "r_mbFastEnable", + "Toggle on/off fast high quality motion blur" + ], + [ + "r_mbFastPreset", + "Changes motion blur quality" + ], + [ + "r_mdao", + "Enable the medium distance ambient occlusion feature" + ], + [ + "r_mdaoAsyncOccluderGen", + "The occluder generation step is performed via async compute" + ], + [ + "r_mdaoBoneInfluenceRadiusScale", + "Scale for the bone influence radius for mdao" + ], + [ + "r_mdaoCapsuleStrength", + "MDAO strength for capsule occluders" + ], + [ + "r_mdaoMinBoneBoundsToOcclude", + "Minimum volume of the bone collider to create occluders for" + ], + [ + "r_mdaoOccluderCullDistance", + "Culling distance for mdao occluders" + ], + [ + "r_mdaoOccluderFadeOutStartDistance", + "Fade out distance for mdao occluders" + ], + [ + "r_mdaoUseTweaks", + "Use r_mdao* dvars instead of the current light set values for MDAO" + ], + [ + "r_mdaoVolumeStrength", + "MDAO strength for volume occluders" + ], + [ + "r_mediumLodDist", + "Distance for medium level of detail" + ], + [ + "r_mode", + "Display mode" + ], + [ + "r_modelLightingMap", + "Replace all model lighting maps (light grid) with pure black" + ], + [ + "r_monitor", + "Index of the monitor to use in a multi monitor system; 0 picks automatically." + ], + [ + "r_mpRimColor", + "Change character's rim color for multiplayer" + ], + [ + "r_mpRimDiffuseTint", + "Change character's rim diffuse tint for multiplayer." + ], + [ + "r_mpRimStrength", + "Change character's rim color for multiplayer" + ], + [ + "r_multiGPU", + "Enable multi GPU compat mode." + ], + [ + "r_normalMap", + "Replace all normal maps with a flat normal map" + ], + [ + "r_nvidiaGPU", + "Enable NV API." + ], + [ + "r_offchipTessellationAllowed", + "Enable off-chip tessellation support." + ], + [ + "r_offchipTessellationTfThreshold", + "Tessellation factor threshold for off-chip." + ], + [ + "r_offchipTessellationWaveThreshold", + "Domain shader wave threshold for off-chip." + ], + [ + "r_omitUnusedRenderTargets", + "Omit unused render targets to save memory. Changing this requires a vid_restart." + ], + [ + "r_outdoor", + "Prevents snow from going indoors" + ], + [ + "r_outdoorFeather", + "Outdoor z-feathering value" + ], + [ + "r_particleHdr", + "Enable Hdr Particle Features" + ], + [ + "r_patchCountAllowed", + "Enable run-time setting of patch count per draw call." + ], + [ + "r_picmip", + "Picmip level of color maps. If r_picmip_manual is 0, this is read-only." + ], + [ + "r_picmip_bump", + "Picmip level of normal maps. If r_picmip_manual is 0, this is read-only." + ], + [ + "r_picmip_spec", + "Picmip level of specular maps. If r_picmip_manual is 0, this is read-only." + ], + [ + "r_picmip_water", + "Picmip level of water maps." + ], + [ + "r_polygonOffsetBias", + "Offset bias for decal polygons; bigger values z-fight less but poke through walls more" + ], + [ + "r_polygonOffsetClamp", + "Offset clamp for decal polygons; bigger values z-fight less but poke through walls more" + ], + [ + "r_polygonOffsetScale", + "Offset scale for decal polygons; bigger values z-fight less but poke through walls more" + ], + [ + "r_portalBevels", + "Helps cull geometry by angles of portals that are acute when projected onto the screen, value is the cosine of the angle" + ], + [ + "r_portalBevelsOnly", + "Use screen-space bounding box of portals rather than the actual shape of the portal projected onto the screen" + ], + [ + "r_portalMinClipArea", + "Don't clip child portals by a parent portal smaller than this fraction of the screen area." + ], + [ + "r_portalMinRecurseDepth", + "Ignore r_portalMinClipArea for portals with fewer than this many parent portals." + ], + [ + "r_portalWalkLimit", + "Stop portal recursion after this many iterations. Useful for debugging portal errors." + ], + [ + "r_postAA", + "Post process antialiasing mode" + ], + [ + "r_postfx_enable", + "Enable post-processing effects such as color correction, bloom, depth-of-field, etc." + ], + [ + "r_preloadShaders", + "Force D3D to draw dummy geometry with all shaders during level load; may fix long pauses at level start." + ], + [ + "r_primaryLightTweakDiffuseStrength", + "Tweak the diffuse intensity for primary lights" + ], + [ + "r_primaryLightTweakSpecularStrength", + "Tweak the specular intensity for primary lights" + ], + [ + "r_primaryLightUseTweaks", + "" + ], + [ + "r_reactiveMotionActorRadius", + "Radial distance from the ai characters that influences reactive motion models (inches)" + ], + [ + "r_reactiveMotionActorVelocityMax", + "AI velocity considered the maximum when determining the length of motion tails (inches/sec)" + ], + [ + "r_reactiveMotionActorZOffset", + "Distance from the actor origin along Z direction where the actor's reactive motion effector sphere is centered at." + ], + [ + "r_reactiveMotionEffectorStrengthScale", + "Additional scale for the effector influence, as a factor of the model part distance from the effector center and model part extents" + ], + [ + "r_reactiveMotionHelicopterLimit", + "Maximum number of helicopter entities that actively influence reactive motion. Can increase CPU cost of the scene." + ], + [ + "r_reactiveMotionHelicopterRadius", + "Radial distance from the helicopter that influences reactive motion models (inches)" + ], + [ + "r_reactiveMotionHelicopterStrength", + "Scales the influence of helicopter wind tunnel motion" + ], + [ + "r_reactiveMotionPlayerHeightAdjust", + "Amount to adjust the vertical distance of the effector from the player position (inches)" + ], + [ + "r_reactiveMotionPlayerRadius", + "Radial distance from the player that influences reactive motion models (inches)" + ], + [ + "r_reactiveMotionPlayerZOffset", + "Distance from the player origin along Z direction where the player's reactive motion effector sphere is centered at." + ], + [ + "r_reactiveMotionVelocityTailScale", + "Additional scale for the velocity-based motion tails, as a factor of the effector radius" + ], + [ + "r_reactiveMotionWindAmplitudeScale", + "Scales amplitude of wind wave motion" + ], + [ + "r_reactiveMotionWindAreaScale", + "Scales distribution of wind motion" + ], + [ + "r_reactiveMotionWindDir", + "Controls the global wind direction" + ], + [ + "r_reactiveMotionWindFrequencyScale", + "Scales frequency of wind wave motion" + ], + [ + "r_reactiveMotionWindStrength", + "Scale of the global wind direction (inches/sec)" + ], + [ + "r_reflectionProbeMap", + "Replace all reflection probes with pure black" + ], + [ + "r_reflectionProbeNmlLuminance", + "Enable/disable shader code for computing luminance during reflection probe denormalization. This is just an experiment.\n" + ], + [ + "r_rimLight0Color", + "" + ], + [ + "r_rimLight0Heading", + "Rim Light 0 heading in degrees" + ], + [ + "r_rimLight0Pitch", + "Rim Light 0 pitch in degrees -90 is noon." + ], + [ + "r_rimLightBias", + "How much to bias the n.l calculation" + ], + [ + "r_rimLightDiffuseIntensity", + "Strength of the diffuse component of the rim light." + ], + [ + "r_rimLightFalloffMaxDistance", + "Distance at which the rim light hits intensity of 100%." + ], + [ + "r_rimLightFalloffMinDistance", + "Distance at which the rim light hits intensity of 100%." + ], + [ + "r_rimLightFalloffMinIntensity", + "Intensity of the effect at and before minDistance." + ], + [ + "r_rimLightPower", + "Power to raise the n.l calculation" + ], + [ + "r_rimLightSpecIntensity", + "Strength of the spec ( additive) component of the rim light" + ], + [ + "r_rimLightUseTweaks", + "Turn on rim lighting tweaks" + ], + [ + "r_scaleViewport", + "Scale 3D viewports by this fraction. Use this to see if framerate is pixel shader bound." + ], + [ + "r_sceneMipShowOverlay", + "Toggles scene mip rendertarget overlay" + ], + [ + "r_showLightGrid", + "Show light grid debugging information (2: detailed, 3: detailed for this box only)" + ], + [ + "r_showLightGridAvgProbes", + "show an array of boxes which are using the light grid average color at its location" + ], + [ + "r_showLightGridDetailInfo", + "Show more details for light grid debugging." + ], + [ + "r_showLightProbes", + "Show the light probes at the light grid sample locations in world space centered around the camera." + ], + [ + "r_showMissingLightGrid", + "Use rainbow colors for entities that are outside the light grid" + ], + [ + "r_showModelLightingLowWaterMark", + "" + ], + [ + "r_showPortals", + "Show portals for debugging" + ], + [ + "r_showPortalsOverview", + "Render 2d XY portal overlay scaled to fit to this distance. Useful for debugging portal errors." + ], + [ + "r_showReflectionProbeSelection", + "Show reflection probe selection" + ], + [ + "r_singleCell", + "Only draw things in the same cell as the camera. Most useful for seeing how big the current cell is." + ], + [ + "r_skipPvs", + "Skipt the determination of what is in the potentially visible set (disables most drawing)" + ], + [ + "r_sky_fog_intensity", + "Amount of sky fog fading" + ], + [ + "r_sky_fog_max_angle", + "End of angular sky fog fading" + ], + [ + "r_sky_fog_min_angle", + "Start of angular sky fog fading" + ], + [ + "r_skyFogUseTweaks", + "Enable dvar control of sky fog" + ], + [ + "r_smaaThreshold", + "SMAA edge detection threshold" + ], + [ + "r_smodelInstancedRenderer", + "Render static models with instanced renderer" + ], + [ + "r_smodelInstancedThreshold", + "Minimum number of static model instances before instanced rendering is used" + ], + [ + "r_smp_backend", + "Process renderer back end in a separate thread" + ], + [ + "r_smp_worker", + "Process renderer front end in a separate thread" + ], + [ + "r_smp_worker_thread0", + "" + ], + [ + "r_smp_worker_thread1", + "" + ], + [ + "r_smp_worker_thread2", + "" + ], + [ + "r_smp_worker_thread3", + "undefined" + ], + [ + "r_smp_worker_thread4", + "undefined" + ], + [ + "r_smp_worker_thread5", + "undefined" + ], + [ + "r_smp_worker_thread6", + "undefined" + ], + [ + "r_smp_worker_thread7", + "undefined" + ], + [ + "r_specOccMap", + "Replace all specular occlusion maps with pure black (fully occluded) or pure white (not occluded)" + ], + [ + "r_specularColorScale", + "Set greater than 1 to brighten specular highlights" + ], + [ + "r_specularMap", + "Replace all specular maps with pure black (off) or pure white (super shiny)" + ], + [ + "r_spotLightEntityShadows", + "Enable entity shadows for spot lights." + ], + [ + "r_spotLightShadows", + "Enable shadows for spot lights." + ], + [ + "r_ssao", + "Screen Space Ambient Occlusion mode" + ], + [ + "r_ssaoDebug", + "Render calculated or blurred Screen Space Ambient Occlusion values" + ], + [ + "r_ssaoDebugMip", + "Selects which mip level to render when r_ssaoDebug is enabled. If 0 and r_ssaoDownsample is enabled, will render mip 1." + ], + [ + "r_ssaoDepthScale", + "Scale applied to depth values used for occlusion tests." + ], + [ + "r_ssaoDepthScaleViewModel", + "Scale applied to depth values used for occlusion tests." + ], + [ + "r_ssaoDownsample", + "Screen Space Ambient Occlusion calculation occurs at half linear resolution" + ], + [ + "r_ssaoFadeDepth", + "Depth at which the SSAO begins to fade out. It fades at even increments of this distance (e.g. it's at 1 for depth r_ssaoFadeDepth, 1/2 for depth 2*r_ssaoFadeDepth, etc.)" + ], + [ + "r_ssaoGapFalloff", + "Falloff used to blend between creases (that should darken) and silhouettes (that should not darken). Lower values falloff more quickly." + ], + [ + "r_ssaoGradientFalloff", + "Falloff used to fade out the effect for steep depth gradients (i.e. surfaces nearly parallel to the camera direction). This fixes sampling artifacts that appear for surfaces nearly parallel to the camera direction (commonly occuring for flat ground planes)." + ], + [ + "r_ssaoMaxStrengthDepth", + "Depth at which SSAO strength is at its maximum" + ], + [ + "r_ssaoMethod", + "Screen Space Ambient Occlusion method (original or IW6, both are volumetric obscurance)" + ], + [ + "r_ssaoMinPixelWidth", + "Minimum pixel width of the effect. When the effect is smaller than this, it is culled entirely." + ], + [ + "r_ssaoMinStrengthDepth", + "Depth at which SSAO strength is zero, effectively disabled" + ], + [ + "r_ssaoMultiRes", + "Screen Space Ambient Occlusion calculation occurs at half linear resolution" + ], + [ + "r_ssaoPower", + "Power curve applied to SSAO factor" + ], + [ + "r_ssaoRejectDepth", + "Depth at which the SSAO is disabled. Smaller values result in more rejected pixels which is faster, but limits the distance at which the effect is visible." + ], + [ + "r_ssaoSampleCount", + "Selects the number of samples used for SSAO" + ], + [ + "r_ssaoScriptScale", + "Allows script to lerp to disable or enable the SSAO. This applies a scalar value to the SSAO strength. When set to 0, this effectively disables SSAO." + ], + [ + "r_ssaoStrength", + "Strength of Screen Space Ambient Occlusion effect" + ], + [ + "r_ssaoUseTweaks", + "Use r_ssao* dvars instead of the current light set values for SSAO" + ], + [ + "r_ssaoWidth", + "The width of the SSAO effect, in pixels at 720p. Larger values increase area but lower effective quality." + ], + [ + "r_sse_skinning", + "Use Streaming SIMD Extensions for skinning" + ], + [ + "r_ssrBlendScale", + "Add extra scale to ssr weight versus reflection probe weight, >1 value will make ssr more obvious." + ], + [ + "r_ssrFadeInDuration", + "Duration of the screen-space reflection fade-in, which occurs whenever the reflection source buffer is invalidated due to view changes (in particular, dual-view scope transitions)." + ], + [ + "r_ssrPositionCorrection", + "Screen space reflection position correction blend factor" + ], + [ + "r_ssrRoughnessMipParameters", + "X: mirror mip; Y: roughest mip; Z: roughness middle point, may need different value for different screen resolution on PC." + ], + [ + "r_sssBlendWeight", + "Controls the blend between the wide (zero) and narrow (one) gaussians" + ], + [ + "r_sssDebugMaterial", + "Debug Feature: toggle materials with SSS" + ], + [ + "r_sssEnable", + "Enables the subsurface scattering effect (note that disabling SSS will not prevent the filter from running)" + ], + [ + "r_sssGlobalRadius", + "Controls the global radius (in inches)" + ], + [ + "r_sssJitterRadius", + "Percentage of the kernel to be jittered" + ], + [ + "r_sssNarrowRadius", + "Controls the narrow Gaussian radius" + ], + [ + "r_sssPreset", + "Changes subsurface scattering quality" + ], + [ + "r_sssWideRadius", + "Controls the wide Gaussian radius" + ], + [ + "r_subdiv", + "Enables Catmull Clark surface subdivision." + ], + [ + "r_subdivLimit", + "Set the maximum Catmull Clark subdivision level." + ], + [ + "r_subdivPatchCount", + "Patches per thread group for sub-division surfaces." + ], + [ + "r_subdomainLimit", + "Maximum number of extra tessellation subdivisions using instancing (max tess amts are 0:64, 1:128, 2:192, 3:256, max instances used are 0:1, 1:4, 2:9, 3:12)" + ], + [ + "r_subdomainScale", + "Debug only: Scales the extra subdivision amount (for values < 1, not all instanced sub triangles will draw)." + ], + [ + "r_subwindow", + "subwindow to draw: left, right, top, bottom" + ], + [ + "r_sun_from_dvars", + "Set sun flare values from dvars rather than the level" + ], + [ + "r_sun_fx_position", + "Position in degrees of the sun effect" + ], + [ + "r_sunblind_fadein", + "time in seconds to fade blind from 0% to 100%" + ], + [ + "r_sunblind_fadeout", + "time in seconds to fade blind from 100% to 0%" + ], + [ + "r_sunblind_max_angle", + "angle from sun in degrees inside which effect is max" + ], + [ + "r_sunblind_max_darken", + "0-1 fraction for how black the world is at max blind" + ], + [ + "r_sunblind_min_angle", + "angle from sun in degrees outside which effect is 0" + ], + [ + "r_sunflare_fadein", + "time in seconds to fade alpha from 0% to 100%" + ], + [ + "r_sunflare_fadeout", + "time in seconds to fade alpha from 100% to 0%" + ], + [ + "r_sunflare_max_alpha", + "0-1 vertex color and alpha of sun at max effect" + ], + [ + "r_sunflare_max_angle", + "angle from sun in degrees inside which effect is max" + ], + [ + "r_sunflare_max_size", + "largest size of flare effect in pixels at 640x480" + ], + [ + "r_sunflare_min_angle", + "angle from sun in degrees outside which effect is 0" + ], + [ + "r_sunflare_min_size", + "smallest size of flare effect in pixels at 640x480" + ], + [ + "r_sunflare_shader", + "name for flare effect; can be any material" + ], + [ + "r_sunglare_fadein", + "time in seconds to fade glare from 0% to 100%" + ], + [ + "r_sunglare_fadeout", + "time in seconds to fade glare from 100% to 0%" + ], + [ + "r_sunglare_max_angle", + "angle from sun in degrees inside which effect is max" + ], + [ + "r_sunglare_max_lighten", + "0-1 fraction for how white the world is at max glare" + ], + [ + "r_sunglare_min_angle", + "angle from sun in degrees inside which effect is max" + ], + [ + "r_sunInfDist", + "Sun infinite distance used to place sun fx" + ], + [ + "r_sunshadowmap_cmdbuf_worker", + "Process shadowmap command buffer in a separate thread" + ], + [ + "r_sunsprite_shader", + "name for static sprite; can be any material" + ], + [ + "r_sunsprite_size", + "diameter in pixels at 640x480 and 80 fov" + ], + [ + "r_surfaceHDRScalarUseTweaks", + "Enables lit and unlit surface scalar tweaks" + ], + [ + "r_tessellation", + "Enables tessellation of world geometry, with an optional cutoff distance." + ], + [ + "r_tessellationCutoffDistance", + "Distance at which world geometry ceases to tessellate." + ], + [ + "r_tessellationCutoffFalloff", + "Range over which tessellation is faded out, up to the cutoff." + ], + [ + "r_tessellationEyeScale", + "Scale applied due to eye * object normal for less tessellation on facing polygons." + ], + [ + "r_tessellationFactor", + "Target edge length, based on dividing full window height by this factor, for dynamic tessellation. Use zero to disable tessellation." + ], + [ + "r_tessellationHeightAuto", + "Correctly auto scale displacement heights for layers to grow as texture is stretched over larger surface areas to preserve feature proportions." + ], + [ + "r_tessellationHeightScale", + "Displacement height scale factor." + ], + [ + "r_tessellationHybrid", + "Hybrid per pixel displacement scale." + ], + [ + "r_tessellationLodBias", + "Displacement map lod bias." + ], + [ + "r_texFilterAnisoMax", + "Maximum anisotropy to use for texture filtering" + ], + [ + "r_texFilterAnisoMin", + "Minimum anisotropy to use for texture filtering (overridden by max)" + ], + [ + "r_texFilterDisable", + "Disables all texture filtering (uses nearest only.)" + ], + [ + "r_texFilterMipBias", + "Change the mipmap bias" + ], + [ + "r_texFilterMipMode", + "Forces all mipmaps to use a particular blend between levels (or disables mipping.)" + ], + [ + "r_texFilterProbeBilinear", + "Force reflection probe to use bilinear filter" + ], + [ + "r_texShowMipMode", + "Forces textures with the specified mip filtering to draw black." + ], + [ + "r_thermalColorOffset", + "Offset of the thermal colors (offset + scale*color)" + ], + [ + "r_thermalColorScale", + "Scale of the thermal colors (offset + scale*color)" + ], + [ + "r_thermalDetailScale", + "Scale of the detail that is added to the thermal map from the normal map (multiplies the detail amount from AssetManager)" + ], + [ + "r_thermalFadeColor", + "Color the thermal fades to at distance." + ], + [ + "r_thermalFadeControl", + "Select thermal fade mode" + ], + [ + "r_thermalFadeMax", + "Distance at which thermal stops fading" + ], + [ + "r_thermalFadeMin", + "Distance at which thermal starts fading" + ], + [ + "r_tonemap", + "HDR Tonemapping mode" + ], + [ + "r_tonemapAdaptSpeed", + "HDR Tonemap exposure adaptation speed" + ], + [ + "r_tonemapAuto", + "HDR Tonemapping performs auto-exposure" + ], + [ + "r_tonemapAutoExposureAdjust", + "HDR Tonemap Auto Exposure Adjust value (set to 0.0 for automatic adjustment)" + ], + [ + "r_tonemapBlack", + "HDR Filmic Tonemap black point" + ], + [ + "r_tonemapBlend", + "HDR Tonemapping blends between exposures" + ], + [ + "r_tonemapCrossover", + "HDR Filmic Tonemap crossover point" + ], + [ + "r_tonemapDarkEv", + "HDR Tonemap Dark EV" + ], + [ + "r_tonemapDarkExposureAdjust", + "HDR Tonemap Dark Exposure Adjust" + ], + [ + "r_tonemapExposure", + "HDR Tonemap exposure (in EV) override (only works in non-auto mode)" + ], + [ + "r_tonemapExposureAdjust", + "HDR Tonemap exposure adjustment (in EV, 0 is no adjustment, works like a camera where +1 reduces EV by 1)" + ], + [ + "r_tonemapGamma", + "HDR Tonemap gamma curve power" + ], + [ + "r_tonemapHighlightRange", + "HDR Tonemap dynamic range, which determines white point luminance" + ], + [ + "r_tonemapLightEv", + "HDR Tonemap Light EV" + ], + [ + "r_tonemapLightExposureAdjust", + "HDR Tonemap Light Exposure Adjust" + ], + [ + "r_tonemapLockAutoExposureAdjust", + "HDR Tonemapping lock auto exposure adjust" + ], + [ + "r_tonemapMaxExposure", + "HDR Tonemap maximum exposure (in EV)" + ], + [ + "r_tonemapMaxExposureAdjust", + "HDR Tonemap Max Exposure Adjust" + ], + [ + "r_tonemapMidEv", + "HDR Tonemap Mid EV" + ], + [ + "r_tonemapMidExposureAdjust", + "HDR Tonemap Mid Exposure Adjust" + ], + [ + "r_tonemapMinExposureAdjust", + "HDR Tonemap Min Exposure Adjust" + ], + [ + "r_tonemapShoulder", + "HDR Filmic Tonemap shoulder control (0 is linear)" + ], + [ + "r_tonemapToe", + "HDR Filmic Tonemap toe control (0 is linear)" + ], + [ + "r_tonemapUseCS", + "HDR Tonemapping uses compute shader." + ], + [ + "r_tonemapUseTweaks", + "Override tone map LightSet settings with tweak dvar values." + ], + [ + "r_tonemapWhite", + "HDR Filmic Tonemap white point" + ], + [ + "r_ui3d_debug_display", + "Show UI3D debug overlay" + ], + [ + "r_ui3d_h", + "ui3d texture window height" + ], + [ + "r_ui3d_use_debug_values", + "Use UI debug values" + ], + [ + "r_ui3d_w", + "ui3d texture window width" + ], + [ + "r_ui3d_x", + "ui3d texture window x" + ], + [ + "r_ui3d_y", + "ui3d texture window y" + ], + [ + "r_uiBlurDstMode", + "UI blur distortion mode. Fast uses the scene mip map render target, PostSun uses a downsampled post sun resolve buffer, PostSun HQ uses a gaussian blurred post sun resolve buffer." + ], + [ + "r_umbra", + "Enables Umbra-based portal culling." + ], + [ + "r_umbraAccurateOcclusionThreshold", + "The distance (in inches) to which accurate occlusion information is gathered. -1.0 = deduced automatically." + ], + [ + "r_umbraExclusive", + "Toggle Umbra for exclusive static culling (disables static portal dpvs)" + ], + [ + "r_umbraQueryParts", + "The number of parts the Umbra query frustum is broken into for async query processing as an M x N grid (0, 0 = all queries are synchronous)." + ], + [ + "r_umbraUseBadPlaces", + "Enable/disable ability to disable umbra when inside special volumes defined in mp/umbraBadPlaces.csv." + ], + [ + "r_umbraUseDpvsCullDist", + "Use cull distance from the DPVS instead of the far plane distance." + ], + [ + "r_unlitSurfaceHDRScalar", + "Vision set based scalar applied to unlit surfaces to balance those surfaces with the luminance of the scene" + ], + [ + "r_useComputeSkinning", + "Enables compute shader (GPU) skinning." + ], + [ + "r_useLayeredMaterials", + "Set to true to use layered materials on shader model 3 hardware" + ], + [ + "r_useLightGridDefaultFXLightingLookup", + "Enable/disable default fx lighting lookup\n" + ], + [ + "r_useLightGridDefaultModelLightingLookup", + "Enable/disable default model lighting lookup\n" + ], + [ + "r_useShadowGeomOpt", + "Enable iwRad shadow geometry optimization. It only works when we have the data generated in iwRad." + ], + [ + "r_useSunShadowPortals", + "Enable sun shadow portals when dir light change and not using cached shadow." + ], + [ + "r_useXAnimIK", + "Enables IK animation." + ], + [ + "r_vc_makelog", + "Enable logging of light grid points for the vis cache. 1 starts from scratch, 2 appends." + ], + [ + "r_vc_showlog", + "Show this many rows of light grid points for the vis cache" + ], + [ + "r_veil", + "Apply veiling luminance (HDR glow)" + ], + [ + "r_veilAntialiasing", + "Veil antialiasing mode (downsample technique used for first mip)." + ], + [ + "r_veilBackgroundStrength", + "Strength of background when applying veiling luminance (HDR glow)" + ], + [ + "r_veilFalloffScale1", + "Controls the size of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)" + ], + [ + "r_veilFalloffScale2", + "Controls the size of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)" + ], + [ + "r_veilFalloffWeight1", + "Controls the weight of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)" + ], + [ + "r_veilFalloffWeight2", + "Controls the weight of individual Gaussians (Gaussians 4-6 in XYZ, where Gaussian 6 is the wider one)" + ], + [ + "r_veilFilter", + "Changes the veil filtering mode" + ], + [ + "r_veilPreset", + "Changes veil sampling quality" + ], + [ + "r_veilRadius", + "Controls the radius of the first Gaussian in virtual pixels (remaining Gaussians follow proportionally)." + ], + [ + "r_veilStrength", + "Strength of veiling luminance (HDR glow)" + ], + [ + "r_veilUseTweaks", + "Override veil LightSet settings with tweak dvar values." + ], + [ + "r_velocityPrepass", + "Perform velocity rendering during the depth prepass" + ], + [ + "r_viewModelLightAmbient", + "" + ], + [ + "r_viewModelPrimaryLightTweakDiffuseStrength", + "Tweak the diffuse intensity for view model primary lights" + ], + [ + "r_viewModelPrimaryLightTweakSpecularStrength", + "Tweak the specular intensity for view model primary lights" + ], + [ + "r_viewModelPrimaryLightUseTweaks", + "" + ], + [ + "r_volumeLightScatter", + "Enables volumetric light scattering" + ], + [ + "r_volumeLightScatterAngularAtten", + "Distance of sun from center of screen before angular attenuation starts for god rays" + ], + [ + "r_volumeLightScatterBackgroundDistance", + "Distance at which pixels are considered background for volume light scatter effect" + ], + [ + "r_volumeLightScatterColor", + "" + ], + [ + "r_volumeLightScatterDepthAttenFar", + "Pixels >= than this depth recieve full volume light scatter." + ], + [ + "r_volumeLightScatterDepthAttenNear", + "Pixels <= than this depth recieve no volume light scatter." + ], + [ + "r_volumeLightScatterEv", + "Light intensity (in EV) for volumetric light scattering" + ], + [ + "r_volumeLightScatterLinearAtten", + "Coefficient of linear attenuation of god rays" + ], + [ + "r_volumeLightScatterQuadraticAtten", + "Coefficient of quadratic attenuation of god rays)" + ], + [ + "r_volumeLightScatterUseTweaks", + "Enables volumetric light scattering tweaks" + ], + [ + "r_vsync", + "Enable v-sync before drawing the next frame to avoid 'tearing' artifacts." + ], + [ + "r_warningRepeatDelay", + "Number of seconds after displaying a \"per-frame\" warning before it will display again" + ], + [ + "r_wideTessFactorsThreshold", + "If a surface has more than this many triangles, process triangles in parallel instead of surfaces." + ], + [ + "r_zfar", + "Change the distance at which culling fog reaches 100% opacity; 0 is off" + ], + [ + "r_znear", + "Things closer than this aren't drawn. Reducing this increases z-fighting in the distance." + ], + [ + "radarjamDistMax", + "" + ], + [ + "radarjamDistMin", + "" + ], + [ + "radarjamSinCurve", + "" + ], + [ + "radius_damage_debug", + "Turn on debug lines for radius damage traces" + ], + [ + "ragdoll_baselerp_time", + "Default time ragdoll baselerp bones take to reach the base pose" + ], + [ + "ragdoll_bullet_force", + "Bullet force applied to ragdolls" + ], + [ + "ragdoll_bullet_upbias", + "Upward bias applied to ragdoll bullet effects" + ], + [ + "ragdoll_dump_anims", + "Dump animation data when ragdoll fails" + ], + [ + "ragdoll_enable", + "Turn on ragdoll death animations" + ], + [ + "ragdoll_explode_force", + "Explosive force applied to ragdolls" + ], + [ + "ragdoll_explode_upbias", + "Upwards bias applied to ragdoll explosion effects" + ], + [ + "ragdoll_exploding_bullet_force", + "Force applied to ragdolls from explosive bullets" + ], + [ + "ragdoll_exploding_bullet_upbias", + "Upwards bias applied to ragdoll from explosive bullets" + ], + [ + "ragdoll_idle_min_velsq", + "Minimum squared speed a ragdoll body needs to be moving before it will shut down due to time" + ], + [ + "ragdoll_jitter_scale", + "Scale up or down the effect of physics jitter on ragdolls" + ], + [ + "ragdoll_jointlerp_time", + "Default time taken to lerp down ragdoll joint friction" + ], + [ + "ragdoll_link_to_moving_platform", + "Enable client-side linking of ragdolls to script brush models when they go idle." + ], + [ + "ragdoll_max_life", + "Max lifetime of a ragdoll system in msec" + ], + [ + "ragdoll_max_simulating", + "Max number of simultaneous active ragdolls - archived" + ], + [ + "ragdoll_max_stretch_pct", + "Force ragdoll limbs to not stretch more than this percentage in one frame" + ], + [ + "ragdoll_mp_limit", + "Max number of simultaneous active ragdolls - archived" + ], + [ + "ragdoll_mp_resume_share_after_killcam", + "Msec after returning from killcam that splitscreen players will share ragdolls again." + ], + [ + "ragdoll_resolve_penetration_bias", + "Bias value on force to push ragdolls out of environment." + ], + [ + "ragdoll_rotvel_scale", + "Ragdoll rotational velocity estimate scale" + ], + [ + "ragdoll_self_collision_scale", + "Scale the size of the collision capsules used to prevent ragdoll limbs from interpenetrating" + ], + [ + "ragdoll_stretch_iters", + "Iterations to run the alternate limb solver" + ], + [ + "rankedPlayEndMatchKeepLobby", + "keep the lobby if the lobby host is in our private party." + ], + [ + "rankedPlaylistLockoutDuration", + "Time in seconds to lock the ranked play playlist if a player quit the match early." + ], + [ + "rate", + "Player's preferred network rate" + ], + [ + "RemoteCameraSounds_DryLevel", + "" + ], + [ + "RemoteCameraSounds_RoomType", + "" + ], + [ + "RemoteCameraSounds_WetLevel", + "" + ], + [ + "requireOpenNat", + "" + ], + [ + "restrictMapPacksToGroups", + "Restrict map pack usage to needing all maps in an ala carte package in order to use as search criteria" + ], + [ + "riotshield_bullet_damage_scale", + "Value to scale bullet damage to deployed riotshield." + ], + [ + "riotshield_deploy_limit_radius", + "Min distance deployed riotshields must be from each other." + ], + [ + "riotshield_deploy_trace_parallel", + "Report collisions when riotshield traces are parallel to plane of triangle. If disabled traces parallel to triangle planes do not report collisions at all." + ], + [ + "riotshield_deployed_health", + "Deployed riotshield health." + ], + [ + "riotshield_destroyed_cleanup_time", + "Time (in seconds) destroyed riotshield model persists before disappearing" + ], + [ + "riotshield_explosive_damage_scale", + "Value to scale explosive damage to deployed riotshield.." + ], + [ + "riotshield_melee_damage_scale", + "Value to scale melee damage to deployed riotshield." + ], + [ + "riotshield_projectile_damage_scale", + "Value to scale projectile damage to deployed riotshield." + ], + [ + "s_aggregate_ping_offset", + "offset to apply to aggregate ping values" + ], + [ + "s_aggregate_ping_scale", + "8-bit fixed-point aggregate ping scaler value" + ], + [ + "s_avg_max_weighting", + "weighting from 0-256 of party average ping vs. worst ping" + ], + [ + "s_ds_pingclient_reping_wait_db", + "wait this# of frames for the db thread to settle down before repinging" + ], + [ + "s_use_aggregate_datacenter_pings", + "use newer system for aggregating party pings" + ], + [ + "safeArea_adjusted_horizontal", + "User-adjustable horizontal safe area as a fraction of the screen width" + ], + [ + "safeArea_adjusted_vertical", + "User-adjustable vertical safe area as a fraction of the screen height" + ], + [ + "safeArea_horizontal", + "Horizontal safe area as a fraction of the screen width" + ], + [ + "safeArea_vertical", + "Vertical safe area as a fraction of the screen height" + ], + [ + "scr_conf_numlives", + "" + ], + [ + "scr_conf_playerrespawndelay", + "" + ], + [ + "scr_conf_roundlimit", + "" + ], + [ + "scr_conf_scorelimit", + "" + ], + [ + "scr_conf_timelimit", + "" + ], + [ + "scr_conf_waverespawndelay", + "" + ], + [ + "scr_conf_winlimit", + "" + ], + [ + "scr_default_maxagents", + "" + ], + [ + "scr_diehard", + "" + ], + [ + "scr_disableClientSpawnTraces", + "" + ], + [ + "scr_dm_numlives", + "" + ], + [ + "scr_dm_playerrespawndelay", + "" + ], + [ + "scr_dm_roundlimit", + "" + ], + [ + "scr_dm_scorelimit", + "" + ], + [ + "scr_dm_timelimit", + "" + ], + [ + "scr_dm_waverespawndelay", + "" + ], + [ + "scr_dm_winlimit", + "" + ], + [ + "scr_dom_numlives", + "" + ], + [ + "scr_dom_playerrespawndelay", + "" + ], + [ + "scr_dom_roundlimit", + "" + ], + [ + "scr_dom_scorelimit", + "" + ], + [ + "scr_dom_timelimit", + "" + ], + [ + "scr_dom_waverespawndelay", + "" + ], + [ + "scr_dom_winlimit", + "" + ], + [ + "scr_explBulletMod", + "" + ], + [ + "scr_game_allowkillcam", + "script allow killcam" + ], + [ + "scr_game_deathpointloss", + "" + ], + [ + "scr_game_forceuav", + "" + ], + [ + "scr_game_graceperiod", + "" + ], + [ + "scr_game_hardpoints", + "" + ], + [ + "scr_game_killstreakdelay", + "" + ], + [ + "scr_game_lockspectatorpov", + "Lock spectator mode globally, 0=freelook/unlocked, 1=first_person, 2=third_person" + ], + [ + "scr_game_onlyheadshots", + "" + ], + [ + "scr_game_perks", + "" + ], + [ + "scr_game_spectatetype", + "" + ], + [ + "scr_game_suicidepointloss", + "" + ], + [ + "scr_gameended", + "" + ], + [ + "scr_hardcore", + "" + ], + [ + "scr_horde_difficulty", + "" + ], + [ + "scr_horde_maxagents", + "" + ], + [ + "scr_horde_numlives", + "" + ], + [ + "scr_horde_playerrespawndelay", + "" + ], + [ + "scr_horde_roundlimit", + "" + ], + [ + "scr_horde_scorelimit", + "" + ], + [ + "scr_horde_timelimit", + "" + ], + [ + "scr_horde_waverespawndelay", + "" + ], + [ + "scr_horde_winlimit", + "" + ], + [ + "scr_infect_numlives", + "" + ], + [ + "scr_infect_playerrespawndelay", + "" + ], + [ + "scr_infect_roundlimit", + "" + ], + [ + "scr_infect_timelimit", + "" + ], + [ + "scr_infect_waverespawndelay", + "" + ], + [ + "scr_infect_winlimit", + "" + ], + [ + "scr_isgamescom", + "script use gamescom demo flow" + ], + [ + "scr_maxPerPlayerExplosives", + "" + ], + [ + "scr_nukeCancelMode", + "" + ], + [ + "scr_nukeTimer", + "" + ], + [ + "scr_patientZero", + "" + ], + [ + "scr_player_forcerespawn", + "" + ], + [ + "scr_player_healthregentime", + "" + ], + [ + "scr_player_maxhealth", + "" + ], + [ + "scr_player_numlives", + "" + ], + [ + "scr_player_respawndelay", + "" + ], + [ + "scr_player_sprinttime", + "" + ], + [ + "scr_player_suicidespawndelay", + "" + ], + [ + "scr_RequiredMapAspectratio", + "" + ], + [ + "scr_riotShieldXPBullets", + "" + ], + [ + "scr_sd_bombtimer", + "" + ], + [ + "scr_sd_defusetime", + "" + ], + [ + "scr_sd_multibomb", + "" + ], + [ + "scr_sd_numlives", + "" + ], + [ + "scr_sd_planttime", + "" + ], + [ + "scr_sd_playerrespawndelay", + "" + ], + [ + "scr_sd_roundlimit", + "" + ], + [ + "scr_sd_roundswitch", + "" + ], + [ + "scr_sd_scorelimit", + "" + ], + [ + "scr_sd_timelimit", + "" + ], + [ + "scr_sd_waverespawndelay", + "" + ], + [ + "scr_sd_winlimit", + "" + ], + [ + "scr_sr_bombtimer", + "" + ], + [ + "scr_sr_defusetime", + "" + ], + [ + "scr_sr_multibomb", + "" + ], + [ + "scr_sr_numlives", + "" + ], + [ + "scr_sr_planttime", + "" + ], + [ + "scr_sr_playerrespawndelay", + "" + ], + [ + "scr_sr_roundlimit", + "" + ], + [ + "scr_sr_roundswitch", + "" + ], + [ + "scr_sr_scorelimit", + "" + ], + [ + "scr_sr_timelimit", + "" + ], + [ + "scr_sr_waverespawndelay", + "" + ], + [ + "scr_sr_winlimit", + "" + ], + [ + "scr_team_fftype", + "script team friendly fire type" + ], + [ + "scr_team_respawntime", + "" + ], + [ + "scr_team_teamkillpointloss", + "" + ], + [ + "scr_team_teamkillspawndelay", + "" + ], + [ + "scr_thirdPerson", + "" + ], + [ + "scr_tispawndelay", + "" + ], + [ + "scr_war_halftime", + "" + ], + [ + "scr_war_numlives", + "" + ], + [ + "scr_war_playerrespawndelay", + "" + ], + [ + "scr_war_roundlimit", + "" + ], + [ + "scr_war_roundswitch", + "" + ], + [ + "scr_war_scorelimit", + "" + ], + [ + "scr_war_timelimit", + "" + ], + [ + "scr_war_waverespawndelay", + "" + ], + [ + "scr_war_winlimit", + "" + ], + [ + "scr_xpscale", + "" + ], + [ + "screenshots_active", + "Are we allowed to enable Screenshots or not" + ], + [ + "search_weight_asn", + "The weight used for the asn in weighted matchmaking." + ], + [ + "search_weight_country_code", + "The weight used for the country code in weighted matchmaking." + ], + [ + "search_weight_lat_long", + "The weight used for the lat long in weighted matchmaking." + ], + [ + "sensitivity", + "Mouse sensitivity" + ], + [ + "sentry_placement_feet_offset", + "Position of the feet from the center axis." + ], + [ + "sentry_placement_feet_trace_dist_z", + "Max distance for a foot to be considered touching the ground" + ], + [ + "sentry_placement_trace_dist", + "Distance along the trace axis where the sentry will attempt to position itself" + ], + [ + "sentry_placement_trace_min_normal", + "Minimum normal to accept a sentry position" + ], + [ + "sentry_placement_trace_parallel", + "Enable turret traces that are parallel to plane of triangle. If 0, traces parallel to triangle planes do not report collisions at all. If 2 (debug-only), then trace code ping pongs between new and old." + ], + [ + "sentry_placement_trace_pitch", + "Pitch used for the trace axis" + ], + [ + "sentry_placement_trace_radius", + "Radius of the bound used for the placement trace" + ], + [ + "sentry_placement_trace_radius_canon_safety", + "Extra radius used in the forward direction to compensate for the canon length" + ], + [ + "server1", + "" + ], + [ + "server10", + "" + ], + [ + "server11", + "" + ], + [ + "server12", + "" + ], + [ + "server13", + "" + ], + [ + "server14", + "" + ], + [ + "server15", + "" + ], + [ + "server16", + "" + ], + [ + "server2", + "" + ], + [ + "server3", + "" + ], + [ + "server4", + "" + ], + [ + "server5", + "" + ], + [ + "server6", + "" + ], + [ + "server7", + "" + ], + [ + "server8", + "" + ], + [ + "server9", + "" + ], + [ + "session_immediateDeleteTinySessions", + "Whether to immediately delete sessions with 1 user" + ], + [ + "session_modify_retry_on_failure", + "Enable session modify retry on failures." + ], + [ + "session_nonblocking", + "Non-blocking Session code" + ], + [ + "sf_use_bw", + "Intel Cheat - CoD Noir." + ], + [ + "sf_use_chaplin", + "Intel Cheat - Ragtime Warfare." + ], + [ + "sf_use_clustergrenade", + "Intel Cheat - Cluster Bombs." + ], + [ + "sf_use_contrast", + "Intel Cheat - Super Contrast." + ], + [ + "sf_use_ignoreammo", + "Intel Cheat - Infinite Ammo." + ], + [ + "sf_use_invert", + "Intel Cheat - Photo-Negative." + ], + [ + "sf_use_lemonade_mode", + "Intel Cheat - Lemon-nade." + ], + [ + "sf_use_melon_mode", + "Intel Cheat - Melon Heads." + ], + [ + "sf_use_ragdoll_mode", + "Intel Cheat - Ragdoll Impact." + ], + [ + "sf_use_slowmo", + "Intel Cheat - Slow-Mo Ability." + ], + [ + "sf_use_tire_explosion", + "Intel Cheat - A Bad Year." + ], + [ + "sf_use_tracksuit_mode", + "Intel Cheat - Zakhaev's Sons." + ], + [ + "shortversion", + "Short game version" + ], + [ + "showDebugAmmoCounter", + "Show the debug ammo counter when unable to show ar ammo counter" + ], + [ + "showPlaylistTotalPlayers", + "Toggle the display of the total number of players in a playlist and online" + ], + [ + "sm_cacheSpotShadows", + "Cache spot shadow maps, improves shadow map performance at the cost of memory (requires vid_restart)" + ], + [ + "sm_cacheSpotShadowsEnabled", + "Enables caching of spot shadows." + ], + [ + "sm_cacheSunShadow", + "Cache sun shadow map, improves shadow map performance at the cost of memory (requires vid_restart)" + ], + [ + "sm_cacheSunShadowEnabled", + "Enables caching of sun-based shadows." + ], + [ + "sm_cameraOffset", + "" + ], + [ + "sm_dynlightAllSModels", + "Enable, from script, rendering all static models in dynamic light volume when shadow mapping" + ], + [ + "sm_enable", + "Enable shadow mapping" + ], + [ + "sm_fastSunShadow", + "Fast sun shadow" + ], + [ + "sm_lightScore_eyeProjectDist", + "When picking shadows for primary lights, measure distance from a point this far in front of the camera." + ], + [ + "sm_lightScore_spotProjectFrac", + "When picking shadows for primary lights, measure distance from a point this far in front of the camera." + ], + [ + "sm_maxLightsWithShadows", + "Limits how many primary lights can have shadow maps" + ], + [ + "sm_minSpotLightScore", + "Minimum score (based on intensity, radius, and position relative to the camera) for a spot light to have shadow maps." + ], + [ + "sm_polygonOffsetBias", + "Shadow map offset bias" + ], + [ + "sm_polygonOffsetClamp", + "Shadow map offset clamp" + ], + [ + "sm_polygonOffsetPreset", + "Shadow map polygon offset preset." + ], + [ + "sm_polygonOffsetScale", + "Shadow map offset scale" + ], + [ + "sm_qualitySpotShadow", + "Fast spot shadow" + ], + [ + "sm_shadowUseTweaks", + "Override shadow LightSet settings with tweak dvar values." + ], + [ + "sm_spotDistCull", + "Distance cull spot shadows" + ], + [ + "sm_spotEnable", + "Enable spot shadow mapping from script" + ], + [ + "sm_spotFilterRadius", + "Spot soft shadows filter radius" + ], + [ + "sm_spotLightScoreModelScale", + "Scale the calculated spot light score by this value if the light currently only affects static or script brush models." + ], + [ + "sm_spotLightScoreRadiusPower", + "Power to apply to light radius when determining spot light shadowing score (1.0 means radius scales up score a lot, 0.0 means don't scale score using radius)" + ], + [ + "sm_spotLimit", + "Limit number of spot shadows from script" + ], + [ + "sm_spotShadowFadeTime", + "How many seconds it takes for a primary light shadow map to fade in or out" + ], + [ + "sm_strictCull", + "Strict shadow map cull" + ], + [ + "sm_sunEnable", + "Enable sun shadow mapping from script" + ], + [ + "sm_sunFilterRadius", + "Sun soft shadows filter radius" + ], + [ + "sm_sunSampleSizeNear", + "Shadow sample size" + ], + [ + "sm_sunShadowBoundsMax", + "Max Shadow Bounds" + ], + [ + "sm_sunShadowBoundsMin", + "Min Shadow Bounds" + ], + [ + "sm_sunShadowBoundsOverride", + "Override Shadow Bounds" + ], + [ + "sm_sunShadowCenter", + "Sun shadow center, 0 0 0 means don't override" + ], + [ + "sm_sunShadowCenterMode", + "When false center value only used for far map, when true sets both maps" + ], + [ + "sm_sunShadowScale", + "Sun shadow scale optimization" + ], + [ + "sm_sunShadowScaleLocked", + "Lock usage of sm_sunShadowScale at 1" + ], + [ + "sm_usedSunCascadeCount", + "How many shadow cascade we are using" + ], + [ + "snd_allowHeadphoneHRTF", + "Enable HRTF over headphones" + ], + [ + "snd_announcerDisabled", + "Disable all in-game announcers" + ], + [ + "snd_announcerVoicePrefix", + "Local mp announcer voice to use" + ], + [ + "snd_battlechatterDisabled", + "Disable all in-game battle chatter" + ], + [ + "snd_cinematicVolumeScale", + "Scales the volume of Bink videos." + ], + [ + "snd_detectedSpeakerConfig", + "speaker configuration:\n0: autodetect\n1: mono\n2: stereo\n4: quadrophonic\n6: 5.1 surround\n8: 7.1 surround" + ], + [ + "snd_dopplerAuditionEnable", + "Enables doppler calculation preview mode" + ], + [ + "snd_dopplerBaseSpeedOfSound", + "The base speed of sound used in doppler calculation" + ], + [ + "snd_dopplerEnable", + "Enables doppler calculation" + ], + [ + "snd_dopplerPitchMax", + "Maximum pitch that can be legally applied by doppler" + ], + [ + "snd_dopplerPitchMin", + "Minimum pitch that can be legally applied by doppler" + ], + [ + "snd_dopplerPlayerVelocityScale", + "The scale of the player velocity, relative the the sound source velocity, when applied to the doppler calculation" + ], + [ + "snd_dopplerSmoothing", + "Smoothing factor applied to doppler to eliminate jitter or sudden acceleration changes" + ], + [ + "snd_draw3D", + "Draw the position and info of world sounds" + ], + [ + "snd_drawInfo", + "Draw debugging information for sounds" + ], + [ + "snd_enable2D", + "Enable 2D sounds" + ], + [ + "snd_enable3D", + "Enable 3D sounds" + ], + [ + "snd_enableEq", + "Enable equalization filter" + ], + [ + "snd_enableReverb", + "Enable sound reverberation" + ], + [ + "snd_enableStream", + "Enable streamed sounds" + ], + [ + "snd_envFollowerBuffScale", + "Amount of buffer to use for envelope follower. Smaller value indicates faster envelope." + ], + [ + "snd_errorOnMissing", + "Cause a Com_Error if a sound file is missing." + ], + [ + "snd_hitsoundDisabled", + "Disable the hit indicator sound" + ], + [ + "snd_inheritSecondaryPitchVol", + "Set to true for secondary aliases to inherit the pitch of the parent" + ], + [ + "snd_levelFadeTime", + "The amout of time in milliseconds for all audio to fade in at the start of a level" + ], + [ + "snd_loadFadeTime", + "Fade time for loading from a checkpoint after death." + ], + [ + "snd_loopFadeTime", + "Fade-in time for looping sounds." + ], + [ + "snd_musicDisabled", + "Disable all in-game music" + ], + [ + "snd_musicDisabledForCustomSoundtrack", + "Disable all in-game music due to user playing a custom soundtrack" + ], + [ + "snd_occlusionDelay", + "Minimum delay in (ms) between occlusion updates" + ], + [ + "snd_occlusionLerpTime", + "Time to lerp to target occlusion lerp when occluded" + ], + [ + "snd_peakLimiterCompression", + "Peak limiter compression factor. The output data is scaled by this and then normalized: F < 1 = disabled; F >= 1 enabled." + ], + [ + "snd_peakLimiterDecay", + "Peak limiter compression decay ratio." + ], + [ + "snd_peakLimiterSustainFrames", + "Number of frames to sustain the limiter peak. 1 frame = 10 msec." + ], + [ + "snd_premixVolume", + "Game sound pre-mix volume" + ], + [ + "snd_reverbZoneOutsideFactor", + "When a 3d sound is played in a different reverb zone than the player, this factor will be applied to its wet level." + ], + [ + "snd_slaveFadeTime", + "The amount of time in milliseconds for a 'slave' sound\nto fade its volumes when a master sound starts or stops" + ], + [ + "snd_speakerConfig", + "speaker configuration:\n0: autodetect\n1: mono\n2: stereo\n4: quadrophonic\n6: 5.1 surround\n8: 7.1 surround" + ], + [ + "snd_touchStreamFilesOnLoad", + "Check whether stream sound files exist while loading" + ], + [ + "snd_useOldPanning", + "Use old and busted panning" + ], + [ + "snd_virtualChannelInfo", + "Display virtual voice info." + ], + [ + "snd_virtualMinDur", + "The minimum duration (in seconds) of a sound if it is to be added to the virtual voice buffer." + ], + [ + "snd_virtualMinPri", + "The minimum priority of an alias if it is to be added to the virtual voice buffer." + ], + [ + "snd_virtualMinTimeLeftToRevive", + "The minimum time (in ms) left in a sample in order to attempt to revive it." + ], + [ + "snd_virtualReviveVoices", + "Whether or not to restore virtual voices." + ], + [ + "snd_virtualWaitToReviveTime", + "The minimum time (in ms) to wait before trying to revive the voice." + ], + [ + "snd_volume", + "Game sound master volume" + ], + [ + "speech_active", + "Are we allowed to enable Speech or not" + ], + [ + "splitscreen", + "Current game is a splitscreen game" + ], + [ + "steam_ingame_p2p_throttle", + "Time, in MS, to wait between P2P packet lookups when in-game" + ], + [ + "stringtable_debug", + "spam debug info for stringtable lookups" + ], + [ + "sv_allowClientConsole", + "Allow remote clients to access the console" + ], + [ + "sv_allowedClan1", + "" + ], + [ + "sv_allowedClan2", + "" + ], + [ + "sv_archiveClientsPositions", + "Archive the client positions to speed up SV_GetClientPositionsAtTime" + ], + [ + "sv_checkMinPlayers", + "Check min players. 0 disables" + ], + [ + "sv_clientArchive", + "Have the clients archive data to save bandwidth on the server" + ], + [ + "sv_connectTimeout", + "seconds without any message when a client is loading" + ], + [ + "sv_cumulThinkTime", + "Max client think per server 50 msec frame" + ], + [ + "sv_error_on_baseline_failure", + "Throw an error if the const baseline data is invalid." + ], + [ + "sv_exponentialBackoffAfterNonAckedMsgs", + "start exponential backoff on msg frequency if the client has not acked the last X messages" + ], + [ + "sv_hostname", + "Host name of the server" + ], + [ + "sv_hugeSnapshotDelay", + "How long to wait before building a new snapshot after a 'huge' snapshot is sent" + ], + [ + "sv_hugeSnapshotSize", + "Size of a snapshot to be considered 'huge'" + ], + [ + "sv_kickBanTime", + "Time in seconds for a player to be banned from the server after being kicked" + ], + [ + "sv_local_client_snapshot_msec", + "Local client snapshot rate, add to cl_penaltyTime" + ], + [ + "sv_maxclients", + "The maximum number of clients that can connect to a server" + ], + [ + "sv_minPingClamp", + "Clamp the minimum ping to this value" + ], + [ + "sv_network_fps", + "Number of times per second the server checks for net messages" + ], + [ + "sv_numExpBackoffBeforeReleasingCachedSnapshots", + "if a client is under an exponential backoff over this dvar, then we will release all the cached snapshot data he owns and will send him a baseline if he reconnects" + ], + [ + "sv_paused", + "Pause the server" + ], + [ + "sv_privateClients", + "Maximum number of private clients allowed on the server" + ], + [ + "sv_privateClientsForClients", + "The # of private clients (we send this to clients)" + ], + [ + "sv_privatePassword", + "password for the privateClient slots" + ], + [ + "sv_reconnectlimit", + "minimum seconds between connect messages" + ], + [ + "sv_rejoinTimeout", + "seconds without any message before allowing a rejoin" + ], + [ + "sv_remote_client_snapshot_joiningstate_msec", + "Remote client snapshot rate during join (until the client acked his first delta message)" + ], + [ + "sv_remote_client_snapshot_msec", + "Remote client snapshot rate, add to cl_penaltyTime" + ], + [ + "sv_resetOnSpawn", + "Have clients reset some player state fields when spawning rather than sending them over the network" + ], + [ + "sv_running", + "Server is running" + ], + [ + "sv_sayName", + "" + ], + [ + "sv_showAverageBPS", + "Show average bytes per second for net debugging" + ], + [ + "sv_testValue", + "Max antilag rewind" + ], + [ + "sv_timeout", + "seconds without any message" + ], + [ + "sv_trackFrameMsecThreshold", + "server frame time that will trigger script time tracking." + ], + [ + "sv_useExtraCompress", + "Use zlib compress for gamestate/baseline/score packets" + ], + [ + "sv_zlib_threshold", + "Message size threshold which triggers more aggressive compression" + ], + [ + "sv_zombietime", + "seconds to sync messages after disconnect" + ], + [ + "svwp", + "playerdata server write protection: 0 = disable, 1 = silent, 2 = kick" + ], + [ + "syncTimeTimeout", + "default timeout for sync time task (in seconds)" + ], + [ + "sys_configSum", + "Configuration checksum" + ], + [ + "sys_configureGHz", + "Normalized total CPU power, based on cpu type, count, and speed; used in autoconfigure" + ], + [ + "sys_cpuGHz", + "Measured CPU speed" + ], + [ + "sys_cpuName", + "CPU name description" + ], + [ + "sys_gpu", + "GPU description" + ], + [ + "sys_lockThreads", + "Prevents specified threads from changing CPUs; improves profiling and may fix some bugs, but can hurt performance" + ], + [ + "sys_quitMigrateTime", + "Time in msec to wait for host migration when user closes the window" + ], + [ + "sys_smp_allowed", + "Allow multi-threading" + ], + [ + "sys_SSE", + "Operating system allows Streaming SIMD Extensions" + ], + [ + "sys_sysMB", + "Physical memory in the system" + ], + [ + "systemlink", + "Current game is a system link game" + ], + [ + "systemlink_host", + "Local client is hosting system link game" + ], + [ + "tb_report", + "tb event record" + ], + [ + "team_rebalance", + "rebalance" + ], + [ + "teambalance_option", + "Selects active teambalance algorithm. 0 = heuristic 1 = exhaustive" + ], + [ + "theater_active", + "Are we allowed to show theater or not." + ], + [ + "thermal_playerModel", + "Model to draw for players when in thermal vision mode" + ], + [ + "thermalBlurFactorNoScope", + "Amount of blur to use when drawing blur through a weapon's thermal scope." + ], + [ + "thermalBlurFactorScope", + "Amount of blur to use when drawing blur through a weapon's thermal scope." + ], + [ + "timescale", + "Set the game speed." + ], + [ + "tokensEnabled", + "Is token economy enabled" + ], + [ + "tracer_explosiveColor1", + "The 1st color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_explosiveColor2", + "The 2nd color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_explosiveColor3", + "The 3rd color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_explosiveColor4", + "The 4th color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_explosiveColor5", + "The 5th color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_explosiveOverride", + "When turned on, will apply an override to the tracer settings when shooting explosive bullets." + ], + [ + "tracer_explosiveWidth", + "The width of a bullet tracer when using explosive bullets" + ], + [ + "tracer_firstPersonMaxWidth", + "The maximum width our OWN tracers can be when looking through our ADS" + ], + [ + "tracer_stoppingPowerColor1", + "The 1st color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_stoppingPowerColor2", + "The 2nd color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_stoppingPowerColor3", + "The 3rd color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_stoppingPowerColor4", + "The 4th color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_stoppingPowerColor5", + "The 5th color of a bullet tracer when using explosive bullets" + ], + [ + "tracer_stoppingPowerOverride", + "When turned on, will apply an override to the tracer settings when shooting explosive bullets." + ], + [ + "tracer_stoppingPowerWidth", + "The width of a bullet tracer when using explosive bullets" + ], + [ + "tracer_thermalWidthMult", + "The multiplier applied to the base width when viewed in thermal vision" + ], + [ + "transients_verbose", + "Verbose logging information for transient fastfiles." + ], + [ + "triggerDLCEnumerationOnSocialConfigLoad", + "Triggers a new DLC enumeration after social config has loaded." + ], + [ + "ui_allow_controlschange", + "" + ], + [ + "ui_allow_teamchange", + "" + ], + [ + "ui_autodetectGamepad", + "undefined" + ], + [ + "ui_autodetectGamepadDone", + "undefined" + ], + [ + "ui_bigFont", + "Big font scale" + ], + [ + "ui_blurAmount", + "Max amount to blur background menu items." + ], + [ + "ui_blurDarkenAmount", + "Amount to darken blurred UI." + ], + [ + "ui_blurTime", + "Time in milliseconds to fade in/out the blur." + ], + [ + "ui_borderLowLightScale", + "Scales the border color for the lowlight color on certain UI borders" + ], + [ + "ui_browserFriendlyfire", + "Friendly fire is active" + ], + [ + "ui_browserKillcam", + "Kill cam is active" + ], + [ + "ui_browserMod", + "UI Mod value" + ], + [ + "ui_browserShowDedicated", + "Show dedicated servers only" + ], + [ + "ui_browserShowEmpty", + "Show empty servers" + ], + [ + "ui_browserShowFull", + "Show full servers" + ], + [ + "ui_browserShowPassword", + "Show servers that are password protected" + ], + [ + "ui_browserShowPure", + "Show pure servers only" + ], + [ + "ui_buildLocation", + "Where to draw the build number" + ], + [ + "ui_buildSize", + "Font size to use for the build number" + ], + [ + "ui_challenge_1_ref", + "" + ], + [ + "ui_challenge_2_ref", + "" + ], + [ + "ui_challenge_3_ref", + "" + ], + [ + "ui_challenge_4_ref", + "" + ], + [ + "ui_challenge_5_ref", + "" + ], + [ + "ui_challenge_6_ref", + "" + ], + [ + "ui_challenge_7_ref", + "" + ], + [ + "ui_changeclass_menu_open", + "" + ], + [ + "ui_changeteam_menu_open", + "" + ], + [ + "ui_cinematicsTimestamp", + "Shows cinematics timestamp on subtitle UI elements." + ], + [ + "ui_class_menu_open", + "" + ], + [ + "ui_connectScreenTextGlowColor", + "Glow color applied to the mode and map name strings on the connect screen." + ], + [ + "ui_contextualMenuLocation", + "Contextual menu location from where you entered the store." + ], + [ + "ui_controls_menu_open", + "" + ], + [ + "ui_currentFeederMapIndex", + "Currently selected map" + ], + [ + "ui_currentMap", + "Current map index" + ], + [ + "ui_customClassName", + "Custom Class name" + ], + [ + "ui_customModeEditName", + "Name to give the currently edited custom game mode when editing is complete" + ], + [ + "ui_customModeName", + "Custom game mode name" + ], + [ + "ui_danger_team", + "" + ], + [ + "ui_debugMode", + "Draw ui debug info on the screen." + ], + [ + "ui_disableInGameStore", + "This will disable the ingame store button on the xbox live menu." + ], + [ + "ui_disableTokenRedemption", + "This will disable the token redemption option in the in-game store menu." + ], + [ + "ui_drawCrosshair", + "Whether to draw crosshairs." + ], + [ + "ui_editSquadMemberIndex", + "Which squad member is currently being edited" + ], + [ + "ui_extraBigFont", + "Extra big font scale" + ], + [ + "ui_game_state", + "" + ], + [ + "ui_gametype", + "Current game type" + ], + [ + "ui_halftime", + "" + ], + [ + "ui_hitloc_0", + "" + ], + [ + "ui_hitloc_1", + "" + ], + [ + "ui_hitloc_2", + "" + ], + [ + "ui_hitloc_3", + "" + ], + [ + "ui_hitloc_4", + "" + ], + [ + "ui_hitloc_5", + "" + ], + [ + "ui_hitloc_damage_0", + "" + ], + [ + "ui_hitloc_damage_1", + "" + ], + [ + "ui_hitloc_damage_2", + "" + ], + [ + "ui_hitloc_damage_3", + "" + ], + [ + "ui_hitloc_damage_4", + "" + ], + [ + "ui_hitloc_damage_5", + "" + ], + [ + "ui_hud_hardcore", + "Whether the HUD should be suppressed for hardcore mode" + ], + [ + "ui_hud_obituaries", + "" + ], + [ + "ui_inactiveBaseColor", + "The local player's rank/stats font color when shown in lobbies and parties" + ], + [ + "ui_inactivePartyColor", + "" + ], + [ + "ui_inGameStoreOpen", + "is the InGameStore open" + ], + [ + "ui_inhostmigration", + "" + ], + [ + "ui_joinGametype", + "Game join type" + ], + [ + "ui_loadMenuName", + "Frontend menu will start on this level instead of lockout" + ], + [ + "ui_mapname", + "Current map name" + ], + [ + "ui_mapvote_entrya_gametype", + "Primary map vote entry game type" + ], + [ + "ui_mapvote_entrya_mapname", + "Primary map vote entry map name" + ], + [ + "ui_mapvote_entryb_gametype", + "Secondary map vote entry game type" + ], + [ + "ui_mapvote_entryb_mapname", + "Secondary map vote entry map name" + ], + [ + "ui_maxclients", + "undefined" + ], + [ + "ui_missingMapName", + "Name of map to show in missing content error" + ], + [ + "ui_mousePitch", + "" + ], + [ + "ui_multiplayer", + "True if the game is multiplayer" + ], + [ + "ui_myPartyColor", + "Player name font color when in the same party as the local player" + ], + [ + "ui_netGametype", + "Game type" + ], + [ + "ui_netGametypeName", + "Displayed game type name" + ], + [ + "ui_netSource", + "The network source where:\n 0:Local\n 1:Internet\n 2:Favourites" + ], + [ + "ui_onlineRequired", + "UI requires online connection to be present." + ], + [ + "ui_opensummary", + "" + ], + [ + "ui_override_halftime", + "" + ], + [ + "ui_partyFull", + "True if the current party is full." + ], + [ + "ui_playerPartyColor", + "" + ], + [ + "ui_playlistActionButtonAlpha", + "The current alpha of the playlist selection button" + ], + [ + "ui_playlistCategoryDisabledColor", + "The color of playlist categories when disabled" + ], + [ + "ui_playlistCategoryEnabledColor", + "The color of playlist categories when enabled" + ], + [ + "ui_promotion", + "" + ], + [ + "ui_remoteTankUseTime", + "" + ], + [ + "ui_scorelimit", + "" + ], + [ + "ui_selectedFeederMap", + "Current preview game type" + ], + [ + "ui_serverStatusTimeOut", + "Time in milliseconds before a server status request times out" + ], + [ + "ui_show_store", + "Use to enable the store button" + ], + [ + "ui_showDLCMaps", + "Whether to display the DLC maps." + ], + [ + "ui_showInfo", + "" + ], + [ + "ui_showList", + "Show onscreen list of currently visible menus" + ], + [ + "ui_showmap", + "" + ], + [ + "ui_showMenuOnly", + "If set, only menus using this name will draw." + ], + [ + "ui_showMinimap", + "" + ], + [ + "ui_sliderSteps", + "The number of steps for a slider itemdef" + ], + [ + "ui_smallFont", + "Small font scale" + ], + [ + "ui_textScrollFadeTime", + "Text scrolling takes this long (seconds) to fade out at the end before restarting" + ], + [ + "ui_textScrollPauseEnd", + "Text scrolling waits this long (seconds) before starting" + ], + [ + "ui_textScrollPauseStart", + "Text scrolling waits this long (seconds) before starting" + ], + [ + "ui_textScrollSpeed", + "Speed at which text scrolls vertically" + ], + [ + "ui_timelimit", + "" + ], + [ + "uiscript_debug", + "spam debug info for the ui script" + ], + [ + "unlock_breadcrumb_killswitch", + "True to enable unlock breadcrumbs" + ], + [ + "uno_current_tos_version", + "Current Uno Terms of Service Version" + ], + [ + "use_filtered_query_pass", + "Dictates whether to use the filtered query for MMing or not" + ], + [ + "use_weighted_dlc_exactmatch_pass", + "Dictates whether to use a search weighted pass with the DLC match set to exact for MMing or not" + ], + [ + "use_weighted_pass", + "Dictates whether to use the search weighted pass for MMing or not" + ], + [ + "useCPMarkerForCPOwnership", + "If set, we will check the player inventory to see if he owns the redeemedItem for a contentPack if this contentPack is not available for the player" + ], + [ + "useonlinestats", + "Whether to use online stats when in offline modes" + ], + [ + "useRelativeTeamColors", + "Whether to use relative team colors." + ], + [ + "userFileFetchTimeout", + "default timeout for user files FETCH tasks (in seconds)" + ], + [ + "userGroup_active", + "Are we allowed to show Usergroups or not" + ], + [ + "userGroup_cool_off_time", + "Cool off time between calls to fetch the elite clan" + ], + [ + "userGroup_coop_delay", + "Delay between a player joining a coop lobby and the DW user group task starting" + ], + [ + "userGroup_max_retry_time", + "Max time that the usergroup read find can retry" + ], + [ + "userGroup_refresh_time_secs", + "Time in seconds between re-sending lobby group data to confirmed users." + ], + [ + "userGroup_retry_step", + "Step in m/s for the usegroup read retry" + ], + [ + "userGroup_RetryTime", + "Time in ms between sending lobby group data retrys." + ], + [ + "useStatsGroups", + "If true then StatsGroups are in use for all playerdata.ddl accessing." + ], + [ + "useTagFlashSilenced", + "When true, silenced weapons will use \"tag_flash_silenced\" instead of \"tag_flash\"." + ], + [ + "using_mlg", + "MLG feature on/off" + ], + [ + "validate_apply_clamps", + "True if individual stat validation failure reverts the value" + ], + [ + "validate_apply_revert", + "True if individual stat validation failure reverts the value" + ], + [ + "validate_apply_revert_full", + "True if any individual stat validation failure causes a full stats revert" + ], + [ + "validate_clamp_assists", + "The maximum number of assists a player can make in a match" + ], + [ + "validate_clamp_experience", + "The maximum experience a player can gain in a match" + ], + [ + "validate_clamp_headshots", + "The maximum number of headshots a player can make in a match" + ], + [ + "validate_clamp_hits", + "The maximum number of hits player can make in a match" + ], + [ + "validate_clamp_kills", + "The maximum number of kills a player can make in a match" + ], + [ + "validate_clamp_losses", + "The maximum number of losses a player can make in a match" + ], + [ + "validate_clamp_misses", + "The maximum number of misses player can make in a match" + ], + [ + "validate_clamp_ties", + "The maximum number of ties a player can make in a match" + ], + [ + "validate_clamp_totalshots", + "The maximum number of totalshots player can make in a match" + ], + [ + "validate_clamp_weaponXP", + "The maximum experience a weapon can gain in a match" + ], + [ + "validate_clamp_wins", + "The maximum number of wins a player can make in a match" + ], + [ + "validate_drop_on_fail", + "True if stats validation failure results in dropping from the match" + ], + [ + "veh_aiOverSteerScale", + "Scaler used to cause ai vehicles to over steer" + ], + [ + "veh_boneControllerLodDist", + "Distance at which bone controllers are not updated." + ], + [ + "veh_boneControllerUnLodDist", + "Distance at which bone controllers start updating when not moving." + ], + [ + "vehAudio_inAirPitchDownLerp", + "Rate at which the pitch lerps down" + ], + [ + "vehAudio_inAirPitchUpLerp", + "Rate at which the pitch lerps up" + ], + [ + "vehAudio_spawnVolumeTime", + "Seconds it takes for spawned vehicles to reach full volume." + ], + [ + "vehCam_freeLook", + "Enables free look mode" + ], + [ + "vehCam_mode", + "Camera modes: 1st person, 3rd person, or both" + ], + [ + "vehDroneDebugDrawPath", + "Debug render the drone draw paths." + ], + [ + "vehHelicopterBoundsRadius", + "The radius of the collision volume to be used when colliding with world geometry." + ], + [ + "vehHelicopterDecelerationFwd", + "Set the deceleration of the player helicopter (as a fraction of acceleration) in the direction the chopper is facing. So 1.0 makes it equal to the acceleration." + ], + [ + "vehHelicopterDecelerationSide", + "Set the side-to-side deceleration of the player helicopter (as a fraction of acceleration). So 1.0 makes it equal to the acceleration." + ], + [ + "vehHelicopterDecelerationUp", + "Set the vertical deceleration of the player helicopter (as a fraction of acceleration). So 1.0 makes it equal to the acceleration." + ], + [ + "vehHelicopterHeadSwayDontSwayTheTurret", + "If set, the turret will not fire through the crosshairs, but straight ahead of the vehicle, when the player is not freelooking." + ], + [ + "vehHelicopterHoverSpeedThreshold", + "The speed below which the player helicopter begins to jitter the tilt, for hovering" + ], + [ + "vehHelicopterInvertUpDown", + "Invert the altitude control on the player helicopter." + ], + [ + "vehHelicopterJitterJerkyness", + "Specifies how jerky the tilt jitter should be" + ], + [ + "vehHelicopterLookaheadTime", + "How far ahead (in seconds) the player helicopter looks ahead, to avoid hard collisions. (Like driving down the highway, you should keep 2 seconds distance between you and the vehicle in front of you)" + ], + [ + "vehHelicopterMaxAccel", + "Maximum horizontal acceleration of the player helicopter (in MPH per second)" + ], + [ + "vehHelicopterMaxAccelVertical", + "Maximum vertical acceleration of the player helicopter (in MPH per second)" + ], + [ + "vehHelicopterMaxPitch", + "Maximum pitch of the player helicopter" + ], + [ + "vehHelicopterMaxRoll", + "Maximum roll of the player helicopter" + ], + [ + "vehHelicopterMaxSpeed", + "Maximum horizontal speed of the player helicopter (in MPH)" + ], + [ + "vehHelicopterMaxSpeedVertical", + "Maximum vertical speed of the player helicopter (in MPH)" + ], + [ + "vehHelicopterMaxYawAccel", + "Maximum yaw acceleration of the player helicopter" + ], + [ + "vehHelicopterMaxYawRate", + "Maximum yaw speed of the player helicopter" + ], + [ + "vehHelicopterPitchOffset", + "The resting pitch of the helicopter" + ], + [ + "vehHelicopterRightStickDeadzone", + "Dead-zone for the axes of the right thumbstick. This helps to better control the two axes separately." + ], + [ + "vehHelicopterScaleMovement", + "Scales down the smaller of the left stick axes." + ], + [ + "vehHelicopterSoftCollisions", + "Player helicopters have soft collisions (slow down before they collide)." + ], + [ + "vehHelicopterStrafeDeadzone", + "Dead-zone so that you can fly straight forward easily without accidentally strafing (and thus rolling)." + ], + [ + "vehHelicopterTiltFromAcceleration", + "The amount of tilt caused by acceleration" + ], + [ + "vehHelicopterTiltFromControllerAxes", + "The amount of tilt caused by the desired velocity (i.e., the amount of controller stick deflection)" + ], + [ + "vehHelicopterTiltFromDeceleration", + "The amount of tilt caused by deceleration" + ], + [ + "vehHelicopterTiltFromFwdAndYaw", + "The amount of roll caused by yawing while moving forward." + ], + [ + "vehHelicopterTiltFromFwdAndYaw_VelAtMaxTilt", + "The forward speed (as a fraction of top speed) at which the tilt due to yaw reaches is maximum value." + ], + [ + "vehHelicopterTiltFromVelocity", + "The amount of tilt caused by the current velocity" + ], + [ + "vehHelicopterTiltMomentum", + "The amount of rotational momentum the helicopter has with regards to tilting." + ], + [ + "vehHelicopterTiltSpeed", + "The rate at which the player helicopter's tilt responds" + ], + [ + "vehHelicopterYawOnLeftStick", + "The yaw speed created by the left stick when pushing the stick diagonally (e.g., moving forward and strafing slightly)." + ], + [ + "vehicle_debug_render_spline_plane", + "Do we want to render the spline plane data" + ], + [ + "vehicle_pathsmooth", + "Smoothed vehicle pathing." + ], + [ + "vehUGVPitchTrack", + "UGV body pitch orientation speed" + ], + [ + "vehUGVRollTrack", + "UGV body roll orientation speed" + ], + [ + "vehUGVWheelInfluence", + "UGV wheel influence on the orientation of the body" + ], + [ + "vehWalkerControlMode", + "Walker controls (0==move no turn, 1=move and turn, 2=move relative(tank))" + ], + [ + "version", + "Game version" + ], + [ + "vid_xpos", + "Game window horizontal position" + ], + [ + "vid_ypos", + "Game window vertical position" + ], + [ + "viewangNow", + "" + ], + [ + "viewModelDebugNotetracks", + "Enable display of viewmodel notetrack debug info." + ], + [ + "viewModelHacks", + "Enabled depth hack and remove viewmodel from shadows." + ], + [ + "viewposNow", + "" + ], + [ + "virtualLobbyActive", + "Indicates the VL is actively being displayed." + ], + [ + "virtualLobbyAllocated", + "Indicates the first VL zone has been loaded." + ], + [ + "virtualLobbyEnabled", + "VirtualLobby is enabled (must be true before loading UI zone)" + ], + [ + "virtualLobbyInFiringRange", + "VirtualLobby is in firing range mode" + ], + [ + "virtualLobbyMap", + "VirtualLobby map to load (must be set before starting vl)" + ], + [ + "virtualLobbyMembers", + "Number of members in the VirtualLobby (set by script)" + ], + [ + "virtualLobbyPresentable", + "Indicates to LUA the VirtualLobby is ready to be displayed (set by script)." + ], + [ + "virtualLobbyReady", + "Indicates to LUA the VirtualLobby is loaded and running (set by script)." + ], + [ + "vl_clan_models_loaded", + "Indicates to LUA when all models are loaded for the clan highlights so it can begin the fade-in without any popping(set by script)." + ], + [ + "voMtxEnable", + "When set (e.g. via config), will enable voice over packs" + ], + [ + "waypointAerialIconMaxSize", + "Max size of aerial targeting waypoints." + ], + [ + "waypointAerialIconMinSize", + "Min size of aerial targeting waypoints." + ], + [ + "waypointAerialIconScale", + "Base scale of aerial targeting waypoints." + ], + [ + "waypointDebugDraw", + "" + ], + [ + "waypointDistScaleRangeMax", + "Distance from player that icon distance scaling ends." + ], + [ + "waypointDistScaleRangeMin", + "Distance from player that icon distance scaling ends." + ], + [ + "waypointDistScaleSmallest", + "Smallest scale that the distance effect uses." + ], + [ + "waypointIconHeight", + "" + ], + [ + "waypointIconWidth", + "" + ], + [ + "waypointOffscreenCornerRadius", + "Size of the rounded corners." + ], + [ + "waypointOffscreenDistanceThresholdAlpha", + "Distance from the threshold over which offscreen objective icons lerp their alpha." + ], + [ + "waypointOffscreenPadBottom", + "" + ], + [ + "waypointOffscreenPadLeft", + "" + ], + [ + "waypointOffscreenPadRight", + "" + ], + [ + "waypointOffscreenPadTop", + "" + ], + [ + "waypointOffscreenPointerDistance", + "Distance from the center of the offscreen objective icon to the center its arrow." + ], + [ + "waypointOffscreenPointerHeight", + "" + ], + [ + "waypointOffscreenPointerWidth", + "" + ], + [ + "waypointOffscreenRoundedCorners", + "Off-screen icons take rounded corners when true. 90-degree corners when false." + ], + [ + "waypointOffscreenScaleLength", + "How far the offscreen icon scale travels from full to smallest scale." + ], + [ + "waypointOffscreenScaleSmallest", + "Smallest scale that the offscreen effect uses." + ], + [ + "waypointPlayerOffsetCrouch", + "For waypoints pointing to players, how high to offset off of their origin when they are prone." + ], + [ + "waypointPlayerOffsetProne", + "For waypoints pointing to players, how high to offset off of their origin when they are prone." + ], + [ + "waypointPlayerOffsetStand", + "For waypoints pointing to players, how high to offset off of their origin when they are prone." + ], + [ + "waypointScreenCenterFadeAdsMin", + "When 'waypointScreenCenterFadeRadius' enabled, minimum amount that waypoint will fade when in ads" + ], + [ + "waypointScreenCenterFadeHipMin", + "When 'waypointScreenCenterFadeRadius' enabled, minimum amount that waypoint will fade when in ads" + ], + [ + "waypointScreenCenterFadeRadius", + "Radius from screen center that a waypoint will start fading out. Setting to 0 will turn this off" + ], + [ + "waypointSplitscreenScale", + "Scale applied to waypoint icons in splitscreen views." + ], + [ + "waypointTweakY", + "" + ], + [ + "weap_thermoDebuffMod", + "" + ], + [ + "wideScreen", + "True if the game video is running in 16x9 aspect, false if 4x3." + ], + [ + "winvoice_loopback", + "Echo microphone input locally" + ], + [ + "winvoice_mic_mute", + "Mute the microphone" + ], + [ + "winvoice_mic_outTime", + "Microphone voice amount of silence before we cut the mic" + ], + [ + "winvoice_mic_reclevel", + "Microphone recording level" + ], + [ + "winvoice_mic_scaler", + "Microphone scaler value" + ], + [ + "winvoice_mic_threshold", + "Microphone voice threshold" + ], + [ + "winvoice_save_voice", + "Write voice data to a file" + ], + [ + "xanim_disableIK", + "Disable inverse kinematics solvers" + ], + [ + "xblive_competitionmatch", + "MLG Rules?" + ], + [ + "xblive_hostingprivateparty", + "true only if we're hosting a party" + ], + [ + "xblive_loggedin", + "User is logged into xbox live" + ], + [ + "xblive_privatematch", + "Current game is a private match" + ], + [ + "xblive_privatematch_solo", + "Current game is an Extinction solo match" + ], + [ + "xphys_maxJointPositionError", + "If a joints with position error exceeding this value is detected, then the whole xphys system gets snapped back to the animation pose" + ], + [ + "aa_player_kills", + "Player kills" + ], + [ + "aa_player_damage_dealt", + "Player damages dealt" + ], + [ + "aa_ads_damage_dealt", + "Player damages dealt in ads" + ], + [ + "aa_deaths", + "Player deaths" + ], + [ + "aa_time_tracking", + "Time in game" + ] +] diff --git a/src/client/resources/icon.ico b/src/client/resources/icon.ico index b6c8943d..ba668330 100644 Binary files a/src/client/resources/icon.ico and b/src/client/resources/icon.ico differ diff --git a/src/client/resources/icon.png b/src/client/resources/icon.png index c7f4980a..625055ba 100644 Binary files a/src/client/resources/icon.png and b/src/client/resources/icon.png differ diff --git a/src/client/resources/main.html b/src/client/resources/main.html index 8faf0f5c..6e87228b 100644 --- a/src/client/resources/main.html +++ b/src/client/resources/main.html @@ -12,6 +12,7 @@ -ms-user-select: none; cursor: default; box-sizing: border-box; + line-height: 130%; } h5 { @@ -21,16 +22,16 @@ html, body { + overflow: hidden; -ms-overflow-style: none; margin: 0; - background: url('') no-repeat center center fixed; + background: url('') no-repeat center center fixed; background-size: cover; background-position: center; background-repeat: no-repeat; + background-color: #202020; color: white; - font-family: "Segoe UI Light", "Segoe UI", "Lucida Sans", Arial, sans-serif; - font-style: normal; - font-weight: lighter; + font-family: "HModMix-Medium", "Segoe UI Light", "Segoe UI", "Lucida Sans", Arial, sans-serif; display: flex; min-height: 100vh; flex-direction: column; @@ -44,10 +45,9 @@ } p { - font-weight: 200; - font-size: 1.4em; + font-size: 3.1vw; color: rgb(255, 255, 255); - text-shadow: 1px 1px #000000; + text-shadow: 1px 1px 3px #000000, 0px 0 20px #000000; } a { @@ -60,14 +60,14 @@ } p a { - font-weight: bold; - color: #236b2f; + color: #66e391; text-decoration: none; - text-shadow: 2px 2px #000000; + text-shadow: 2px 1px 1px #000000, 1px 1px 3px #000000, 0 0 13px #000000; } p a:hover { - color: #00ff4c; + color: #a1fe65; + cursor: pointer; } nav { @@ -107,7 +107,7 @@ nav .nav-link { display: inline-block; - font-size: 1.3em; + font-size: 2.8vw; color: white; text-transform: uppercase; padding: 0 10px; @@ -115,15 +115,25 @@ padding: 10px 0px; margin-right: 15px; margin-bottom: -1px; + text-shadow: none; } nav .nav-link.active { border-bottom: 1px solid white; } + nav .nav-link:hover { + cursor: pointer; + } + nav .nav-link:hover, nav .nav-link.active { - text-shadow: 0px 0px 8px rgba(255, 255, 255, 0.748); + text-shadow: 0px 0px 10px rgba(255, 255, 255, 0.5); + } + + nav .nav-link.active { + border-bottom: 1px solid white; + cursor: default; } nav .nav-link:first-child { @@ -196,12 +206,10 @@ margin-left: 5px; margin-top: 0; letter-spacing: 1px; - font-weight: 200; } .container h3 { margin-bottom: 0px; - font-weight: 500; } .full-row .card { @@ -242,7 +250,7 @@ flex-direction: column; overflow: hidden; border: 1px solid #454a4d; - border-radius: 10px; + border-radius: 8px; box-shadow: 0px 0px 5px 0px rgba(0, 0, 0, 0.5); transition: all 0.5s ease; } @@ -253,7 +261,6 @@ .card .card-img { overflow: hidden; - height: 185px; } .card img { @@ -263,6 +270,7 @@ object-fit: cover; z-index: 1; transition: all 0.5s ease; + cursor: pointer; } .card.full .card-img { @@ -286,15 +294,16 @@ .card:hover .card-content { background-color: rgba(255, 255, 255, 0.321); color: rgb(255, 255, 255); - text-shadow: 1px 1px #000000; + text-shadow: 1px 1px 3px #000000; + cursor: pointer; } .card .card-content span { - font-size: 1.7em; - font-weight: 500; + font-size: 3.8vw; text-transform: uppercase; - letter-spacing: 1px; - text-shadow: 2px 2px #000000; + letter-spacing: 2px; + text-shadow: 0.3px 0 0 currentColor, 0 0.3px 0 currentColor, 2px 2px 4px #0a320a, 0px 0px 12px #317e37, 0px 1px 2px #003a02; + cursor: pointer; } .card .card-content p { @@ -305,6 +314,7 @@ opacity: 1; bottom: 0; border-color: rgba(0, 0, 0, 0.25); + cursor: pointer; } .card .card-action { @@ -319,8 +329,8 @@ text-transform: uppercase; padding: 7px; box-sizing: border-box; - font-weight: 500; transition: opacity bottom 0.5s; + font-size: 2.2vw; } .card .card-action.hidden-action { @@ -339,7 +349,6 @@ background-color: rgba(0, 0, 0, 0.2); border: 1px solid rgba(255, 255, 255, 0.42); color: white; - font-weight: 600; padding: 10px 100px; border-radius: 1px; transition: all 0.5s ease; @@ -391,6 +400,14 @@ background: linear-gradient(145deg, rgba(121, 121, 121, 1) 0%, rgba(0, 212, 255, 1) 100%); transition: all 0.2s linear; } + + @font-face { + font-family: 'HModMix-Medium'; + src: url('data:font/woff;charset=utf-8;base64,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') format('woff'); + font-weight: normal; + font-style: normal; + font-display: swap; + } @@ -420,7 +437,7 @@
+ src="" />
Singleplayer @@ -439,7 +456,7 @@
+ src="" />
Multiplayer @@ -466,22 +483,22 @@
-
+

Check out H1-Mod official links:
- Discord, - Website, - Docs, + Discord, + Website, + Docs, GitHub

This project is based on work from S1x and H2-Mod. + onClick="window.external.openUrl('https://github.com/fedddddd/h2-mod');">H2-Mod

diff --git a/src/client/resources/ui_scripts/updater.lua b/src/client/resources/ui_scripts/updater.lua index ac1b556a..2cb0761b 100644 --- a/src/client/resources/ui_scripts/updater.lua +++ b/src/client/resources/ui_scripts/updater.lua @@ -28,7 +28,7 @@ function startupdatecheck(popup, autoclose) LUI.yesnopopup({ title = Engine.Localize("@MENU_NOTICE"), - text = Engine.Localize("UPDATER_POPUP_AVAILABLE_UPDATE_TEXT"), + text = Engine.Localize("@MENU_CCS_NEW_PATCH_NOTICE") .. " " .. Engine.Localize("@MENU_DOWNLOAD_AUTOUPDATE_PATCH"), callback = function(result) if (result) then startupdatedownload(popup, autoclose) @@ -75,7 +75,7 @@ function startupdatedownload(popup, autoclose) if (updater.isrestartrequired()) then LUI.confirmationpopup({ title = Engine.Localize("@MENU_CCS_RESTART_CONFIRMATION_TITLE"), - text = Engine.Localize("UPDATER_POPUP_RESTART_POPUP_TEXT"), + text = Engine.Localize("@MENU_CCS_RESTART_CONFIRMATION_TEXT"), buttontext = Engine.Localize("@MENU_CCS_RESTART_BUTTON_LABEL"), callback = function() updater.relaunch() diff --git a/src/client/std_include.hpp b/src/client/std_include.hpp index 02d361c2..fe3276ab 100644 --- a/src/client/std_include.hpp +++ b/src/client/std_include.hpp @@ -48,6 +48,8 @@ #include #include #include +#include +#include // min and max is required by gdi, therefore NOMINMAX won't work #ifdef max @@ -83,12 +85,17 @@ #include #include #include +#include #include #include #include #include + +#pragma warning(push) +#pragma warning(disable: 4459) #include +#pragma warning(pop) #define RAPIDJSON_NOEXCEPT #define RAPIDJSON_ASSERT(cond) if(cond); else throw std::runtime_error("rapidjson assert fail"); diff --git a/src/client/utils/hash.cpp b/src/client/utils/hash.cpp new file mode 100644 index 00000000..2995c796 --- /dev/null +++ b/src/client/utils/hash.cpp @@ -0,0 +1,157 @@ +#include + +#include "game/game.hpp" + +#include "hash.hpp" +#include "component/console.hpp" + +#include +#include +#include + +namespace utils::hash +{ + namespace + { + constexpr auto read_buffer_size = 1ull * 1024ull * 1024ull; // 1MB + + std::string get_file_hash_pakfile(std::ifstream& file_stream, const std::size_t file_size, + const std::string& filename) + { + if (file_size < sizeof(game::XPakHeader)) + { + return {}; + } + + game::XPakHeader header{}; + file_stream.read(reinterpret_cast(&header), sizeof(game::XPakHeader)); + + constexpr auto hash_size = sizeof(game::DB_AuthHash); + + game::DB_AuthHash empty_hash{}; + if (!std::memcmp(header.hash.bytes, empty_hash.bytes, hash_size)) + { + console::warn("Computing pakfile hash because its missing, this may take some time...\n"); + + hash_state state; + sha256_init(&state); + + std::string buffer; + buffer.resize(read_buffer_size); + + auto bytes_to_read = file_size - sizeof(game::XPakHeader); + while (bytes_to_read > 0) + { + const auto read_size = std::min(read_buffer_size, bytes_to_read); + file_stream.read(buffer.data(), read_size); + sha256_process(&state, reinterpret_cast(buffer.data()), static_cast(read_size)); + bytes_to_read -= read_size; + } + + file_stream.close(); + + if (sha256_done(&state, header.hash.bytes) != CRYPT_OK) + { + return {}; + } + + std::fstream out_stream; + out_stream.open(filename, std::ios_base::binary | std::ios_base::out | std::ios_base::in); + out_stream.write(reinterpret_cast(&header), sizeof(game::XPakHeader)); + } + + const auto hash_str = std::string{header.hash.bytes, header.hash.bytes + hash_size}; + + return utils::string::dump_hex(hash_str, ""); + } + + std::string get_file_hash_generic(std::ifstream& file_stream, const std::size_t file_size) + { + auto crc_value = crc32(0L, Z_NULL, 0); + auto bytes_to_read = file_size; + + std::string buffer; + buffer.resize(read_buffer_size); + + while (bytes_to_read > 0) + { + const auto read_size = std::min(bytes_to_read, read_buffer_size); + file_stream.read(buffer.data(), read_size); + crc_value = crc32(crc_value, reinterpret_cast(buffer.data()), static_cast(read_size)); + bytes_to_read -= read_size; + } + + std::string hash; + hash.append(reinterpret_cast(&crc_value), sizeof(crc_value)); + return utils::string::dump_hex(hash, ""); + } + + std::string get_pakfile_buffer_hash(std::string& buffer) + { + if (buffer.size() < sizeof(game::XPakHeader)) + { + return {}; + } + + constexpr auto hash_size = sizeof(game::DB_AuthHash); + const auto header = reinterpret_cast(buffer.data()); + + game::DB_AuthHash empty_hash{}; + if (!std::memcmp(header->hash.bytes, empty_hash.bytes, hash_size)) + { + console::warn("Computing pakfile hash because its missing, this may take some time...\n"); + const auto hash_start = reinterpret_cast(buffer.data() + sizeof(game::XPakHeader)); + const auto len = buffer.size() - sizeof(game::XPakHeader); + const auto hash = utils::cryptography::sha256::compute(hash_start, len, false); + std::memcpy(header->hash.bytes, hash.data(), sizeof(header->hash)); + } + + std::string hash = {header->hash.bytes, header->hash.bytes + hash_size}; + return utils::string::dump_hex(hash, ""); + } + + std::string get_generic_buffer_hash(const std::string& buffer) + { + auto crc_value = crc32(0L, Z_NULL, 0); + crc_value = crc32(crc_value, reinterpret_cast(buffer.data()), + static_cast(buffer.size())); + std::string hash; + hash.append(reinterpret_cast(&crc_value), sizeof(crc_value)); + return utils::string::dump_hex(hash, ""); + } + } + + std::string get_file_hash(const std::string& file) + { + std::ifstream file_stream(file, std::ios::binary); + if (!file_stream.is_open()) + { + return {}; + } + + file_stream.seekg(0, std::ios::end); + const auto file_size = static_cast(file_stream.tellg()); + file_stream.seekg(0, std::ios::beg); + + if (file.ends_with(".pak")) + { + return get_file_hash_pakfile(file_stream, file_size, file); + } + else + { + return get_file_hash_generic(file_stream, file_size); + } + } + + std::string get_buffer_hash(std::string& buffer, const std::string& filename) + { + if (filename.ends_with(".pak")) + { + return get_pakfile_buffer_hash(buffer); + } + else + { + return get_generic_buffer_hash(buffer); + } + } +} diff --git a/src/client/utils/hash.hpp b/src/client/utils/hash.hpp new file mode 100644 index 00000000..4e5c9e56 --- /dev/null +++ b/src/client/utils/hash.hpp @@ -0,0 +1,7 @@ +#pragma once + +namespace utils::hash +{ + std::string get_file_hash(const std::string& file); + std::string get_buffer_hash(std::string& buffer, const std::string& filename); +} diff --git a/src/common/utils/flags.cpp b/src/common/utils/flags.cpp index a90247ac..539b6640 100644 --- a/src/common/utils/flags.cpp +++ b/src/common/utils/flags.cpp @@ -6,42 +6,110 @@ namespace utils::flags { - void parse_flags(std::vector& flags) + namespace { - int num_args; - auto* const argv = CommandLineToArgvW(GetCommandLineW(), &num_args); + bool parsed = false; - flags.clear(); + using flag_map_t = std::unordered_map>; - if (argv) + flag_map_t& get_flags() { - for (auto i = 0; i < num_args; ++i) - { - std::wstring wide_flag(argv[i]); - if (wide_flag[0] == L'-') - { - wide_flag.erase(wide_flag.begin()); - const auto flag = string::convert(wide_flag); - flags.emplace_back(string::to_lower(flag)); - } - } + static flag_map_t map = {}; + return map; + } - LocalFree(argv); + void parse_flags(flag_map_t& flags) + { + int num_args; + auto* const argv = CommandLineToArgvW(GetCommandLineW(), &num_args); + + flags.clear(); + + if (argv) + { + std::optional last_flag{}; + for (auto i = 0; i < num_args; ++i) + { + std::wstring wide_flag(argv[i]); + if (wide_flag[0] == L'-') + { + wide_flag.erase(wide_flag.begin()); + const auto flag = string::convert(wide_flag); + + last_flag = flag; + flags[flag] = {}; + } + else if (last_flag.has_value()) + { + const auto& flag = last_flag.value(); + + flags[flag] = string::convert(wide_flag); + last_flag = {}; + } + } + + LocalFree(argv); + } + } + + void check_parse_flags() + { + if (!parsed) + { + parse_flags(get_flags()); + parsed = true; + } } } bool has_flag(const std::string& flag) { - static auto parsed = false; - static std::vector enabled_flags; + check_parse_flags(); - if (!parsed) + for (const auto& [name, value] : get_flags()) { - parse_flags(enabled_flags); - parsed = true; + if (string::to_lower(name) == string::to_lower(flag)) + { + return true; + } } - return std::ranges::any_of(enabled_flags.cbegin(), enabled_flags.cend(), - [flag](const auto& elem) { return elem == string::to_lower(flag); }); + return false; + } + + std::optional get_flag(const std::string& flag) + { + check_parse_flags(); + + for (const auto& [name, value] : get_flags()) + { + if (string::to_lower(name) == string::to_lower(flag)) + { + return value; + } + } + + return {}; + } + + std::optional get_flag(const std::string& flag, const std::string& shortname) + { + auto value = get_flag(flag); + if (!value.has_value()) + { + value = get_flag(shortname); + } + return value; + } + + std::string get_flag(const std::string& flag, const std::string& shortname, + const std::string& default_) + { + const auto value = get_flag(flag, shortname); + if (!value.has_value()) + { + return default_; + } + return value.value(); } } diff --git a/src/common/utils/flags.hpp b/src/common/utils/flags.hpp index cf304b20..c38b076b 100644 --- a/src/common/utils/flags.hpp +++ b/src/common/utils/flags.hpp @@ -1,8 +1,13 @@ #pragma once #include +#include namespace utils::flags { bool has_flag(const std::string& flag); + std::optional get_flag(const std::string& flag); + std::optional get_flag(const std::string& flag, const std::string& shortname); + std::string get_flag(const std::string& flag, const std::string& shortname, + const std::string& default_); } diff --git a/src/common/utils/hook.cpp b/src/common/utils/hook.cpp index be05cb96..5e6bc9bb 100644 --- a/src/common/utils/hook.cpp +++ b/src/common/utils/hook.cpp @@ -3,6 +3,13 @@ #include +Mem seg_ptr(const SReg& segment, const uint64_t off) +{ + auto mem = ptr_abs(off); + mem.setSegment(segment); + return mem; +} + namespace utils::hook { namespace @@ -307,4 +314,31 @@ namespace utils::hook return extract(data + 1); } + + uint8_t* allocate_somewhere_near(const void* base_address, const size_t size) + { + size_t offset = 0; + while (true) + { + offset += size; + auto* target_address = static_cast(base_address) - offset; + if (utils::hook::is_relatively_far(base_address, target_address)) + { + return nullptr; + } + + const auto res = VirtualAlloc(const_cast(target_address), size, MEM_RESERVE | MEM_COMMIT, + PAGE_EXECUTE_READWRITE); + if (res) + { + if (utils::hook::is_relatively_far(base_address, target_address)) + { + VirtualFree(res, 0, MEM_RELEASE); + return nullptr; + } + + return static_cast(res); + } + } + } } diff --git a/src/common/utils/hook.hpp b/src/common/utils/hook.hpp index bb24f8ce..911b183d 100644 --- a/src/common/utils/hook.hpp +++ b/src/common/utils/hook.hpp @@ -1,11 +1,14 @@ #pragma once #include "signature.hpp" +#include "memory.hpp" #include #include using namespace asmjit::x86; +Mem seg_ptr(const SReg& segment, const uint64_t off); + namespace utils::hook { namespace detail @@ -202,4 +205,61 @@ namespace utils::hook { return static_cast(func)(args...); } + + uint8_t* allocate_somewhere_near(const void* base_address, const size_t size); + + template + void* allocate_far_jump() + { + constexpr auto alloc_size = 0x1000; + constexpr auto far_jmp_size = 0xC; + + const auto alloc_jump_table = [] + { + return reinterpret_cast( + memory::allocate_near(Base, alloc_size, PAGE_EXECUTE_READWRITE)); + }; + + static auto jump_table = alloc_jump_table(); + static auto current_pos = jump_table; + + if (current_pos + far_jmp_size >= jump_table + alloc_size) + { + jump_table = alloc_jump_table(); + current_pos = jump_table; + } + + const auto ptr = current_pos; + current_pos += far_jmp_size; + return ptr; + } + + template + void* create_far_jump(const T dest) + { + static std::unordered_map allocated_jumps; + if (const auto iter = allocated_jumps.find(reinterpret_cast(dest)); iter != allocated_jumps.end()) + { + return iter->second; + } + + const auto pos = allocate_far_jump(); + jump(pos, dest, true); + allocated_jumps.insert(std::make_pair(dest, pos)); + return pos; + } + + template + void far_jump(const size_t address, const T dest) + { + const auto pos = create_far_jump(dest); + jump(address, pos, false); + } + + template + void far_call(const size_t address, const T dest) + { + const auto pos = create_far_jump(dest); + call(address, pos); + } } diff --git a/src/common/utils/http.cpp b/src/common/utils/http.cpp index d7afc2d3..8c0753f4 100644 --- a/src/common/utils/http.cpp +++ b/src/common/utils/http.cpp @@ -43,7 +43,7 @@ namespace utils::http } std::optional get_data(const std::string& url, const std::string& fields, - const headers& headers, const std::function& callback) + const headers& headers, const std::function& callback, int timeout) { curl_slist* header_list = nullptr; auto* curl = curl_easy_init(); @@ -77,6 +77,8 @@ namespace utils::http curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1); curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0); + curl_easy_setopt(curl, CURLOPT_TIMEOUT, timeout); + if (!fields.empty()) { curl_easy_setopt(curl, CURLOPT_POSTFIELDS, fields.data()); diff --git a/src/common/utils/http.hpp b/src/common/utils/http.hpp index d0fdbc2f..0207ff22 100644 --- a/src/common/utils/http.hpp +++ b/src/common/utils/http.hpp @@ -19,7 +19,7 @@ namespace utils::http using headers = std::unordered_map; std::optional get_data(const std::string& url, const std::string& fields = {}, - const headers& headers = {}, const std::function& callback = {}); + const headers& headers = {}, const std::function& callback = {}, int timeout = 0); std::future> get_data_async(const std::string& url, const std::string& fields = {}, const headers& headers = {}, const std::function& callback = {}); } diff --git a/src/common/utils/info_string.cpp b/src/common/utils/info_string.cpp index 56ed56ab..229d0ef8 100644 --- a/src/common/utils/info_string.cpp +++ b/src/common/utils/info_string.cpp @@ -36,25 +36,25 @@ namespace utils buffer = buffer.substr(1); } - auto key_values = string::split(buffer, '\\'); + const auto key_values = string::split(buffer, '\\'); for (size_t i = 0; !key_values.empty() && i < (key_values.size() - 1); i += 2) { const auto& key = key_values[i]; const auto& value = key_values[i + 1]; - this->key_value_pairs_[key] = value; + + if (!this->key_value_pairs_.contains(key)) + { + this->key_value_pairs_[key] = value; + } } } std::string info_string::build() const { - //auto first = true; std::string info_string; for (const auto& [key, val] : this->key_value_pairs_) { - //if (first) first = false; - /*else*/ info_string.append("\\"); - info_string.append(key); info_string.append("\\"); info_string.append(val); diff --git a/src/common/utils/memory.cpp b/src/common/utils/memory.cpp index 12b36e70..9f079814 100644 --- a/src/common/utils/memory.cpp +++ b/src/common/utils/memory.cpp @@ -43,7 +43,7 @@ namespace utils { std::lock_guard _(this->mutex_); - const auto data = memory::allocate(length); + auto* data = memory::allocate(length); this->pool_.push_back(data); return data; } @@ -57,7 +57,7 @@ namespace utils { std::lock_guard _(this->mutex_); - const auto data = memory::duplicate_string(string); + auto* data = memory::duplicate_string(string); this->pool_.push_back(data); return data; } @@ -166,6 +166,33 @@ namespace utils return false; } + void* memory::allocate_near(const size_t address, const size_t size, const std::uint32_t protect) + { + SYSTEM_INFO system_info{}; + GetSystemInfo(&system_info); + + const auto page_size = system_info.dwPageSize; + const auto aligned_size = size + (~size & (page_size - 1)); + auto current_address = address; + + while (true) + { + current_address -= page_size; + + if (current_address <= reinterpret_cast(system_info.lpMinimumApplicationAddress)) + { + return nullptr; + } + + const auto result = VirtualAlloc(reinterpret_cast(current_address), aligned_size, MEM_RESERVE | MEM_COMMIT, protect); + if (result != nullptr) + { + std::memset(result, 0, aligned_size); + return result; + } + } + } + memory::allocator* memory::get_allocator() { return &memory::mem_allocator_; diff --git a/src/common/utils/memory.hpp b/src/common/utils/memory.hpp index 01f9554f..9887e308 100644 --- a/src/common/utils/memory.hpp +++ b/src/common/utils/memory.hpp @@ -69,6 +69,8 @@ namespace utils static bool is_bad_code_ptr(const void* ptr); static bool is_rdata_ptr(void* ptr); + static void* allocate_near(const size_t address, const size_t size, const std::uint32_t protect); + static allocator* get_allocator(); private: diff --git a/src/common/utils/nt.cpp b/src/common/utils/nt.cpp index 4cead0fa..6b7e4f11 100644 --- a/src/common/utils/nt.cpp +++ b/src/common/utils/nt.cpp @@ -15,7 +15,8 @@ namespace utils::nt library library::get_by_address(void* address) { HMODULE handle = nullptr; - GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, static_cast(address), &handle); + GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, + static_cast(address), &handle); return library(handle); } @@ -117,29 +118,29 @@ namespace utils::nt { if (!this->is_valid()) return ""; - auto path = this->get_path(); - const auto pos = path.find_last_of("/\\"); - if (pos == std::string::npos) return path; + const auto path = this->get_path(); + const auto pos = path.generic_string().find_last_of("/\\"); + if (pos == std::string::npos) return path.generic_string(); - return path.substr(pos + 1); + return path.generic_string().substr(pos + 1); } - std::string library::get_path() const + std::filesystem::path library::get_path() const { - if (!this->is_valid()) return ""; + if (!this->is_valid()) return {}; - char name[MAX_PATH] = {0}; - GetModuleFileNameA(this->module_, name, sizeof name); + wchar_t name[MAX_PATH] = { 0 }; + GetModuleFileNameW(this->module_, name, MAX_PATH); - return name; + return { name }; } - std::string library::get_folder() const - { - if (!this->is_valid()) return ""; - const auto path = std::filesystem::path(this->get_path()); - return path.parent_path().generic_string(); + std::filesystem::path library::get_folder() const + { + if (!this->is_valid()) return {}; + + return this->get_path().parent_path().generic_string(); } void library::free() @@ -156,7 +157,12 @@ namespace utils::nt return this->module_; } - void** library::get_iat_entry(const std::string& module_name, const std::string& proc_name) const + void** library::get_iat_entry(const std::string& module_name, std::string proc_name) const + { + return this->get_iat_entry(module_name, proc_name.data()); + } + + void** library::get_iat_entry(const std::string& module_name, const char* proc_name) const { if (!this->is_valid()) return nullptr; @@ -166,7 +172,7 @@ namespace utils::nt auto* const target_function = other_module.get_proc(proc_name); if (!target_function) return nullptr; - auto* header = this->get_optional_header(); + const auto* header = this->get_optional_header(); if (!header) return nullptr; auto* import_descriptor = reinterpret_cast(this->get_ptr() + header->DataDirectory @@ -183,16 +189,22 @@ namespace utils::nt while (original_thunk_data->u1.AddressOfData) { - const size_t ordinal_number = original_thunk_data->u1.AddressOfData & 0xFFFFFFF; - - if (ordinal_number > 0xFFFF) continue; - - if (GetProcAddress(other_module.module_, reinterpret_cast(ordinal_number)) == - target_function) + if (thunk_data->u1.Function == reinterpret_cast(target_function)) { return reinterpret_cast(&thunk_data->u1.Function); } + const size_t ordinal_number = original_thunk_data->u1.AddressOfData & 0xFFFFFFF; + + if (ordinal_number <= 0xFFFF) + { + auto* proc = GetProcAddress(other_module.module_, reinterpret_cast(ordinal_number)); + if (reinterpret_cast(proc) == target_function) + { + return reinterpret_cast(&thunk_data->u1.Function); + } + } + ++original_thunk_data; ++thunk_data; } @@ -208,15 +220,13 @@ namespace utils::nt bool is_wine() { - static std::optional is_wine = {}; - - if (!is_wine.has_value()) + static const auto has_wine_export = []() -> bool { - const utils::nt::library ntdll("ntdll.dll"); - is_wine = ntdll.get_proc("wine_get_version") != nullptr; - } + const library ntdll("ntdll.dll"); + return ntdll.get_proc("wine_get_version"); + }(); - return is_wine.value(); + return has_wine_export; } void raise_hard_exception() @@ -265,8 +275,8 @@ namespace utils::nt } } - CreateProcessA(self.get_path().data(), command_line.data(), nullptr, nullptr, false, NULL, nullptr, current_dir, - &startup_info, &process_info); + CreateProcessA(self.get_path().generic_string().data(), command_line.data(), nullptr, nullptr, false, + CREATE_NEW_CONSOLE, nullptr, current_dir, &startup_info, &process_info); if (process_info.hThread && process_info.hThread != INVALID_HANDLE_VALUE) CloseHandle(process_info.hThread); if (process_info.hProcess && process_info.hProcess != INVALID_HANDLE_VALUE) CloseHandle(process_info.hProcess); @@ -275,5 +285,6 @@ namespace utils::nt void terminate(const uint32_t code) { TerminateProcess(GetCurrentProcess(), code); + _Exit(code); } } diff --git a/src/common/utils/nt.hpp b/src/common/utils/nt.hpp index c4f58afe..58dd6c73 100644 --- a/src/common/utils/nt.hpp +++ b/src/common/utils/nt.hpp @@ -40,23 +40,29 @@ namespace utils::nt operator HMODULE() const; void unprotect() const; - void* get_entry_point() const; - size_t get_relative_entry_point() const; + [[nodiscard]] void* get_entry_point() const; + [[nodiscard]] size_t get_relative_entry_point() const; - bool is_valid() const; - std::string get_name() const; - std::string get_path() const; - std::string get_folder() const; - std::uint8_t* get_ptr() const; + [[nodiscard]] bool is_valid() const; + [[nodiscard]] std::string get_name() const; + [[nodiscard]] std::filesystem::path get_path() const; + [[nodiscard]] std::filesystem::path get_folder() const; + [[nodiscard]] std::uint8_t* get_ptr() const; void free(); - HMODULE get_handle() const; + [[nodiscard]] HMODULE get_handle() const; template - T get_proc(const std::string& process) const + [[nodiscard]] T get_proc(const char* process) const { if (!this->is_valid()) T{}; - return reinterpret_cast(GetProcAddress(this->module_, process.data())); + return reinterpret_cast(GetProcAddress(this->module_, process)); + } + + template + [[nodiscard]] T get_proc(const std::string& process) const + { + return get_proc(process.data()); } template @@ -90,13 +96,14 @@ namespace utils::nt return T(); } - std::vector get_section_headers() const; + [[nodiscard]] std::vector get_section_headers() const; - PIMAGE_NT_HEADERS get_nt_headers() const; - PIMAGE_DOS_HEADER get_dos_header() const; - PIMAGE_OPTIONAL_HEADER get_optional_header() const; + [[nodiscard]] PIMAGE_NT_HEADERS get_nt_headers() const; + [[nodiscard]] PIMAGE_DOS_HEADER get_dos_header() const; + [[nodiscard]] PIMAGE_OPTIONAL_HEADER get_optional_header() const; - void** get_iat_entry(const std::string& module_name, const std::string& proc_name) const; + [[nodiscard]] void** get_iat_entry(const std::string& module_name, std::string proc_name) const; + [[nodiscard]] void** get_iat_entry(const std::string& module_name, const char* proc_name) const; private: HMODULE module_; diff --git a/src/common/utils/string.cpp b/src/common/utils/string.cpp index 30e35e9d..7da02312 100644 --- a/src/common/utils/string.cpp +++ b/src/common/utils/string.cpp @@ -131,6 +131,14 @@ namespace utils::string *out = '\0'; } + std::string strip(const std::string& string) + { + std::string new_string; + new_string.resize(string.size() + 1, 0); + strip(string.data(), new_string.data(), static_cast(new_string.size())); + return new_string.data(); + } + std::string convert(const std::wstring& wstr) { std::string result; diff --git a/src/common/utils/string.hpp b/src/common/utils/string.hpp index 13bcbbf8..fa1c696f 100644 --- a/src/common/utils/string.hpp +++ b/src/common/utils/string.hpp @@ -92,6 +92,7 @@ namespace utils::string std::string get_clipboard_data(); void strip(const char* in, char* out, int max); + std::string strip(const std::string& string); std::string convert(const std::wstring& wstr); std::wstring convert(const std::string& str);