-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathcl_index.js
495 lines (442 loc) · 21.3 KB
/
cl_index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
// ========================================================================================================================
// VARIABLES
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
/**
* nothing really can be changed here, just cached variables
* you actually don't have to change anything in this file at all
* you can tweak everything from the 'sh_config.js' file!!
*/
const Cached = {
_isAlive: true,
_justDied: true,
_reviveAt: 0,
_respawnAt: 0,
_reviveMessageTimer: 0,
_respawnMessageTimer: 0,
_isAutoRespawnAllowed: true,
_isReviveAllowed: false,
_isRespawnAllowed: false,
_goingToClouds: false,
_warningInterval: null,
_warningCounter: 0,
_warningCounterLimit: 0,
_lastKeybindUsedAt: Date.now()
};
const colors = [['~c~', '~r~'], ['~y~', '~y~']];
const colGray = colors[0][0];
let colRed = colors[0][1];
// ========================================================================================================================
// COMMANDS/KEYBINDS STUFF
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
/**
* Add suggestions to the viewable and usable commands
*/
if (Config.Commands.Revive.Enabled) emit('chat:addSuggestion', Config.Commands.Revive.Suggestion.name, Config.Commands.Revive.Suggestion.help, []);
if (Config.Commands.Respawn.Enabled) emit('chat:addSuggestion', Config.Commands.Respawn.Suggestion.name, Config.Commands.Respawn.Suggestion.help, []);
if (Config.Commands.AdRev.Enabled) emit('chat:addSuggestion', Config.Commands.AdRev.Suggestion.name, Config.Commands.AdRev.Suggestion.help, Config.Commands.AdRev.Suggestion.Params);
if (Config.Commands.AdRes.Enabled) emit('chat:addSuggestion', Config.Commands.AdRes.Suggestion.name, Config.Commands.AdRes.Suggestion.help, Config.Commands.AdRes.Suggestion.Params);
if (Config.Commands.AdRevAll.Enabled) emit('chat:addSuggestion', Config.Commands.AdRevAll.Suggestion.name, Config.Commands.AdRevAll.Suggestion.help, []);
if (Config.Commands.AdResAll.Enabled) emit('chat:addSuggestion', Config.Commands.AdResAll.Suggestion.name, Config.Commands.AdResAll.Suggestion.help, []);
if (Config.Commands.ToggleDS.Enabled) emit('chat:addSuggestion', Config.Commands.ToggleDS.Suggestion.name, Config.Commands.ToggleDS.Suggestion.help, []);
emit('chat:removeSuggestion', '/=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRev');
emit('chat:removeSuggestion', '/=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRes');
/**
* Registering keymappings AKA; Keybinds, for player customizability and quick command usage
*/
RegisterKeyMapping('revive', 'Revive', 'keyboard', Config.Commands.Revive.Keybind.DefaultKeybind);
RegisterKeyMapping('respawn', 'Respawn', 'keyboard', Config.Commands.Respawn.Keybind.DefaultKeybind);
RegisterKeyMapping('=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRev', 'Admin Revive', 'keyboard', Config.Commands.AdRev.Keybind.DefaultKeybind);
RegisterKeyMapping('=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRes', 'Admin Respawn', 'keyboard', Config.Commands.AdRes.Keybind.DefaultKeybind);
/**
* Revive command/Keybind
*/
if (Config.Commands.Revive.Enabled) {
RegisterCommand('revive', () => {
if (GetEntityHealth(PlayerPedId()) <= 1) {
if (Cached._isReviveAllowed) {
RevivePed(PlayerPedId());
return emit('chat:addMessage', { args: Config.Commands.Revive.Messages.Revive });
} else {
return DisplayWarning();
}
} else {
return emit('chat:addMessage', { args: Config.Commands.Revive.Messages.Alive });
}
});
}
/**
* Respawn command/Keybind
*/
if (Config.Commands.Respawn.Enabled) {
RegisterCommand('respawn', () => {
if (GetEntityHealth(PlayerPedId()) <= 1) {
if (Cached._isRespawnAllowed) {
RespawnPed(PlayerPedId());
emit('chat:addMessage', { args: Config.Commands.Respawn.Messages.Respawn });
return;
} else {
return DisplayWarning();
}
} else {
return emit('chat:addMessage', { args: Config.Commands.Respawn.Messages.Alive });
}
});
}
/**
* Admin Revive command/Keybind
* This is the client side version of the 'AdRev' command,
* this only exists for the keybind, hence the random and hard to remember command.
*/
if (Config.Commands.AdRev.Enabled) {
RegisterCommand('=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRev', () => {
console.log(AreKeybindsOnCooldown);
if (!AreKeybindsOnCooldown()) {
Cached._lastKeybindUsedAt = Date.now();
emitNet('DeathScript:Admin:CheckAce', GetPlayerServerId(GetPlayerIndex()), 'AdRev');
} else {
emit('chat:addMessage', { args: Config.Commands.AdRev.Messages.ToStaff.Cooldown });
}
});
}
/**
* Admin Respawn command/Keybind
* This is the client side version of the 'AdRes' command,
* this only exists for the keybind, hence the random and hard to remember command.
*/
if (Config.Commands.AdRes.Enabled) {
RegisterCommand('=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRes', () => {
if (!AreKeybindsOnCooldown()) {
Cached._lastKeybindUsedAt = Date.now();
emitNet('DeathScript:Admin:CheckAce', GetPlayerServerId(GetPlayerIndex()), 'AdRes');
} else {
emit('chat:addMessage', { args: Config.Commands.AdRes.Messages.ToStaff.Cooldown });
}
});
}
// ========================================================================================================================
// EVENTS
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
/**
* spawning the player then disabling 'AutoSpawn'
* otherwise the player will be stuck in the 'awaiting resources' screen forever
* you'd also need to delay the 'AutoSpawn' so it triggers a few seconds after the player's ped spawns are ready to play
*/
on('onClientMapStart', async () => {
exports.spawnmanager.spawnPlayer();
await Delay(1500);
exports.spawnmanager.setAutoSpawn(false);
});
/**
* this event is used by both 'Adrev' & 'Adrevall' commands
* as there is no need for a new event for it
* just pass a parameter and check if it's true or false
*/
onNet('DeathScript:Admin:Revive', (Moderator, Everyone = false) => {
if (Everyone) {
if (GetEntityHealth(PlayerPedId()) <= 1) {
RevivePed(PlayerPedId());
if (Moderator !== GetPlayerServerId(GetPlayerIndex())) emit('chat:addMessage', { args: Config.Commands.AdRevAll.Messages.ToPlayer.Revived });
return;
}
} else {
RevivePed(PlayerPedId());
}
});
/**
* this event is used by both 'Adres' & 'Adresall' commands
* as there is no need for a new event for it
* just pass a parameter and check if it's true or false
* same shizz
*/
onNet('DeathScript:Admin:Respawn', (Moderator, Everyone = false) => {
if (Everyone) {
if (GetEntityHealth(PlayerPedId()) <= 1) {
RespawnPed(PlayerPedId());
if (Moderator !== GetPlayerServerId(GetPlayerIndex())) emit('chat:addMessage', { args: Config.Commands.AdResAll.Messages.ToPlayer.Respawned });
return;
}
} else {
RespawnPed(PlayerPedId());
}
});
/**
* This event is used as a callback for the "DeathScript:Admin:CheckAce" event used by
* the "=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRes" and "=-+_+-=Death-_._-Script=-+_+-=Key-_._-bind=-+_+-=AdRev" commands.
* This will only be readable if 'AdRev or 'AdRes' or both were enabled.
*/
if (Config.Commands.AdRev.Enabled || Config.Commands.AdRes.Enabled) {
onNet('DeathScript:Admin:CheckAce:Return', async (Moderator, Command, AceAllowed) => {
if (AceAllowed) {
const input = await KeyboardInput(`Player ID: [leave empty to ${Command === 'AdRev' ? 'revive' : 'respawn'} yourself]`, '', 5);
if (!input) return;
emitNet('DeathScript:Admin:ValidatePlayer', Moderator, input, Command);
} else {
emit('chat:addMessage', { args: Config.Commands[Command].Messages.ToStaff.NoPermission });
}
});
}
/**
* This is used to update the client side of the script
*/
onNet('DeathScript:Toggle', NewState => {
Config.Enabled = NewState;
});
// ========================================================================================================================
// MAIN THREADS
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
/**
* this only manages timer when the player's ped dies or their health reach 1
*/
setTick(async () => {
if (!Config.Enabled) return;
const Ped = PlayerPedId();
if (IsEntityDead(Ped) && GetEntityHealth(Ped) <= 1) {
const CurrentTimestamp = Date.now();
if (CurrentTimestamp >= Cached._reviveAt) {
if (!Cached._isReviveAllowed) {
Cached._isReviveAllowed = true;
}
} else if (CurrentTimestamp >= Cached._reviveMessageTimer) {
Cached._reviveMessageTimer = Date.now();
if (Cached._isReviveAllowed) {
Cached._isReviveAllowed = false;
}
}
if (CurrentTimestamp >= Cached._respawnAt) {
if (!Cached._isRespawnAllowed) {
Cached._isRespawnAllowed = true;
}
} else if (CurrentTimestamp >= Cached._respawnMessageTimer) {
Cached._respawnMessageTimer = Date.now();
if (Cached._isRespawnAllowed) {
Cached._isRespawnAllowed = false;
}
}
}
await Delay(500);
});
/**
* this manages both Cached variables and the timer display text when the player's ped dies or their health reach 1
*/
setTick(async () => {
const Ped = PlayerPedId();
if (IsEntityDead(Ped) && GetEntityHealth(Ped) <= 1) {
if (!Config.Enabled) {
if (Cached._isAutoRespawnAllowed) {
Cached._isAutoRespawnAllowed = false;
Cached._respawnAt = Date.now() + (Config.Commands.Respawn.AutoRespawnTimer * 1000);
const Tick = setTick(() => {
DrawTextOnScreen(`${colGray}Respawning in ${parseInt((Cached._respawnAt - Date.now()) / 1000) === 1 ? colRed + ' in ' + parseInt((Cached._respawnAt - Date.now()) / 1000) + ' second' : colRed + ' in ' + parseInt((Cached._respawnAt - Date.now()) / 1000) + ' seconds'}`);
});
await Delay(Config.Commands.Respawn.AutoRespawnTimer * 1000);
clearTick(Tick);
RespawnPed(Ped);
}
} else {
if (Cached._justDied) {
Cached._justDied = false;
Cached._isAlive = false;
Cached._reviveAt = Date.now() + (Config.Commands.Revive.WaitTime * 1000);
Cached._respawnAt = Date.now() + (Config.Commands.Respawn.WaitTime * 1000);
}
if (!Cached._goingToClouds) {
SetPlayerInvincible(Ped, true);
SetEntityHealth(Ped, 1);
DrawTextOnScreen(`
${colGray}Revive: ${Cached._isReviveAllowed ? ' ~g~/Revive' : parseInt((Cached._reviveAt - Date.now()) / 1000) === 1 ? colRed + ' In ' + parseInt((Cached._reviveAt - Date.now()) / 1000) + ' second' : colRed + 'In ' + parseInt((Cached._reviveAt - Date.now()) / 1000) + ' seconds'}
${colGray}Respawn: ${Cached._isRespawnAllowed ? ' ~g~/Respawn' : parseInt((Cached._respawnAt - Date.now()) / 1000) === 1 ? colRed + ' In ' + parseInt((Cached._respawnAt - Date.now()) / 1000) + ' second' : colRed + ' In ' + parseInt((Cached._respawnAt - Date.now()) / 1000) + ' seconds'}
`);
}
}
}
});
// ========================================================================================================================
// FUNCTIONS
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
/**
* @name DisplayWarning
* @description This is responsible for the red/yellow text flash when the player tries to revive/respawn before the timers run out to get their attention
* @returns returnless
* @example DisplayWarning()
*/
const DisplayWarning = () => {
if (Cached._warningInterval) return Cached._warningCounterLimit = Cached._warningCounterLimit + 2;
Cached._warningCounter = 0;
Cached._warningCounterLimit = 9;
Cached._warningInterval = setInterval(() => {
Cached._warningCounter++;
PlaySoundFrontend(...Config.Sounds.Warning);
colRed = colors[0][1] === colRed ? colors[1][1] : colors[0][1];
if (Cached._warningCounter > Cached._warningCounterLimit || Cached._isReviveAllowed && Cached._isRespawnAllowed) {
clearInterval(Cached._warningInterval);
Cached._warningInterval = null;
colRed = colors[0][1];
}
}, 400);
};
/**
* @name ResetVariables
* @description Resets the usable Cached variables to be ready for next use
* @returns returnless
* @example ResetVariables(Ped)
*/
const ResetVariables = () => {
Cached._isAlive = true;
Cached._justDied = true;
Cached._reviveAt = 0;
Cached._respawnAt = 0;
Cached._reviveMessageTimer = 0;
Cached._respawnMessageTimer = 0;
Cached._isReviveAllowed = false;
Cached._isRespawnAllowed = false;
Cached._isAutoRespawnAllowed = true;
};
/**
* @name RevivePed
* @description Revives/Ressurect the player's ped after the revive timer runs out using the '/revive' command or the keybind
* @param {*} Ped The players' ped
* @returns returnless
* @example RevivePed(Ped)
*/
const RevivePed = Ped => {
const PedCoords = GetEntityCoords(Ped, true);
const heading = GetEntityHeading(Ped);
NetworkResurrectLocalPlayer(...PedCoords, heading, true, false);
SetPlayerInvincible(Ped, false);
ClearPedBloodDamage(Ped);
ResetVariables();
};
/**
* @name RespawnPed
* @description Respawns the player's ped after the respawn timer runs out using the '/respawn' command or the keybind
* @param {*} Ped The players' ped
* @returns returnless
* @example RespawnPed(Ped)
*/
const RespawnPed = async Ped => {
const { X, Y, Z, Heading } = GetNearestHospital(Ped);
let tick;
if (Config.GoToClouds) { // You can disable the clouds transition for a quick and instant respawn from the 'sh_config.js' file.
Cached._goingToClouds = true; // this is only a Cached variable; and doesn't stop the transition, if you want to remove the transition do so from the 'sh_config.js' file.
if (!IsPlayerSwitchInProgress()) {
SwitchOutPlayer(PlayerPedId(), 0, 1); // Make player meet god
}
tick = setTick(() => {
SetCloudHatOpacity(0.1);
HideHudAndRadarThisFrame();
});
while (GetPlayerSwitchState() !== 5) { // Wait until player cam is in the sky
await Delay(1);
}
}
RequestCollisionAtCoord(X, Y, Z);
FreezeEntityPosition(Ped, true);
SetEntityCoordsNoOffset(Ped, X, Y, Z, false, false, false, true);
NetworkResurrectLocalPlayer(X, Y, Z, Heading, true, false);
SetPlayerInvincible(Ped, false);
ClearPedBloodDamage(Ped);
ResetVariables();
if (Config.GoToClouds) { // You can disable the clouds transition for a quick and instant respawn from the 'sh_config.js' file.
SwitchInPlayer(PlayerPedId()); // Get the player back to us
while (GetPlayerSwitchState() !== 12) { // Wait until player cam is back to the ground
await Delay(1);
}
clearTick(tick);
Cached._goingToClouds = false; // this is only a Cached variable; and doesn't stop the transition, if you want to remove the transition do so from the 'sh_config.js' file.
}
FreezeEntityPosition(PlayerPedId(), false);
TriggerEvent('playerSpawned', X, Y, Z, Heading); // Triggers the 'playerSpawned' event in 'spawnmanager' resource for better compatibility with other resources that utilize this event
};
/**
* @name GetNearestHospital
* @description Gets the nearest registered hospital to the ped, registered hospitals can be editted from the 'sh_config.js' file.
* @param {*} Ped The players' ped
* @returns {object} [Distance, X, Y, Z, Heading]
* @example GetNearestHospital(Ped)
*/
const GetNearestHospital = Ped => {
let Hospitals = Config.Hospitals;
Hospitals = Hospitals.map(({ X, Y, Z, Heading, Name }) => ({ Distance: Vdist(...GetEntityCoords(Ped), X, Y, Z), X: X, Y: Y, Z: Z, Heading: Heading, Name: Name }));
const ClosestHospital = Hospitals.reduce((a, b) => a.Distance < b.Distance ? a : b);
console.log(ClosestHospital)
return ClosestHospital;
};
/**
* @name DrawTextOnScreen
* @description Draw a specified text on the screen
* @param {string} Text The 'specified text' :p
* @returns returnless
* @example DrawTextOnScreen(Text)
*/
const DrawTextOnScreen = Text => {
SetTextFont(4);
SetTextScale(0.4, 0.4);
SetTextOutline();
SetTextEntry('STRING');
AddTextComponentString(Text);
DrawText(0.00, 0.465);
};
/**
* @name Delay
* @description A shortcut funtion for an awaited promise use to delay intervals/ticks; a JS alternative to the 'Citizen.Wait' native in LUA/C#
* @param {interger} MS The amount in milliseconds
* @returns returnless
* @example Delay(MS)
*/
const Delay = MS => {
return new Promise(resolve => setTimeout(resolve, MS));
};
/**
* @name AreKeybindsOnCooldown
* @description Checks when the last time a keybind that uses a server side event was used, if was used within the last second return false, else true
* @returns {boolean} True/False
* @example AreKeybindsOnCooldown()
*/
const AreKeybindsOnCooldown = () => {
if (Cached._lastKeybindUsedAt + 1500 > Date.now()) {
return true;
} else {
return false;
}
};
/**
* Credit to Flatracer (https://forum.cfx.re/u/flatracer)
* From this post (https://forum.cfx.re/t/use-displayonscreenkeyboard-properly/51143/2)
* Converted from LUA to JavaScript by me ItsAmmarB
*/
const KeyboardInput = async (TextEntry, ExampleText, MaxStringLenght) => {
// TextEntry --> The Text above the typing field in the black square
// ExampleText --> An Example Text, what it should say in the typing field
// MaxStringLenght --> Maximum String Lenght
AddTextEntry('FMMC_KEY_TIP1', TextEntry); // Sets the Text above the typing field in the black square
DisplayOnscreenKeyboard(1, 'FMMC_KEY_TIP1', '', ExampleText, '', '', '', MaxStringLenght); // Actually calls the Keyboard Input
while (UpdateOnscreenKeyboard() !== 1 && UpdateOnscreenKeyboard() !== 2) { // While typing is not aborted and not finished, this loop waits
await Delay(50);
}
if (UpdateOnscreenKeyboard() !== 2) {
let result = GetOnscreenKeyboardResult(); // Gets the result of the typing
await Delay(50); // Little Time Delay, so the Keyboard won't open again if you press enter to finish the typing
if (!result) result = GetPlayerServerId(GetPlayerIndex()); // if submitted empty then set the player's ID as the result, AKA; self-admin-revive/self-admin-respawn
return result; // Returns the result
} else {
await Delay(50); // Little Time Delay, so the Keyboard won't open again if you press enter to finish the typing
return null; // Returns nil if the typing got aborted
}
};
// ========================================================================================================================
// EXPORTS
// ------------------------------------------------------------------------------------------------------------------------
// MAKES SURE YOU KNOW WHAT YOU'RE DOING BEFORE YOU CHANGE ANYTHING
// ========================================================================================================================
exports('Revive', () => RevivePed(PlayerPedId()));
exports('Respawn', () => RespawnPed(PlayerPedId()));