Обновление кода плагина

Сообщения
317
Реакции
131
Помог
4 раз(а)
Есть крутой плагин Juicy Corpse от VEN, я использовал его на зомби сервере, и зомби ели трупы издавая звуки, это классно и не обычно
Код:
/*
    google translate:
    Описание:
    Просто используйте тело мертвого игрока, чтобы восстановить свое здоровье.
    Но будьте осторожны, другие игроки могут услышать ваше жадное всасывание.
    Трупы сочные! :P

    Квары:
    total_juice (по дефолту: 30) - кол-во сока трупа / total amount of a corpse's juice
    suction_amount (по дефолту: 3) - кол-во восстановленного здоровья на всасывание / amount of restored health per suction
*/
#include <amxmodx>
#include <fakemeta>
#include <xs>
#include <biohazard>

#define PLUGIN_NAME "Juicy Corpse"
#define PLUGIN_VERSION "0.4"
#define PLUGIN_AUTHOR "VEN"

#define HUNGRY_PHRASE_ENABLED
// когда ест труп, выводятся сообщения
#define SUCTION_MESSAGE_NUM 2
new HUNGRY_PHRASE[SUCTION_MESSAGE_NUM][] =
{
    " Мммм... Вкуснятина...",
    " Мозги что надо... А добавка будет?!"
}

#define MAX_DISTANCE 90         // макс. дистанция поедания
#define MAX_VIEWANGLE_DIFF 60     //
#define SUCTION_INTERVAL 1.0     // интервал +hp
#define HEALTH_LIMIT 3500.0     // макс. лимит

#define ACCESS ADMIN_LEVEL_H     // флаг доступа

// звуки поедания
#define SUCTION_SOUND_NUM 2
new SUCTION_FILE[SUCTION_SOUND_NUM][] =
{
    "zo_idle1.wav",
    "zo_alert10.wav"
}

#define CVAR_TOTAL_JUICE_NAME "total_juice"
#define CVAR_TOTAL_JUICE_DEF "1000" // кол-во жизней в трупе

#define CVAR_SUCTION_AMOUNT_NAME "suction_amount"
#define CVAR_SUCTION_AMOUNT_DEF "100" // кол-во восстановленного здоровья в секунду из трупа

#define FORMULA_TOTAL_SUCTION_TIME(%1,%2) floatround((SUCTION_INTERVAL / (%1)) * (%2))
#define FORMULA_SUCKED_OUT_PERCENT(%1,%2) floatround((100.0 * ((%2) - (%1))) / (%2))

#define IN_USE (1<<5)
#define DEAD_DEAD 2

new USE_SOUND[] = "common/wpn_denyselect.wav" // звук нажатия на Е

// интервал чего то
#define DEAD_FLAG_CHECK_INTERVAL 1.5
#define DEAD_FLAG_CHECK_MAX_TIME 1.0

#define MAX_PLAYERS 32
new g_origin[MAX_PLAYERS + 1][3]
new g_juice[MAX_PLAYERS + 1]

new g_pcvar_total_juice
new g_pcvar_suction_amount

new g_msgid_bartime2

public plugin_precache()
{
    for (new i = 0; i < SUCTION_SOUND_NUM; ++i)
        precache_sound(SUCTION_FILE[i])
}

public plugin_init()
{
    register_plugin(PLUGIN_NAME, PLUGIN_VERSION, PLUGIN_AUTHOR)

    //register_event("ResetHUD", "event_hud_reset", "be")
    RegisterHookChain(RG_CBasePlayer_Killed, "HC_CBasePlayer_Killed_Post", true)
    register_event("DeathMsg", "event_death", "a")

    register_forward(FM_EmitSound, "forward_emit_sound")

    g_pcvar_total_juice = register_cvar(CVAR_TOTAL_JUICE_NAME, CVAR_TOTAL_JUICE_DEF)
    g_pcvar_suction_amount = register_cvar(CVAR_SUCTION_AMOUNT_NAME, CVAR_SUCTION_AMOUNT_DEF)

    g_msgid_bartime2 = get_user_msgid("BarTime2")
}

//public event_hud_reset(id)
public HC_CBasePlayer_Killed_Post(id)
{
    g_juice[id] = 0
}

public event_death()
{
    new id = read_data(2)

    if (task_exists(id))
        remove_task(id)

    set_task(DEAD_FLAG_CHECK_INTERVAL, "task_dead_flag_check", id, _, _, "b")
}

public client_disconnected(id)
{
    g_juice[id] = 0

    if (task_exists(id))
        remove_task(id)
}

public task_dead_flag_check(id)
{
    if (pev(id, pev_deadflag) == DEAD_DEAD || ++g_juice[id] * DEAD_FLAG_CHECK_INTERVAL > DEAD_FLAG_CHECK_MAX_TIME)
    {
        get_user_origin(id, g_origin[id])

        new Float:mins[3]
        pev(id, pev_mins, mins)
        g_origin[id][2] = g_origin[id][2] + 2 - floatround(fm_distance_to_foothold(id) - mins[2])

        g_juice[id] = get_pcvar_num(g_pcvar_total_juice)

        remove_task(id)
    }
}

public forward_emit_sound(id, channel, sound[])
{
    if (!id || !equali(sound, USE_SOUND) || !is_user_alive(id) || !(pev(id, pev_button) & IN_USE) || task_exists(id) || !is_user_zombie(id))
        return FMRES_IGNORED
    
    if(~get_user_flags(id) & ACCESS) {
        return FMRES_IGNORED
    }

    new Float:health

    pev(id, pev_health, health)
    if (health >= HEALTH_LIMIT)
        return FMRES_IGNORED

    new corpse = can_drink(id)
    if (!corpse)
        return FMRES_IGNORED

    if (g_msgid_bartime2)
    {
        new total = get_pcvar_num(g_pcvar_total_juice)
        msg_bartime2(id, FORMULA_TOTAL_SUCTION_TIME(get_pcvar_num(g_pcvar_suction_amount), total), FORMULA_SUCKED_OUT_PERCENT(g_juice[corpse], total))
    }

    engclient_cmd(id, "say", HUNGRY_PHRASE[random(SUCTION_MESSAGE_NUM)])
    suction_sound(id)

    new param[2]
    param[0] = id
    param[1] = corpse
    set_task(SUCTION_INTERVAL, "task_drink_juice", id, param, 2, "b")

    return FMRES_SUPERCEDE
}

public task_drink_juice(param[2])
{
    new id = param[0]
    new corpse = param[1]

    if (is_user_alive(id) && (pev(id, pev_button) & IN_USE) && can_drink(id, corpse) && is_user_zombie(id))
    {
        new Float:health
        pev(id, pev_health, health)
        if (health < HEALTH_LIMIT)
        {
            suction_sound(id)
            new amount = get_pcvar_num(g_pcvar_suction_amount)
            set_pev(id, pev_health, health + amount)
            g_juice[corpse] -= amount

            return
        }
    }

    if (g_msgid_bartime2)
        msg_bartime2(id, 0, 0)

    remove_task(id)
}

can_drink(id, corpse = 0)
{
    new origin[3], Float:fcorpsepos[3]
    get_user_origin(id, origin)
    for (new i = 1; i <= MAX_PLAYERS; ++i)
    {
        if (g_juice[i] <= 0 || get_distance(origin, g_origin[i]) > MAX_DISTANCE)
            continue

        fm_IVecFVec(g_origin[i], fcorpsepos)
        if (!fm_is_visible(id, fcorpsepos) || get_view_angle_diff(id, fcorpsepos) > MAX_VIEWANGLE_DIFF)
            continue

        if (!corpse || corpse == i)
            return i
    }

    return 0
}

suction_sound(id)
{
    emit_sound(id, CHAN_AUTO, SUCTION_FILE[random(SUCTION_SOUND_NUM)], VOL_NORM, ATTN_NORM, 0, PITCH_NORM)
}

stock fm_IVecFVec(const IVec[3], Float:FVec[3])
{
    FVec[0] = float(IVec[0])
    FVec[1] = float(IVec[1])
    FVec[2] = float(IVec[2])

    return 1
}

stock bool:fm_is_visible(index, const Float:point[3])
{
    new Float:origin[3], Float:view_ofs[3], Float:eyespos[3]
    pev(index, pev_origin, origin)
    pev(index, pev_view_ofs, view_ofs)
    xs_vec_add(origin, view_ofs, eyespos)

    engfunc(EngFunc_TraceLine, eyespos, point, 0, index)

    new Float:fraction
    global_get(glb_trace_fraction, fraction)
    if (fraction == 1.0)
        return true

    return false
}

stock Float:get_view_angle_diff(index, Float:vec_c[3])
{
    new Float:vec_a[3], Float:vec_b[3], viewend[3]
    new Float:origin[3], Float:view_ofs[3]
    pev(index, pev_origin, origin)
    pev(index, pev_view_ofs, view_ofs)
    xs_vec_add(origin, view_ofs, vec_a)

    get_user_origin(index, viewend, 3)
    fm_IVecFVec(viewend, vec_b)

    new Float:a = get_distance_f(vec_b, vec_c)
    new Float:b = get_distance_f(vec_a, vec_c)
    new Float:c = get_distance_f(vec_a, vec_b)

    return floatacos((b*b + c*c - a*a) / (2 * b * c), _:degrees)
}

stock Float:fm_distance_to_foothold(index)
{
    new Float:mins[3], Float:maxs[3]
    pev(index, pev_absmin, mins)
    pev(index, pev_absmax, maxs)

    new Float:start[3]
    start[1] = mins[1]
    start[2] = mins[2] + 10

    new Float:dest[3], Float:end[3]
    dest[1] = mins[1]
    dest[2] = -8191.0

    new index[4] = {0, 0, 1, 0}, Float:value[4]
    value[0] = mins[0]
    value[1] = maxs[0]
    value[2] = maxs[1]
    value[3] = mins[0]
    new Float:ret = -8191.0

    for (new i = 0; i < 4; ++i)
    {
        start[index[i]] = value[i]
        dest[index[i]] = value[i]
        engfunc(EngFunc_TraceLine, start, dest, 0, index)
        global_get(glb_trace_endpos, end)
        if (end[2] > ret)
            ret = end[2]
    }

    ret = mins[2] - ret

    return ret > 0 ? ret : 0.0
}

stock msg_bartime2(index, scale, start_percent)
{
    message_begin(MSG_ONE, g_msgid_bartime2, _, index)
    write_short(scale)
    write_short(start_percent)
    message_end()
}
Хотел перевести на реапи, но не хватает знаний. Надеюсь на подсказки
Что-то я уже поменял, остановился на думках чем заменить register_event("DeathMsg", "event_death", "a")
 
Сообщения
317
Реакции
131
Помог
4 раз(а)
Xelson, ага спасибо, еще убрал инклюд биохазарда и подключил 1 натив. А что на счет engclient_cmd, будет работать на пиратках и стимах?
 
Сообщения
141
Реакции
201
Помог
5 раз(а)
Сообщения
317
Реакции
131
Помог
4 раз(а)
Ну а в общем, нет ничего больше старого или ужасного?
5 Авг 2019
Кажется где-то читал что new id = read_data(2) не есть хорошо
 
Сообщения
212
Реакции
334
Помог
3 раз(а)
read_data() нужен для считывания аргументов клиентских сообщений. Вы больше не хукаете клиентские сообщения, поэтому, этот код не нужен. Надо использовать аргументы, передаваемые ReAPI в обработчик смерти игрока.
 
Сообщения
317
Реакции
131
Помог
4 раз(а)
Верно?
Код:
//public event_death()
public HC_CBasePlayer_Killed_Post(id)
{
    //new id = read_data(2)

    if (task_exists(id))
        remove_task(id)

    set_task(DEAD_FLAG_CHECK_INTERVAL, "task_dead_flag_check", id, _, _, "b")
}
 

Ayk

Сообщения
763
Реакции
479
Помог
19 раз(а)

Пользователи, просматривающие эту тему

Сейчас на форуме нет ни одного пользователя.
Сверху Снизу