84e80140a1ed144070121b10f0a93837de6d110f
[voretournament/voretournament.git] / data / qcsrc / client / View.qc
1 #define rkt_size 32\r
2 #define rld_size_x 256\r
3 #define rld_size_y 16\r
4 \r
5 entity porto;\r
6 float trace_networkentity;\r
7 float Q3SURFACEFLAG_SLICK = 2; // low friction surface\r
8 float DPCONTENTS_SOLID = 1; // blocks player movement\r
9 float DPCONTENTS_BODY = 32; // blocks player movement\r
10 float DPCONTENTS_CORPSE = 64; // blocks player movement\r
11 float DPCONTENTS_PLAYERCLIP = 256; // blocks player movement\r
12 void Porto_Draw()\r
13 {\r
14         vector dir;\r
15 \r
16         if(spectatee_status)\r
17                 return;\r
18         if(intermission == 1)\r
19                 return;\r
20         if(intermission == 2)\r
21                 return;\r
22         if (getstati(STAT_HEALTH) <= 0)\r
23                 return;\r
24 \r
25         dir = view_forward;\r
26 \r
27         if(angles_held_status)\r
28         {\r
29                 makevectors(angles_held);\r
30                 dir = v_forward;\r
31         }\r
32 }\r
33 \r
34 /**\r
35  * Checks whether the server initiated a map restart (stat_game_starttime changed)\r
36  *\r
37  * TODO: Use a better solution where a common shared entitiy is used that contains\r
38  * timelimit, fraglimit and game_starttime! Requires engine changes (remove STAT_TIMELIMIT\r
39  * and STAT_FRAGLIMIT to be auto-sent)\r
40  */\r
41 void CheckForGamestartChange() {\r
42         float startTime;\r
43         startTime = getstatf(STAT_GAMESTARTTIME);\r
44         if (previous_game_starttime != startTime) {\r
45                 if ((time + 5.0) < startTime) {\r
46                         //if connecting to server while restart was active don't always play prepareforbattle\r
47                         sound(self, CHAN_VOICE, strcat("announcer/", cvar_string("cl_announcer"), "/prepareforbattle.wav"), VOL_BASEVOICE, ATTN_NONE);\r
48                 }\r
49                 if (time < startTime) {\r
50                         restartAnnouncer = spawn();\r
51                         restartAnnouncer.think = restartAnnouncer_Think;\r
52                         restartAnnouncer.nextthink = startTime - floor(startTime - time); //synchronize nextthink to startTime\r
53                 }\r
54         }\r
55         previous_game_starttime = startTime;\r
56 }\r
57 \r
58 void Porto_Init()\r
59 {\r
60         porto = spawn();\r
61         porto.classname = "porto";\r
62         porto.draw = Porto_Draw;\r
63         porto.dphitcontentsmask = DPCONTENTS_SOLID | DPCONTENTS_BODY | DPCONTENTS_PLAYERCLIP;\r
64 }\r
65 \r
66 float drawtime;\r
67 float avgspeed;\r
68 vector GetCurrentFov(float fov)\r
69 {\r
70         float zoomsensitivity, zoomspeed, zoomfactor, zoomdir, velocityzoom;\r
71 \r
72         zoomsensitivity = cvar("cl_zoomsensitivity");\r
73         zoomfactor = cvar("cl_zoomfactor");\r
74         if(zoomfactor < 1 || zoomfactor > 16)\r
75                 zoomfactor = 2.5;\r
76         zoomspeed = cvar("cl_zoomspeed");\r
77         if(zoomspeed >= 0)\r
78                 if(zoomspeed < 0.5 || zoomspeed > 16)\r
79                         zoomspeed = 3.5;\r
80 \r
81         zoomdir = button_zoom;\r
82         if(spectatee_status > 0 || isdemo())\r
83         {\r
84                 if(spectatorbutton_zoom)\r
85                         zoomdir = 0 + !zoomdir;\r
86                 // do not even THINK about removing this 0\r
87                 // _I_ know what I am doing\r
88                 // fteqcc does not\r
89         }\r
90 \r
91         if(zoomdir)\r
92                 zoomin_effect = 0;\r
93 \r
94         if(zoomin_effect || camera_active)\r
95         {\r
96                 current_viewzoom = min(1, current_viewzoom + drawframetime);\r
97         }\r
98         else\r
99         {\r
100                 if(zoomspeed < 0) // instant zoom\r
101                 {\r
102                         if(zoomdir)\r
103                                 current_viewzoom = 1 / zoomfactor;\r
104                         else\r
105                                 current_viewzoom = 1;\r
106                 }\r
107                 else\r
108                 {\r
109                         if(zoomdir)\r
110                                 current_viewzoom = 1 / bound(1, 1 / current_viewzoom + drawframetime * zoomspeed * (zoomfactor - 1), zoomfactor);\r
111                         else\r
112                                 current_viewzoom = bound(1 / zoomfactor, current_viewzoom + drawframetime * zoomspeed * (1 - 1 / zoomfactor), 1);\r
113                 }\r
114         }\r
115 \r
116         if(almost_equals(current_viewzoom, 1))\r
117                 current_zoomfraction = 0;\r
118         else if(almost_equals(current_viewzoom, 1/zoomfactor))\r
119                 current_zoomfraction = 1;\r
120         else\r
121                 current_zoomfraction = (current_viewzoom - 1) / (1/zoomfactor - 1);\r
122 \r
123         if(zoomsensitivity < 1)\r
124                 setsensitivityscale(pow(current_viewzoom, 1 - zoomsensitivity));\r
125         else\r
126                 setsensitivityscale(1);\r
127 \r
128         velocityzoom = bound(0, drawframetime / max(0.000000001, cvar_or("cl_velocityzoomtime", 0.3)), 1);\r
129         avgspeed = avgspeed * (1 - velocityzoom) + (vlen(pmove_vel) / 1000) * velocityzoom;\r
130         velocityzoom = exp(float2range11(avgspeed * -cvar_or("cl_velocityzoom", 0) / 1) * 1);\r
131 \r
132         //print(ftos(avgspeed), " avgspeed, ", ftos(cvar_or("cl_velocityzoom", 0)), " cvar, ", ftos(velocityzoom), " return\n"); // for debugging\r
133 \r
134         float frustumx, frustumy, fovx, fovy;\r
135         frustumy = tan(fov * M_PI / 360.0) * 0.75 * current_viewzoom * velocityzoom;\r
136         frustumx = frustumy * vid_width / vid_height / vid_pixelheight;\r
137         fovx = atan2(frustumx, 1) / M_PI * 360.0;\r
138         fovy = atan2(frustumy, 1) / M_PI * 360.0;\r
139 \r
140         return '1 0 0' * fovx + '0 1 0' * fovy;\r
141 }\r
142 \r
143 // this function must match W_SetupShot!\r
144 float zoomscript_caught;\r
145 \r
146 vector wcross_origin;\r
147 float wcross_scale_prev, wcross_alpha_prev;\r
148 vector wcross_color_prev;\r
149 float wcross_scale_goal_prev, wcross_alpha_goal_prev;\r
150 vector wcross_color_goal_prev;\r
151 float wcross_changedonetime;\r
152 \r
153 string wcross_name_goal_prev, wcross_name_goal_prev_prev;\r
154 float wcross_resolution_goal_prev, wcross_resolution_goal_prev_prev;\r
155 float wcross_name_changestarttime, wcross_name_changedonetime;\r
156 float wcross_name_alpha_goal_prev, wcross_name_alpha_goal_prev_prev;\r
157 entity trueaim;\r
158 entity trueaim_rifle;\r
159 \r
160 #define SHOTTYPE_HITTEAM 1\r
161 #define SHOTTYPE_HITOBSTRUCTION 2\r
162 #define SHOTTYPE_HITWORLD 3\r
163 #define SHOTTYPE_HITENEMY 4\r
164 \r
165 void TrueAim_Init()\r
166 {\r
167         trueaim = spawn();\r
168         trueaim.classname = "trueaim";\r
169         trueaim.dphitcontentsmask = DPCONTENTS_SOLID | DPCONTENTS_BODY | DPCONTENTS_CORPSE;\r
170         trueaim_rifle = spawn();\r
171         trueaim_rifle.classname = "trueaim_rifle";\r
172         trueaim_rifle.dphitcontentsmask = DPCONTENTS_BODY | DPCONTENTS_CORPSE;\r
173 }\r
174 \r
175 float EnemyHitCheck()\r
176 {\r
177         float t;\r
178         wcross_origin = project_3d_to_2d(trace_endpos);\r
179         wcross_origin_z = 0;\r
180         if(trace_networkentity < 1)\r
181                 return SHOTTYPE_HITWORLD;\r
182         if(trace_networkentity > maxclients)\r
183                 return SHOTTYPE_HITWORLD;\r
184         t = GetPlayerColor(trace_networkentity - 1);\r
185         if(teamplay)\r
186                 if(t == myteam)\r
187                         return SHOTTYPE_HITTEAM;\r
188         if(t == COLOR_SPECTATOR)\r
189                 return SHOTTYPE_HITWORLD;\r
190         return SHOTTYPE_HITENEMY;\r
191 }\r
192 \r
193 float TrueAimCheck()\r
194 {\r
195         float nudge = 1; // added to traceline target and subtracted from result\r
196         vector vecs, trueaimpoint, w_shotorg;\r
197         vector mi, ma, dv;\r
198         float shottype;\r
199         entity ta;\r
200         float mv;\r
201 \r
202         mi = ma = '0 0 0';\r
203         ta = trueaim;\r
204         mv = MOVE_NOMONSTERS;\r
205 \r
206         switch(activeweapon)\r
207         {\r
208