Rev 18 | Go to most recent revision | Details | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
1 | pmbaty | 1 | #include "oil.h" |
2 | #include "brender/brender.h" |
||
3 | #include "finteray.h" |
||
4 | #include "globvars.h" |
||
5 | #include "globvrpb.h" |
||
6 | #include "harness/trace.h" |
||
7 | #include "loading.h" |
||
8 | #include "network.h" |
||
9 | #include "piping.h" |
||
10 | #include "utility.h" |
||
11 | #include <math.h> |
||
12 | #include <stdlib.h> |
||
13 | |||
14 | char* gOil_pixie_names[1] = { "OIL.PIX" }; |
||
15 | int gNext_oil_pixie = 0; |
||
16 | br_scalar gZ_buffer_diff; |
||
17 | br_scalar gMin_z_diff; |
||
18 | br_pixelmap* gOil_pixies[1]; |
||
19 | tOil_spill_info gOily_spills[15]; |
||
20 | |||
21 | // IDA: void __cdecl InitOilSpills() |
||
22 | void InitOilSpills(void) { |
||
23 | int i; |
||
24 | br_model* the_model; |
||
25 | br_material* the_material; |
||
26 | LOG_TRACE("()"); |
||
27 | |||
28 | for (i = 0; i < COUNT_OF(gOil_pixie_names); i++) { |
||
29 | gOil_pixies[i] = LoadPixelmap(gOil_pixie_names[i]); |
||
30 | BrMapAdd(gOil_pixies[i]); |
||
31 | } |
||
32 | |||
33 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
34 | the_material = BrMaterialAllocate(NULL); |
||
35 | BrMaterialAdd(the_material); |
||
36 | the_material->ka = 0.99f; |
||
37 | the_material->kd = 0.0f; |
||
38 | the_material->ks = 0.0f; |
||
39 | the_material->power = 0.0f; |
||
40 | the_material->index_base = 0; |
||
41 | the_material->flags |= BR_MATF_LIGHT; |
||
42 | the_material->flags |= BR_MATF_PERSPECTIVE; |
||
43 | the_material->flags |= BR_MATF_SMOOTH; |
||
44 | // TODO: added by dethrace, investigate why oil spills in OG do not need this flag set to render correctly |
||
45 | the_material->flags |= BR_MATF_TWO_SIDED; |
||
46 | the_material->index_range = 0; |
||
47 | the_material->colour_map = NULL; |
||
48 | BrMatrix23Identity(&the_material->map_transform); |
||
49 | the_material->index_shade = BrTableFind("IDENTITY.TAB"); |
||
50 | BrMaterialUpdate(the_material, BR_MATU_ALL); |
||
51 | the_model = BrModelAllocate(NULL, 4, 2); |
||
52 | the_model->flags |= BR_MODF_KEEP_ORIGINAL; |
||
53 | |||
54 | the_model->faces[0].vertices[0] = 2; |
||
55 | the_model->faces[0].vertices[1] = 1; |
||
56 | the_model->faces[0].vertices[2] = 0; |
||
57 | the_model->faces[0].material = NULL; |
||
58 | the_model->faces[0].smoothing = 1; |
||
59 | the_model->faces[1].vertices[0] = 3; |
||
60 | the_model->faces[1].vertices[1] = 2; |
||
61 | the_model->faces[1].vertices[2] = 0; |
||
62 | the_model->faces[1].material = NULL; |
||
63 | the_model->faces[1].smoothing = 1; |
||
64 | BrVector3Set(&the_model->vertices[0].p, -1.f, 0.f, -1.f); |
||
65 | BrVector2Set(&the_model->vertices[0].map, 0.f, 1.f); |
||
66 | BrVector3Set(&the_model->vertices[1].p, 1.f, 0.f, 1.f); |
||
67 | BrVector2Set(&the_model->vertices[1].map, 0.f, 0.f); |
||
68 | BrVector3Set(&the_model->vertices[2].p, 1.f, 0.f, -1.f); |
||
69 | BrVector2Set(&the_model->vertices[2].map, 1.f, 0.f); |
||
70 | BrVector3Set(&the_model->vertices[3].p, -1.f, 0.f, 1.f); |
||
71 | BrVector2Set(&the_model->vertices[3].map, 1.f, 1.f); |
||
72 | gOily_spills[i].actor = BrActorAllocate(BR_ACTOR_MODEL, NULL); |
||
73 | gOily_spills[i].actor->model = the_model; |
||
74 | gOily_spills[i].actor->render_style = BR_RSTYLE_NONE; |
||
75 | gOily_spills[i].actor->material = the_material; |
||
76 | BrActorAdd(gNon_track_actor, gOily_spills[i].actor); |
||
77 | } |
||
78 | } |
||
79 | |||
80 | // IDA: void __cdecl ResetOilSpills() |
||
81 | void ResetOilSpills(void) { |
||
82 | int i; |
||
83 | LOG_TRACE("()"); |
||
84 | |||
85 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
86 | gOily_spills[i].actor->render_style = BR_RSTYLE_NONE; |
||
87 | gOily_spills[i].car = NULL; |
||
88 | gOily_spills[i].stop_time = 0; |
||
89 | } |
||
90 | } |
||
91 | |||
92 | // IDA: void __usercall QueueOilSpill(tCar_spec *pCar@<EAX>) |
||
93 | void QueueOilSpill(tCar_spec* pCar) { |
||
94 | int i; |
||
95 | int oily_index; |
||
96 | int oldest_one; |
||
97 | tU32 the_time; |
||
98 | tU32 oldest_time; |
||
99 | LOG_TRACE("(%p)", pCar); |
||
100 | |||
101 | oldest_one = 0; |
||
102 | oily_index = -1; |
||
103 | the_time = GetTotalTime(); |
||
104 | oldest_time = GetTotalTime(); |
||
105 | |||
106 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
107 | if (gOily_spills[i].car == pCar && the_time < gOily_spills[i].spill_time + 5000) { |
||
108 | return; |
||
109 | } |
||
110 | } |
||
111 | |||
112 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
113 | if (gOily_spills[i].car == NULL) { |
||
114 | oily_index = i; |
||
115 | break; |
||
116 | } |
||
117 | if (gOily_spills[i].spill_time < oldest_time) { |
||
118 | oldest_time = gOily_spills[i].spill_time; |
||
119 | oldest_one = i; |
||
120 | } |
||
121 | } |
||
122 | |||
123 | if (oily_index < 0) { |
||
124 | oily_index = oldest_one; |
||
125 | } |
||
126 | gOily_spills[oily_index].car = pCar; |
||
127 | gOily_spills[oily_index].spill_time = the_time + 500; |
||
128 | gOily_spills[oily_index].full_size = SRandomBetween(.35f, .6f); |
||
129 | gOily_spills[oily_index].grow_rate = SRandomBetween(3e-5f, 10e-5f); |
||
130 | gOily_spills[oily_index].current_size = .1f; |
||
131 | gOily_spills[oily_index].actor->render_style = BR_RSTYLE_NONE; |
||
132 | } |
||
133 | |||
134 | // IDA: int __usercall OKToSpillOil@<EAX>(tOil_spill_info *pOil@<EAX>) |
||
135 | int OKToSpillOil(tOil_spill_info* pOil) { |
||
136 | //br_scalar temp; // Pierre-Marie Baty -- unused variable |
||
137 | //br_scalar size_with_margin; // Pierre-Marie Baty -- unused variable |
||
138 | br_scalar distance; |
||
139 | br_scalar mr_dotty; |
||
140 | br_vector3 v; |
||
141 | br_vector3 ray_pos; |
||
142 | br_vector3 ray_dir; |
||
143 | br_vector3 normal; |
||
144 | tCar_spec* car; |
||
145 | int i; |
||
146 | int face_count; |
||
147 | int found_one; |
||
148 | br_angle angle_to_rotate_by; |
||
149 | tBounds kev_bounds; |
||
150 | tFace_ref the_list[10]; |
||
151 | tFace_ref* face_ref; |
||
152 | LOG_TRACE("(%p)", pOil); |
||
153 | |||
154 | car = pOil->car; |
||
155 | if (car->driver >= eDriver_net_human && car->damage_units[eDamage_engine].damage_level <= 98 && car->damage_units[eDamage_transmission].damage_level <= 98) { |
||
156 | return 0; |
||
157 | } |
||
158 | angle_to_rotate_by = IRandomBetween(0, 0xffff); |
||
159 | kev_bounds.original_bounds.min.v[0] = -pOil->full_size; |
||
160 | kev_bounds.original_bounds.min.v[1] = 1.5f * car->car_model_actors[car->principal_car_actor].actor->model->bounds.min.v[1]; |
||
161 | kev_bounds.original_bounds.min.v[2] = -pOil->full_size; |
||
162 | kev_bounds.original_bounds.max.v[0] = pOil->full_size; |
||
163 | kev_bounds.original_bounds.max.v[1] = car->car_model_actors[car->principal_car_actor].actor->model->bounds.max.v[1]; |
||
164 | kev_bounds.original_bounds.max.v[2] = pOil->full_size; |
||
165 | BrMatrix34PreRotateY(&pOil->actor->t.t.mat, angle_to_rotate_by); |
||
166 | kev_bounds.mat = &car->car_master_actor->t.t.mat; |
||
167 | face_count = FindFacesInBox(&kev_bounds, the_list, COUNT_OF(the_list)); |
||
168 | BrVector3Set(&v, .0f, .2f, .0f); |
||
169 | BrMatrix34ApplyP(&ray_pos, &v, &car->car_master_actor->t.t.mat); |
||
170 | BrVector3Set(&ray_dir, 0.f, kev_bounds.original_bounds.min.v[1] - kev_bounds.original_bounds.max.v[1], 0.f);\ |
||
171 | if (face_count == 0) { |
||
172 | return 0; |
||
173 | } |
||
174 | found_one = 0; |
||
175 | for (i = 0; i < face_count; i++) { |
||
176 | face_ref = &the_list[i]; |
||
177 | if (!found_one) { |
||
178 | CheckSingleFace(face_ref, &ray_pos, &ray_dir, &normal, &distance); |
||
179 | if (distance < 100.f) { |
||
180 | found_one = 1; |
||
181 | BrVector3Copy((br_vector3*)pOil->actor->t.t.mat.m[1], &normal); |
||
182 | BrVector3Set(&v, 0.f, 0.f, 1.f); |
||
183 | BrVector3Cross((br_vector3*)pOil->actor->t.t.mat.m[0], &normal, &v); |
||
184 | BrVector3Set(&v, 1.f, 0.f, 0.f); |
||
185 | BrVector3Cross((br_vector3*)pOil->actor->t.t.mat.m[2], &normal, &v); |
||
186 | BrVector3Scale(&v, &ray_dir, distance); |
||
187 | BrVector3Add(&pOil->pos, &ray_pos, &v); |
||
188 | BrMatrix34PreRotateY(&pOil->actor->t.t.mat, angle_to_rotate_by); |
||
189 | } |
||
190 | } |
||
191 | } |
||
192 | if (!found_one || normal.v[1] < .97f) { |
||
193 | return 0; |
||
194 | } |
||
195 | for (i = 0; i < face_count; i++) { |
||
196 | face_ref = &the_list[i]; |
||
197 | mr_dotty = BrVector3Dot(&face_ref->normal, &normal); |
||
198 | if (mr_dotty < .98f && (mr_dotty > .8f || !NormalSideOfPlane(&pOil->actor->t.t.translate.t, &face_ref->normal, face_ref->d))) { |
||
199 | return 0; |
||
200 | } |
||
201 | } |
||
202 | return 1; |
||
203 | } |
||
204 | |||
205 | // IDA: void __usercall Vector3Interpolate(br_vector3 *pDst@<EAX>, br_vector3 *pFrom@<EDX>, br_vector3 *pTo@<EBX>, br_scalar pP) |
||
206 | void Vector3Interpolate(br_vector3* pDst, br_vector3* pFrom, br_vector3* pTo, br_scalar pP) { |
||
207 | LOG_TRACE("(%p, %p, %p, %f)", pDst, pFrom, pTo, pP); |
||
208 | |||
209 | pDst->v[0] = (pTo->v[0] - pFrom->v[0]) * pP + pFrom->v[0]; |
||
210 | pDst->v[1] = (pTo->v[1] - pFrom->v[1]) * pP + pFrom->v[1]; |
||
211 | pDst->v[2] = (pTo->v[2] - pFrom->v[2]) * pP + pFrom->v[2]; |
||
212 | } |
||
213 | |||
214 | // IDA: void __usercall EnsureGroundDetailVisible(br_vector3 *pNew_pos@<EAX>, br_vector3 *pGround_normal@<EDX>, br_vector3 *pOld_pos@<EBX>) |
||
215 | void EnsureGroundDetailVisible(br_vector3* pNew_pos, br_vector3* pGround_normal, br_vector3* pOld_pos) { |
||
216 | br_scalar factor; |
||
217 | br_scalar s; |
||
218 | br_scalar dist; |
||
219 | br_vector3 to_camera; |
||
220 | LOG_TRACE("(%p, %p, %p)", pNew_pos, pGround_normal, pOld_pos); |
||
221 | |||
222 | to_camera.v[0] = gCamera_to_world.m[3][0] - pOld_pos->v[0]; |
||
223 | to_camera.v[1] = gCamera_to_world.m[3][1] - pOld_pos->v[1]; |
||
224 | to_camera.v[2] = gCamera_to_world.m[3][2] - pOld_pos->v[2]; |
||
225 | dist = BrVector3Length(&to_camera); |
||
226 | if (dist > BR_SCALAR_EPSILON) { |
||
227 | factor = BrVector3Dot(pGround_normal, &to_camera) / dist; |
||
228 | if (fabsf(factor) <= 0.01f) { |
||
229 | s = 0.01f; |
||
230 | } else { |
||
231 | s = 0.01f / factor; |
||
232 | if (s > 0.1f) { |
||
233 | s = 0.1f; |
||
234 | } |
||
235 | } |
||
236 | Vector3Interpolate(pNew_pos, pOld_pos, (br_vector3*)gCamera_to_world.m[3], s); |
||
237 | } |
||
238 | } |
||
239 | |||
240 | // IDA: void __usercall MungeOilsHeightAboveGround(tOil_spill_info *pOil@<EAX>) |
||
241 | void MungeOilsHeightAboveGround(tOil_spill_info* pOil) { |
||
242 | LOG_TRACE("(%p)", pOil); |
||
243 | |||
244 | EnsureGroundDetailVisible(&pOil->actor->t.t.look_up.t, &pOil->actor->t.t.look_up.up, &pOil->pos); |
||
245 | } |
||
246 | |||
247 | // IDA: void __usercall MungeIndexedOilsHeightAboveGround(int pIndex@<EAX>) |
||
248 | void MungeIndexedOilsHeightAboveGround(int pIndex) { |
||
249 | LOG_TRACE("(%d)", pIndex); |
||
250 | |||
251 | MungeOilsHeightAboveGround(&gOily_spills[pIndex]); |
||
252 | } |
||
253 | |||
254 | // IDA: void __usercall SetInitialOilStuff(tOil_spill_info *pOil@<EAX>, br_model *pModel@<EDX>) |
||
255 | void SetInitialOilStuff(tOil_spill_info* pOil, br_model* pModel) { |
||
256 | LOG_TRACE("(%p, %p)", pOil, pModel); |
||
257 | |||
258 | pModel->vertices[0].p.v[0] = -0.1f; |
||
259 | pModel->vertices[0].p.v[2] = -0.1f; |
||
260 | pModel->vertices[1].p.v[0] = 0.1f; |
||
261 | pModel->vertices[1].p.v[2] = -0.1f; |
||
262 | pModel->vertices[2].p.v[0] = 0.1f; |
||
263 | pModel->vertices[2].p.v[2] = 0.1f; |
||
264 | pModel->vertices[3].p.v[0] = -0.1f; |
||
265 | pModel->vertices[3].p.v[2] = 0.1f; |
||
266 | pOil->actor->render_style = BR_RSTYLE_FACES; |
||
267 | BrMaterialUpdate(pOil->actor->material, BR_MATU_ALL); |
||
268 | BrModelUpdate(pModel, BR_MODU_ALL); |
||
269 | } |
||
270 | |||
271 | // IDA: void __usercall ProcessOilSpills(tU32 pFrame_period@<EAX>) |
||
272 | void ProcessOilSpills(tU32 pFrame_period) { |
||
273 | int i; |
||
274 | tU32 time; |
||
275 | br_model* the_model; |
||
276 | br_scalar grow_amount; |
||
277 | //br_scalar initial_size; // Pierre-Marie Baty -- unused variable |
||
278 | br_scalar this_size; |
||
279 | br_vector3 v; |
||
280 | tNet_message* message; |
||
281 | LOG_TRACE("(%d)", pFrame_period); |
||
282 | |||
283 | time = GetTotalTime(); |
||
284 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
285 | if (gOily_spills[i].car == NULL) { |
||
286 | gOily_spills[i].actor->render_style = BR_RSTYLE_NONE; |
||
287 | } else { |
||
288 | the_model = gOily_spills[i].actor->model; |
||
289 | if (gOily_spills[i].actor->render_style == BR_RSTYLE_NONE && |
||
290 | gOily_spills[i].spill_time <= time && |
||
291 | fabsf(gOily_spills[i].car->v.v[0]) < .01f && |
||
292 | fabsf(gOily_spills[i].car->v.v[1]) < .01f && |
||
293 | fabsf(gOily_spills[i].car->v.v[2]) < .01f) { |
||
294 | if (gAction_replay_mode) { |
||
295 | SetInitialOilStuff(&gOily_spills[i], the_model); |
||
296 | } else { |
||
297 | if (!OKToSpillOil(&gOily_spills[i])) { |
||
298 | gOily_spills[i].car = NULL; |
||
299 | } else { |
||
300 | gOily_spills[i].spill_time = time; |
||
301 | gOily_spills[i].actor->material->colour_map = gOil_pixies[gNext_oil_pixie]; |
||
302 | gNext_oil_pixie++; |
||
303 | if (gNext_oil_pixie >= COUNT_OF(gOil_pixies)) { |
||
304 | gNext_oil_pixie = 0; |
||
305 | } |
||
306 | BrVector3Copy(&gOily_spills[i].original_pos, &gOily_spills[i].car->pos); |
||
307 | PipeSingleOilSpill(i, |
||
308 | &gOily_spills[i].actor->t.t.mat, |
||
309 | gOily_spills[i].full_size, |
||
310 | gOily_spills[i].grow_rate, |
||
311 | gOily_spills[i].spill_time, |
||
312 | gOily_spills[i].stop_time, |
||
313 | gOily_spills[i].car, |
||
314 | &gOily_spills[i].original_pos, |
||
315 | gOily_spills[i].actor->material->colour_map); |
||
316 | gOily_spills[i].stop_time = 0; |
||
317 | SetInitialOilStuff(&gOily_spills[i], the_model); |
||
318 | if (gNet_mode != eNet_mode_none) { |
||
319 | message = NetBuildMessage(30, 0); |
||
320 | message->contents.data.oil_spill.player = NetPlayerFromCar(gOily_spills[i].car)->ID; |
||
321 | message->contents.data.oil_spill.full_size = gOily_spills[i].full_size; |
||
322 | message->contents.data.oil_spill.grow_rate = gOily_spills[i].grow_rate; |
||
323 | message->contents.data.oil_spill.current_size = gOily_spills[i].current_size; |
||
324 | NetGuaranteedSendMessageToAllPlayers(gCurrent_net_game, message, NULL); |
||
325 | } |
||
326 | } |
||
327 | } |
||
328 | } else { |
||
329 | if (gOily_spills[i].actor->render_style == BR_RSTYLE_FACES && |
||
330 | (gOily_spills[i].stop_time == 0 || time < gOily_spills[i].stop_time)) { |
||
331 | BrVector3Sub(&v, &gOily_spills[i].original_pos, &gOily_spills[i].car->pos); |
||
332 | grow_amount = BrVector3LengthSquared(&v); |
||
333 | if (gOily_spills[i].stop_time != 0 || grow_amount <= 0.2f) { |
||
334 | this_size = 0.1f + (time - gOily_spills[i].spill_time) * gOily_spills[i].grow_rate; |
||
335 | if (this_size >= 0.1f) { |
||
336 | gOily_spills[i].actor->render_style = BR_RSTYLE_FACES; |
||
337 | if (this_size <= gOily_spills[i].full_size) { |
||
338 | the_model->vertices[0].p.v[0] = -this_size; |
||
339 | the_model->vertices[0].p.v[2] = -this_size; |
||
340 | the_model->vertices[1].p.v[0] = this_size; |
||
341 | the_model->vertices[1].p.v[2] = -this_size; |
||
342 | the_model->vertices[2].p.v[0] = this_size; |
||
343 | the_model->vertices[2].p.v[2] = this_size; |
||
344 | the_model->vertices[3].p.v[0] = -this_size; |
||
345 | the_model->vertices[3].p.v[2] = this_size; |
||
346 | gOily_spills[i].current_size = this_size; |
||
347 | } else { |
||
348 | the_model->vertices[0].p.v[0] = -gOily_spills[i].full_size; |
||
349 | the_model->vertices[0].p.v[2] = -gOily_spills[i].full_size; |
||
350 | the_model->vertices[1].p.v[0] = gOily_spills[i].full_size; |
||
351 | the_model->vertices[1].p.v[2] = -gOily_spills[i].full_size; |
||
352 | the_model->vertices[2].p.v[0] = gOily_spills[i].full_size; |
||
353 | the_model->vertices[2].p.v[2] = gOily_spills[i].full_size; |
||
354 | the_model->vertices[3].p.v[0] = -gOily_spills[i].full_size; |
||
355 | the_model->vertices[3].p.v[2] = gOily_spills[i].full_size; |
||
356 | gOily_spills[i].current_size = gOily_spills[i].full_size; |
||
357 | } |
||
358 | BrModelUpdate(the_model, BR_MODU_ALL); |
||
359 | } else { |
||
360 | gOily_spills[i].actor->render_style = BR_RSTYLE_NONE; |
||
361 | } |
||
362 | } else { |
||
363 | gOily_spills[i].stop_time = time; |
||
364 | continue; |
||
365 | } |
||
366 | } |
||
367 | } |
||
368 | } |
||
369 | if (gOily_spills[i].actor->render_style == BR_RSTYLE_FACES) { |
||
370 | MungeOilsHeightAboveGround(&gOily_spills[i]); |
||
371 | } |
||
372 | } |
||
373 | } |
||
374 | |||
375 | // IDA: int __cdecl GetOilSpillCount() |
||
376 | int GetOilSpillCount(void) { |
||
377 | //LOG_TRACE("()"); |
||
378 | |||
379 | return COUNT_OF(gOily_spills); |
||
380 | } |
||
381 | |||
382 | // IDA: void __usercall GetOilSpillDetails(int pIndex@<EAX>, br_actor **pActor@<EDX>, br_scalar *pSize@<EBX>) |
||
383 | void GetOilSpillDetails(int pIndex, br_actor** pActor, br_scalar* pSize) { |
||
384 | LOG_TRACE("(%d, %p, %p)", pIndex, pActor, pSize); |
||
385 | |||
386 | if (gOily_spills[pIndex].car != NULL) { |
||
387 | *pActor = gOily_spills[pIndex].actor; |
||
388 | *pSize = gOily_spills[pIndex].full_size; |
||
389 | } else { |
||
390 | *pActor = NULL; |
||
391 | } |
||
392 | } |
||
393 | |||
394 | #define SQR(V) ((V)*(V)) |
||
395 | |||
396 | // IDA: int __usercall PointInSpill@<EAX>(br_vector3 *pV@<EAX>, int pSpill@<EDX>) |
||
397 | int PointInSpill(br_vector3* pV, int pSpill) { |
||
398 | LOG_TRACE("(%p, %d)", pV, pSpill); |
||
399 | |||
400 | return gOily_spills[pSpill].current_size * gOily_spills[pSpill].current_size * 0.8f > SQR(pV->v[0] / WORLD_SCALE - gOily_spills[pSpill].actor->t.t.translate.t.v[0]) |
||
401 | && gOily_spills[pSpill].current_size * gOily_spills[pSpill].current_size * 0.8f > SQR(pV->v[2] / WORLD_SCALE - gOily_spills[pSpill].actor->t.t.translate.t.v[2]) |
||
402 | && fabsf(pV->v[1] / WORLD_SCALE - gOily_spills[pSpill].actor->t.t.translate.t.v[1]) < 0.1f; |
||
403 | } |
||
404 | |||
405 | // IDA: void __usercall GetOilFrictionFactors(tCar_spec *pCar@<EAX>, br_scalar *pFl_factor@<EDX>, br_scalar *pFr_factor@<EBX>, br_scalar *pRl_factor@<ECX>, br_scalar *pRr_factor) |
||
406 | void GetOilFrictionFactors(tCar_spec* pCar, br_scalar* pFl_factor, br_scalar* pFr_factor, br_scalar* pRl_factor, br_scalar* pRr_factor) { |
||
407 | int i; |
||
408 | br_vector3 wheel_world; |
||
409 | LOG_TRACE("(%p, %p, %p, %p, %p)", pCar, pFl_factor, pFr_factor, pRl_factor, pRr_factor); |
||
410 | |||
411 | *pFl_factor = 1.0f; |
||
412 | *pFr_factor = 1.0f; |
||
413 | *pRl_factor = 1.0f; |
||
414 | *pRr_factor = 1.0f; |
||
415 | switch (pCar->driver) { |
||
416 | case eDriver_non_car_unused_slot: |
||
417 | case eDriver_non_car: |
||
418 | return; |
||
419 | default: |
||
420 | break; |
||
421 | } |
||
422 | if (pCar->shadow_intersection_flags != 0) { |
||
423 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
424 | if (((1 << i) & pCar->shadow_intersection_flags) != 0 && gOily_spills[i].car != NULL) { |
||
425 | BrMatrix34ApplyP(&wheel_world, &pCar->wpos[2], &pCar->car_master_actor->t.t.mat); |
||
426 | if (PointInSpill(&wheel_world, i)) { |
||
427 | pCar->oil_remaining[2] = SRandomBetween(1.5f, 2.5f); |
||
428 | } |
||
429 | BrMatrix34ApplyP(&wheel_world, &pCar->wpos[3], &pCar->car_master_actor->t.t.mat); |
||
430 | if (PointInSpill(&wheel_world, i)) { |
||
431 | pCar->oil_remaining[3] = SRandomBetween(1.5f, 2.5f); |
||
432 | } |
||
433 | BrMatrix34ApplyP(&wheel_world, &pCar->wpos[0], &pCar->car_master_actor->t.t.mat); |
||
434 | if (PointInSpill(&wheel_world, i)) { |
||
435 | pCar->oil_remaining[0] = SRandomBetween(1.5f, 2.5f); |
||
436 | } |
||
437 | BrMatrix34ApplyP(&wheel_world, &pCar->wpos[1], &pCar->car_master_actor->t.t.mat); |
||
438 | if (PointInSpill(&wheel_world, i)) { |
||
439 | pCar->oil_remaining[1] = SRandomBetween(1.5f, 2.5f); |
||
440 | } |
||
441 | } |
||
442 | } |
||
443 | } |
||
444 | if (pCar->oil_remaining[2] != 0.0f) { |
||
445 | *pFl_factor = SRandomBetween(0.01f, 0.15f); |
||
446 | } |
||
447 | if (pCar->oil_remaining[3] != 0.0f) { |
||
448 | *pFr_factor = SRandomBetween(0.01f, 0.15f); |
||
449 | } |
||
450 | if (pCar->oil_remaining[0] != 0.0f) { |
||
451 | *pRl_factor = SRandomBetween(0.01f, 0.15f); |
||
452 | } |
||
453 | if (pCar->oil_remaining[1] != 0.0f) { |
||
454 | *pRr_factor = SRandomBetween(0.01f, 0.15f); |
||
455 | } |
||
456 | } |
||
457 | |||
458 | // IDA: void __usercall AdjustOilSpill(int pIndex@<EAX>, br_matrix34 *pMat@<EDX>, br_scalar pFull_size, br_scalar pGrow_rate, tU32 pSpill_time, tU32 pStop_time, tCar_spec *pCar, br_vector3 *pOriginal_pos, br_pixelmap *pPixelmap) |
||
459 | void AdjustOilSpill(int pIndex, br_matrix34* pMat, br_scalar pFull_size, br_scalar pGrow_rate, tU32 pSpill_time, tU32 pStop_time, tCar_spec* pCar, br_vector3* pOriginal_pos, br_pixelmap* pPixelmap) { |
||
460 | LOG_TRACE("(%d, %p, %f, %f, %d, %d, %p, %p, %p)", pIndex, pMat, pFull_size, pGrow_rate, pSpill_time, pStop_time, pCar, pOriginal_pos, pPixelmap); |
||
461 | |||
462 | BrMatrix34Copy(&gOily_spills[pIndex].actor->t.t.mat, pMat); |
||
463 | gOily_spills[pIndex].full_size = pFull_size; |
||
464 | gOily_spills[pIndex].grow_rate = pGrow_rate; |
||
465 | gOily_spills[pIndex].spill_time = pSpill_time; |
||
466 | gOily_spills[pIndex].stop_time = pStop_time; |
||
467 | gOily_spills[pIndex].car = pCar; |
||
468 | BrVector3Copy(&gOily_spills[pIndex].original_pos, pOriginal_pos); |
||
469 | gOily_spills[pIndex].actor->material->colour_map = pPixelmap; |
||
470 | gOily_spills[pIndex].actor->render_style = BR_RSTYLE_NONE; |
||
471 | } |
||
472 | |||
473 | // IDA: void __usercall ReceivedOilSpill(tNet_contents *pContents@<EAX>) |
||
474 | void ReceivedOilSpill(tNet_contents* pContents) { |
||
475 | int i; |
||
476 | int oily_index; |
||
477 | int oldest_one; |
||
478 | tU32 the_time; |
||
479 | tU32 oldest_time; |
||
480 | tCar_spec* car; |
||
481 | LOG_TRACE("(%p)", pContents); |
||
482 | |||
483 | oldest_one = 0; |
||
484 | car = NetCarFromPlayerID(pContents->data.oil_spill.player); |
||
485 | if (car == NULL) { |
||
486 | return; |
||
487 | } |
||
488 | oily_index = -1; |
||
489 | the_time = GetTotalTime(); |
||
490 | oldest_time = GetTotalTime(); |
||
491 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
492 | if (gOily_spills[i].car == car && the_time < gOily_spills[i].spill_time + 5000) { |
||
493 | return; |
||
494 | } |
||
495 | } |
||
496 | for (i = 0; i < COUNT_OF(gOily_spills); i++) { |
||
497 | if (gOily_spills[i].car == NULL) { |
||
498 | oily_index = i; |
||
499 | break; |
||
500 | } |
||
501 | if (gOily_spills[i].spill_time < oldest_time) { |
||
502 | oldest_time = gOily_spills[i].spill_time; |
||
503 | oldest_one = i; |
||
504 | } |
||
505 | } |
||
506 | if (oily_index < 0) { |
||
507 | oily_index = oldest_one; |
||
508 | } |
||
509 | gOily_spills[oily_index].car = car; |
||
510 | gOily_spills[oily_index].spill_time = the_time; |
||
511 | gOily_spills[oily_index].full_size = pContents->data.oil_spill.full_size; |
||
512 | gOily_spills[oily_index].grow_rate = pContents->data.oil_spill.grow_rate; |
||
513 | gOily_spills[oily_index].current_size = pContents->data.oil_spill.current_size; |
||
514 | gOily_spills[oily_index].actor->render_style = BR_RSTYLE_NONE; |
||
515 | } |