Sentry Guns

Сообщения
38
Реакции
-1
Помог
1 раз(а)
Удаляю я значит функции героя, нативы и так далее.
Стояла привязка через натив к герою и я просто ее удалил.
Ибо 4 лвл с ракетами стрелял бы только по герою и по игрокам с невидимостью.
Так вот ракеты работают, но виснет звук выстрела ракет, то бишь он раздается всем и бесконечно.
В чем может быть причина этого?


Код:
/*
#include <amxmodx>
#include <engine>
#include <fun>
#include <cstrike>
#include <fakemeta>
#include <hamsandwich>
#include <xs>

#if AMXX_VERSION_NUM < 183
    #define message_begin_f(%0,%1,%2,%3)    engfunc(EngFunc_MessageBegin,%0,%1,%2,%3)
    #define write_coord_f(%0)        engfunc(EngFunc_WriteCoord,%0)
   
    #include <dhudmessage>
#endif

-native sh_set_hero ( id )
-native sh_get_hero ()

#define is_valid_player(%1) ( 1 <= %1 <= g_iMaxPlayers )
#define is_valid_team(%1) ( 0 < %1 < 3 )

#define is_entity_on_ground(%1) ( entity_get_int ( %1, EV_INT_flags ) & FL_ONGROUND )

#define MAXUPGRADERANGE        75.0
#define SENTRYEXPLODERADIUS        250.0            // радиус отброса при взрыве
#define SENTRYTILTRADIUS            830.0            // likely you won't need to touch this. it's how accurate the cannon will aim at the target vertically (up/down, just for looks, aim is calculated differently)

#define SENTRYMINDISTANCE    256.0

#define MAXSENTRIES 1000

#define TASK_GODMODE 114455

#define SENTRY_INT_TARGET        EV_INT_iuser3
#define SENTRY_TARGET_BITS        6
#define TARGET                    0
#define MASK_TARGET                0xFFFFFFC0 // 11111111111111111111111111000000
new const MASKS_TARGET[1] = {MASK_TARGET}

GetSentryTarget(const SENTRY, const WHO) {
    new data = entity_get_int(SENTRY, SENTRY_INT_TARGET)
    data |= MASKS_TARGET[WHO]
    data ^= MASKS_TARGET[WHO]
    data = (data>>(WHO*SENTRY_TARGET_BITS))
    return data
}
SetSentryTarget(const SENTRY, const WHO, const IS) {
    new data = entity_get_int(SENTRY, SENTRY_INT_TARGET)
    data &= MASKS_TARGET[WHO] // nullify the setting
    data |= (IS<<(WHO*SENTRY_TARGET_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_TARGET, data) // store
}

#define SENTRY_INT_PEOPLE        EV_INT_iuser2 // max 5 users using 6 bits!
#define SENTRY_PEOPLE_BITS        6
#define OWNER                    0
#define UPGRADER_1                1
#define UPGRADER_2                2
#define UPGRADER_3                3
#define UPGRADER_4                4
#define MASK_OWNER                0xFFFFFFC0 // 11111111111111111111111111000000
#define MASK_UPGRADER_1            0xFFFFF03F // 11111111111111111111000000111111
#define MASK_UPGRADER_2            0xFFFC0FFF // 11111111111111000000111111111111
#define MASK_UPGRADER_3            0xFF03FFFF // 11111111000000111111111111111111
#define MASK_UPGRADER_4            0xC0FFFFFF // 11000000111111111111111111111111
new const MASKS_PEOPLE[5] = {MASK_OWNER, MASK_UPGRADER_1, MASK_UPGRADER_2, MASK_UPGRADER_3, MASK_UPGRADER_4}

GetSentryPeople(const SENTRY, const WHO) {
    new data = entity_get_int(SENTRY, SENTRY_INT_PEOPLE)
    data |= MASKS_PEOPLE[WHO]
    data ^= MASKS_PEOPLE[WHO]
    data = (data>>(WHO*SENTRY_PEOPLE_BITS))
    return data
}
SetSentryPeople(const SENTRY, const WHO, const IS) {
    new data = entity_get_int(SENTRY, SENTRY_INT_PEOPLE)
    data &= MASKS_PEOPLE[WHO] // nullify the setting
    data |= (IS<<(WHO*SENTRY_PEOPLE_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_PEOPLE, data) // store
}

#define SENTRY_INT_SETTINGS        EV_INT_iuser1
#define SENTRY_ROCKET_TIME        EV_FL_teleport_time
#define SENTRY_SETTINGS_BITS    3
#define SENTRY_SETTING_FIREMODE    0
#define SENTRY_SETTING_TEAM        1
#define SENTRY_SETTING_LEVEL    2
#define SENTRY_SETTING_PENDDIR    3
#define MASK_FIREMODE            0xFFFFFFF8 // 11111111111111111111111111111000 = FFFFFFFC
#define MASK_TEAM                0xFFFFFFC7 // 11111111111111111111111111000111 = FFFFFFF3
#define MASK_LEVEL                0xFFFFFE3F // 11111111111111111111111000111111 = FFFFFFCF
#define MASK_PENDDIR            0xFFFFF1FF // 11111111111111111111000111111111 = FFFFFF3F
new const MASKS_SETTINGS[4] = {MASK_FIREMODE, MASK_TEAM, MASK_LEVEL, MASK_PENDDIR}

GetSentrySettings(const SENTRY, const SETTING) {
    new data = entity_get_int(SENTRY, SENTRY_INT_SETTINGS)
    data |= MASKS_SETTINGS[SETTING]
    data ^= MASKS_SETTINGS[SETTING]
    //data = (data>>(SETTING*SENTRY_SETTINGS_BITS))
    return (data>>(SETTING*SENTRY_SETTINGS_BITS))
}
SetSentrySettings(const SENTRY, const SETTING, const VALUE) {
    new data = entity_get_int(SENTRY, SENTRY_INT_SETTINGS)
    data &= MASKS_SETTINGS[SETTING] // nullify the setting
    //data |= (VALUE<<(SETTING*SENTRY_SETTINGS_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_SETTINGS, data | (VALUE<<(SETTING*SENTRY_SETTINGS_BITS))) // store
}

GetSentryFiremode(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_FIREMODE)
}
SetSentryFiremode(const SENTRY, const MODE) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_FIREMODE, MODE)
}
CsTeams:GetSentryTeam(const SENTRY) {
    return CsTeams:GetSentrySettings(SENTRY, SENTRY_SETTING_TEAM)
}
SetSentryTeam(const SENTRY, const CsTeams:TEAM) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_TEAM, int:TEAM)
}
GetSentryLevel(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_LEVEL)
}
SetSentryLevel(const SENTRY, const LEVEL) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_LEVEL, LEVEL)
}
GetSentryPenddir(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_PENDDIR)
}
SetSentryPenddir(const SENTRY, const PENDDIR) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_PENDDIR, PENDDIR)
}

#define SENTRY_ENT_BASE            EV_ENT_euser1

#define SENTRY_FL_ANGLE            EV_FL_fuser1
#define SENTRY_FL_SPINSPEED        EV_FL_fuser2
#define SENTRY_FL_MAXSPIN        EV_FL_fuser3
#define SENTRY_FL_LASTTHINK        EV_FL_fuser4

#define SENTRY_DIR_CANNON        0

#define BASE_ENT_SENTRY            EV_ENT_euser1
#define BASE_INT_TEAM            EV_INT_iuser1

#define SENTRY_LEVEL_1            0
#define SENTRY_LEVEL_2            1
#define SENTRY_LEVEL_3            2
#define SENTRY_LEVEL_4            3
#define SENTRY_LEVEL_5            4
#define SENTRY_FIREMODE_NO        0
#define SENTRY_FIREMODE_YES        1
#define SENTRY_FIREMODE_NUTS    2
#define TARGETUPMODIFIER            18.0 // if player ducks on ground, traces don't hit...
#define DMG_BULLET                (1<<1)    //выстрел
#define DMG_BLAST                (1<<6)    // explosive blast damage
#define TE_EXPLFLAG_NONE        0
#define TE_EXPLOSION                3
#define TE_TRACER                6
#define TE_BREAKMODEL            108
#define PENDULUM_MAX            45.0 // how far sentry turret turns in each direction when idle, before turning back
#define PENDULUM_INCREMENT        10.0 // speed of turret turning...
#define SENTRYSHOCKPOWER        3.0 // multiplier, increase to make exploding sentries throw stuff further away
#define CANNONHEIGHTFROMFEET    20.0 // tweakable to make tracer originate from the same height as the sentry's cannon. Also traces rely on this Y-wise offset.
#define PLAYERORIGINHEIGHT        36.0 // this is the distance from a player's EV_VEC_origin to ground, if standing up
#define HEIGHTDIFFERENCEALLOWED    20.0 // increase value to allow building in slopes with higher angles. You can set to 0.0 and you will only be able to build on exact flat ground. note: mostly applies to downhill building, uphill is still likely to "collide" with ground...

#define PLACE_RANGE 45.0

#define SENTRY_RADAR            20 // use as high as possible but should still be working (ie be able to see sentries plotted on radar while in menu, too high values doesn't seem to work)
#define SENTRY_RADAR_TEAMBUILT    21 // same as above

#define RPG_RADIUS    250.0
#define RPG_DAMAGE    150.0
#define RPG_DISTANCE    400.0


new const szModels[][] =
{
    "models/error_csdm/sentry/base.mdl",
    "models/error_csdm/sentry/te_1.mdl",
    "models/error_csdm/sentry/te_2.mdl",
    "models/error_csdm/sentry/te_3.mdl",
   
    "models/error_csdm/sentry/ct_1.mdl",
    "models/error_csdm/sentry/ct_2.mdl",
    "models/error_csdm/sentry/ct_3.mdl",
    "models/error_csdm/sentry/rpgrocket.mdl",
    "models/computergibs.mdl"
}

new const szSounds[][] =
{
    "debris/bustmetal1.wav",
    "debris/bustmetal2.wav",
    "debris/metal1.wav",
    "debris/metal3.wav",
    "error_csdm/sentry/turridle.wav",
    "error_csdm/sentry/turrset.wav",
    "error_csdm/sentry/turrspot.wav",
    "error_csdm/sentry/building.wav",
    "error_csdm/sentry/fire.wav",
    "weapons/rocket1.wav",
    "weapons/tok.wav"
}

new expDecal

#define SENTRYOWNERAWARD    300
#define SENTRYASSISTAWARD    150

#define g_DMG 30 // количество урона от пушки в зависимости от ее уровня
#define g_THINKFREQUENCIES 0.7 // через сколько захватывается цель
#define g_HITRATIOS 0.7 // разброс

new const g_SENTRYCOST[5] = {3500, 4000, 7000, 8000, 9000}                   // стоимость установки
new const g_SENTRYCOST_VIP[5] = {200,300,400,700,800}                // стоимость установки
new const g_COST_VIP[5] = {300, 100, 150, 850, 1000}                // стоимость улучшения пушек VIP
new const Float:g_HEALTHS[5] = {3500.0, 4500.0, 5500.0, 6500.0, 7500.0}    // сколько хп у пушки в зависимости от ее уровня (верхняя часть)
new const g_COST[5] = {3500, 1500, 2500, 10000, 12000}                    // стоимость улучшения пушек

#define g_sentriesNum (g_teamsentriesNum[0]+g_teamsentriesNum[1])
new g_teamsentriesNum[2]  
new g_sentries[MAXSENTRIES]
new g_iPlayerSentries[33]
new g_iPlayerSentriesEdicts[33][5]
new g_sModelIndexFireball
new g_msgDamage
new g_msgDeathMsg
new g_msgScoreInfo
new g_msgHostagePos
new g_msgHostageK
new g_iMaxPlayers
new Float:g_ONEEIGHTYTHROUGHPI
new Float:g_sentryOrigins[32][3]
new bool:g_inBuilding[33]
new sentries_num[33]
new gMsgID
- new g_iTimer[33]
new m_iTrail
new g_Sprt_Tok
enum _:menuStatus {
    CURRENT_SENTRYID
}
new g_MenuStatus[33][menuStatus]
#define SentryClassName "sentry"
#define SentryBaseClassName "sentrybase"
new attach_viewmod[33]

new g_FlagVip
new g_Nagrada
new g_Count_sentry_vip, g_Count_sentry


public plugin_init() {

    register_plugin("Sentry Guns", "1.2", "pro100web")

    register_event ( "Spectator", "ev_Spectation", "a" )

    register_clcmd("sentry_build", "cmd_CreateSentry", 0, "- build a sentry gun where you are")
   
    RegisterHam ( Ham_Spawn, "player", "fw_PlayerSpawn_Post", 1 )

    register_forward ( FM_TraceLine, "fw_TraceLine_Post", 1 )

    RegisterHam ( Ham_TakeDamage, "func_breakable", "fw_TakeDamage" )
   
    register_touch ( SentryClassName, "player", "fw_TouchSentry" )
    register_touch("rpg_rocket","*","fw_RpgTouch")
   
    register_message ( 23, "msg_TempEntity" )
    register_think("sentrybase", "think_sentrybase")
    register_think ( SentryClassName, "fw_ThinkSentry" )

    g_msgDamage = get_user_msgid("Damage")
    g_msgDeathMsg = get_user_msgid("DeathMsg")
    g_msgScoreInfo = get_user_msgid("ScoreInfo")
    g_msgHostagePos = get_user_msgid("HostagePos")
    g_msgHostageK = get_user_msgid("HostageK")
    gMsgID = get_user_msgid("StatusIcon")
   
    g_iMaxPlayers = get_global_int(GL_maxClients)
    g_ONEEIGHTYTHROUGHPI = 180.0 / 3.141592654
   
    expDecal = get_decal_index("{scorch1")
    register_clcmd("sentry_menu", "sentry_menu")
    register_clcmd("say /off", "offview")
    - set_task ( 120.0, "checkhero", .flags = "b" )
   
    set_task(0.2,"informer",_,_,_,"b");
   
    g_FlagVip = register_cvar("er_sentry_flag_vip", "t")                        // Флаг для vip игрока.
    g_Count_sentry_vip = register_cvar("er_sentry_max_vip", "2")                // Сколько пушек может поставить VIP игрок с флагом выше (er_sentry_flag_vip)
    g_Count_sentry = register_cvar("er_sentry_max", "1")                      // Сколько пушек может поставить простой игрок
    g_Nagrada = register_cvar("er_sentry_nagrada", "2000")                        // Награда за разрушение пушки
}

public offview(id)
{
    attach_view(id,id)
    attach_viewmod[id] = 0
}

public sentry_menu(id)
{
    new szItem[128];
    new szItem1[128];
    new msentry_item = menu_create("\wНастройка пушек!", "cl_men")
    //format(szItem1, 127, "\wМодель пушки: \y%s", SzModelSentry[modelSentryVID[id]])
    //menu_additem( msentry_item, szItem1, "1", 0)
    //format(szItem1, 127, "\wСкин пушки: \y%s", SzSkinsSentry[skinSentryVID[id]])
    //menu_additem( msentry_item, szItem1, "2", 0)
    if(sentries_num[id] > 0){
        static ent = -1
        while ((ent = find_ent_by_class(ent, SentryClassName)))
        {      
            if( GetSentryPeople(ent,OWNER) != id) continue;
                   
            if(is_valid_ent(ent))
            {
                format(szItem, 127, "%d", ent)
                new sClassName[11]
                pev ( ent, pev_classname, sClassName, charsmax ( sClassName ) )
                if ( equal ( sClassName, SentryClassName ) )
                {
                    new Float:health = entity_get_float ( ent, EV_FL_health )
                    new level = GetSentryLevel ( ent )
                    format(szItem1, 127, "\wID %d | LVL %d HP %i", ent, level+1, floatround(health))
                    menu_additem( msentry_item, szItem1, szItem, 0)
                }
            }
        }
    }
    menu_setprop(msentry_item, MPROP_EXIT, MEXIT_ALL)
    menu_setprop(msentry_item, MPROP_EXITNAME, "\yВыход")
    menu_display(id, msentry_item, 0)
       
    return PLUGIN_HANDLED
}

public cl_men(id, menu, item) {
    new s_Data[6], s_Name[64], i_Access, i_Callback
    menu_item_getinfo(menu, item, i_Access, s_Data, charsmax(s_Data), s_Name, charsmax(s_Name), i_Callback)
    new i_Key = str_to_num(s_Data)
    menu_destroy(menu)
    switch(i_Key){
        case 0: return 0
        /* case 1: {
            switch(modelSentryVID[id]){
                case 0: modelSentryVID[id] = 1
                case 1: modelSentryVID[id] = 2
                case 2: modelSentryVID[id] = 0
            }
            ChatColor ( id, "^4Вы успешно выбрали модель")
            sentry_menu(id)
        }
        case 2: {
            switch(skinSentryVID[id]){
                case 0: skinSentryVID[id] = 1
                case 1: skinSentryVID[id] = 2
                case 2: skinSentryVID[id] = 0
            }
            ChatColor ( id, "^4Вы успешно выбрали скин")
            sentry_menu(id)
        } */
        default: {
            if ( !is_valid_ent ( i_Key ) ) return 0
            g_MenuStatus[id][CURRENT_SENTRYID] = i_Key
            sentry_menu_func(id)
        }
    }
    return 1
}

public sentry_menu_func(id)
{
    if(sentries_num[id] == 0){
        ChatColor ( id, "^4У тебя нет пушек, что ты хотел этим добится?")
        return HAM_IGNORED
    }
    new szItem1[128];
    new ent = g_MenuStatus[id][CURRENT_SENTRYID];
    if(ent==0) return PLUGIN_HANDLED
   
    if(is_valid_ent(ent))
    {
        new sClassName[11]
        pev ( ent, pev_classname, sClassName, charsmax ( sClassName ) )
        if ( equal ( sClassName, SentryClassName ) )
        {
            new msentry_item = menu_create("\yВыберите пушку!", "cl_menr");
            new Float:health = entity_get_float ( ent, EV_FL_health )
            new level = GetSentryLevel ( ent )
            menu_additem( msentry_item, "\wУничтожить", "1")
            format(szItem1, 127, "\wСмотреть от лица пушки", "3")
            menu_additem( msentry_item, szItem1, "3", 0)
            menu_addblank( msentry_item, 0)
            menu_addblank( msentry_item, 0)
            menu_addblank( msentry_item, 0)
            format(szItem1, 127, "\wID \r%d", ent)
            menu_addtext( msentry_item, szItem1)
            format(szItem1, 127, "Уровень \r%d", level+1)
            menu_addtext( msentry_item, szItem1)
            format(szItem1, 127, "\w[HP \y%d\w]", floatround(health))
            menu_addtext( msentry_item, szItem1)
            menu_setprop(msentry_item, MPROP_EXIT, MEXIT_ALL)
            menu_setprop(msentry_item, MPROP_EXITNAME, "\yВыход")
            menu_display(id, msentry_item, 0)
        }
    }
    return PLUGIN_HANDLED
}

public cl_menr(id, menu, item) {
    new s_Data[6], s_Name[64], i_Access, i_Callback
    menu_item_getinfo(menu, item, i_Access, s_Data, charsmax(s_Data), s_Name, charsmax(s_Name), i_Callback)
    new i_Key = str_to_num(s_Data)
    menu_destroy(menu)
    switch(i_Key){
        case 0: return 1;
        case 1:{
            if ( !is_valid_ent ( g_MenuStatus[id][CURRENT_SENTRYID] ) ) return 0
            ChatColor(id,"Пушка: %d уничтожена",g_MenuStatus[id][CURRENT_SENTRYID])
            sentry_detonate ( g_MenuStatus[id][CURRENT_SENTRYID], false, false )
        }
        /* case 2:{
            if ( !is_valid_ent ( g_MenuStatus[id][CURRENT_SENTRYID] ) ) return 0
            if(cs_get_user_money(id)<9000){
                ChatColor(id,"У вас недостаточно средств. Пушка: %d",g_MenuStatus[id][CURRENT_SENTRYID])
                return 0
            }
            if(SentryPatrWhit[g_MenuStatus[id][CURRENT_SENTRYID]]+3000<8000){
                ChatColor(id,"Пушка: %d Закуплены патроны ",g_MenuStatus[id][CURRENT_SENTRYID])
                SentryPatrWhit[g_MenuStatus[id][CURRENT_SENTRYID]]+=3000
            } else {
                ChatColor(id,"В пушке не может быть более 8000 патронов. Пушка: %d",g_MenuStatus[id][CURRENT_SENTRYID])
            }
        } */
        case 3: {
            attach_viewEx(id, g_MenuStatus[id][CURRENT_SENTRYID])
            set_task ( 15.0, "DelayRemoveAttachView", id-9250)
        }
    }
    g_MenuStatus[id][CURRENT_SENTRYID] = 0
    return 1
}
public DelayRemoveAttachView(idoff){
    new id = idoff+9250
    attach_viewEx(id,id)
}
stock attach_viewEx(id,target){
    attach_view(id,target)
    if(id == target) attach_viewmod[id] = 0, remove_task(id-100)
    else attach_viewmod[id] = target, set_task ( 1.0, "set_taskdhudmessagebg", id-100, .flags = "b" )
}

public set_taskdhudmessagebg(idoff){
    new id = idoff+100
    if(!attach_viewmod[id]) remove_task(idoff)
    new Float:health = entity_get_float ( attach_viewmod[id], EV_FL_health )
    set_dhudmessage ( 255, 255, 255, 0.0, 0.75, 0, 0.0, 0.99, 0.0, 0.0 )
    static tempStatusBuffer[192]
    new level = GetSentryLevel ( attach_viewmod[id] )
    new OwnName[33]
    get_user_name ( GetSentryPeople ( attach_viewmod[id], OWNER ), OwnName, 32 )
    formatex ( tempStatusBuffer, charsmax ( tempStatusBuffer ), "Установил: %s^nЗдоровье: %d/%d^nsay /off", OwnName, floatround(health), floatround(g_HEALTHS[level]))
    show_dhudmessage(id, tempStatusBuffer)
}

public plugin_precache() {
    for(new i=0;i<sizeof(szModels);i++)
        precache_model(szModels[i])
    for(new i=0;i<sizeof(szSounds);i++)
        precache_sound(szSounds[i])
    g_sModelIndexFireball = precache_model("sprites/zerogxplode.spr")
    m_iTrail = precache_model("sprites/smoke.spr")
    g_Sprt_Tok = precache_model("sprites/tok.spr")
}

public plugin_natives ()
{
    register_native ( "get_sentry_people", "native_get_sentry_people", 1 )
    register_native ( "get_sentry_team", "native_get_sentry_team", 1 )
    register_native ( "get_SentryCount", "nativegetSentryCount", 1)
   
    register_native ( "get_SentryLevel", "nativeget_SentryLevel", 1)
    register_native ( "set_freezSentry", "native_CreateSentry", 1)
    register_native ( "set_sentrGrand", "native_CreateSentry", 1)
    register_native ( "native_create_sentry", "native_CreateSentry", 1)
    register_native ( "get_leader", "native_get_leader", 1)
}

public native_get_leader() return 1
public native_CreateSentry(id) return 0

public nativeget_SentryLevel(sentry)
{
    return GetSentryLevel ( sentry )
}

public nativegetSentryCount(id)
{
    return sentries_num[id];
}

public native_get_sentry_people ( sentry, who )
{
    return GetSentryPeople ( sentry, who )
}

public CsTeams:native_get_sentry_team ( sentry )
{
    return GetSentryTeam ( sentry )
}

public informer(id)
{
        set_dhudmessage(255, 0, 0, 0.41, 0.0, 0 , 0.0, 0.1,0.1,0.1);
        show_dhudmessage(id, "Пушек ТТ [%d] ",g_teamsentriesNum[0]);
       
        set_dhudmessage(29, 224, 215, 0.54, 0.0, 0, 0.0, 0.1,0.1,0.1);
        show_dhudmessage(id, "[%d] Пушек КТ",g_teamsentriesNum[1]);
       
}

public checkhero()
{
    //for ( new i = 1; i <= 500; i++ ) menu_destroy(i)
    if ( sh_get_hero () ) return
    new iDifference = g_teamsentriesNum[0]-g_teamsentriesNum[1]
    if(iDifference>=8||iDifference<=-8)
    {
        new players[32],  num, plid, bestfrags = 0, bool:heroenabled
        new iData[2]; iData[0] = 5
        if(g_teamsentriesNum[1] <= (g_sentriesNum * 0.3))
        {
            iData[1] = 1
            TimerStart ( iData )
            get_players(players, num, "ache", "CT")
            heroenabled = true
        } else if(g_teamsentriesNum[0] <= (g_sentriesNum * 0.3)) {
            iData[1] = 0
            TimerStart ( iData )
            get_players(players, num, "ache", "TERRORIST")  
            heroenabled = true
        }
       
        if(heroenabled && num > 1)
        {
            for(--num; num>=0; num--)
            {
                if(get_user_frags(players[num]) >= bestfrags)
                {
                    plid = players[num]
                    bestfrags = get_user_frags(plid)
                }
            }

           
            if ( sh_set_hero ( plid ) )
            {
                sentry_detonate_by_owner ( plid )
            }
           
        }
    }
}

public TimerStart ( data[2] )
{
    if ( !data[0] ) return

    data[0]--

    set_dhudmessage ( 255, 255, 255, -1.0, 0.35, 0, 0.0, 1.0, 0.0, 0.0 )
    show_dhudmessage ( 0, "Выдача гранаты-ауры %s начнется через %d секунд", data[1] ? "контрам" : "террам", data[0] )

    set_task ( 1.0, "TimerStart", 0, data, 2 )
}

public TimerGodMode ( id )
{
    if ( id > g_iMaxPlayers )
        id -= TASK_GODMODE

    if ( !is_user_connected ( id ) || !is_user_alive ( id ) )
        return

    set_dhudmessage ( 255, 255, 255, -1.0, 0.35, 0, 0.0, 1.0, 0.0, 0.0 )

    if ( --g_iTimer[id] > 0 )
    {
        show_dhudmessage ( id, "У вас еще есть %i секунд бессмертия^nГраната аура - это дымовая граната", g_iTimer[id] )
        set_task ( 1.0, "SetGodMode", id+TASK_GODMODE )
    }
    else
    {
        show_dhudmessage ( id, "Режим бессмертия отключен" )
        set_user_godmode ( id, 0 )
    }
}

public ev_Spectation ()
{
    new id = read_data ( 1 )

    if ( is_user_connected ( id ) && cs_get_user_team ( id ) == CS_TEAM_SPECTATOR )
        sentry_detonate_by_owner ( id )
}

public fw_TakeDamage ( ent, idinflictor, idattacker, Float:damage, damagebits )
{
    if ( !is_valid_ent ( ent ) )
        return HAM_IGNORED
   
    new sClassname[11]
    pev ( ent, pev_classname, sClassname, charsmax ( sClassname ) )
   
    if ( equal ( sClassname, SentryClassName ) || equal ( sClassname, "sentrybase" ) )
    {
        if ( sClassname[6] == 'b' )
            ent = entity_get_edict(ent, BASE_ENT_SENTRY)
       
        if ( is_valid_ent ( ent ) )
        {
            new iOwner = GetSentryPeople ( ent, OWNER )
           
            if ( !is_user_connected ( iOwner ) || !is_valid_player ( iOwner ) || !is_user_connected ( idattacker ) || !is_valid_player ( idattacker ) )
                return HAM_SUPERCEDE
       
            if ( cs_get_user_team ( iOwner ) == cs_get_user_team ( idattacker ) && idattacker != iOwner )
                return HAM_SUPERCEDE
           
            if ( pev ( ent, pev_health ) - damage <= 0.0 )
            {
                ChatColor ( idattacker, "^3[^4Информация^3]^1 Вы получили ^4%i^1$ за разрушение пушки", get_pcvar_num(g_Nagrada))
                cs_set_user_money ( idattacker, cs_get_user_money ( idattacker ) + get_pcvar_num(g_Nagrada))
            }
            cs_set_user_money(idattacker,cs_get_user_money(idattacker) + (floatround(damage)-10))
        }
    }
    return HAM_IGNORED  
}

public cmd_CreateSentry ( id )
{
    new iSentry = AimingAtSentry ( id )

    if ( iSentry && entity_range ( iSentry, id ) <= MAXUPGRADERANGE )
        SentryUpgrade ( id, iSentry )
    else
        SentryBuild ( id )

    return PLUGIN_HANDLED
}

public SentryBuild ( id )
{
    if ( !is_user_alive ( id ) )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Мертвым нельзя ставить пушку!" )
        return
    }
    new iSentryCount = sentries_num[id]
    if (get_user_flags(id) & get_pcvar_flags(g_FlagVip))
    {
        if ( iSentryCount >= get_pcvar_num(g_Count_sentry_vip))
        {
            ChatColor ( id, "^3[^4Информация^3]^1 Увы, ты уже построил %d пушки!", get_pcvar_num(g_Count_sentry_vip))
            return
        }
    }
    else
    {
        if ( iSentryCount >= get_pcvar_num(g_Count_sentry))
        {
            ChatColor ( id, "^3[^4Информация^3]^1 Нельзя установить более %d пушки!", get_pcvar_num(g_Count_sentry))
            return
        }
    }
    if ( g_inBuilding[id] )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Эй, не так быстро..." )
        return
    }
    if ( !is_entity_on_ground ( id ) )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Встань на землю, чтобы установить пушку!" )
        return
    }
   
    if (get_user_flags(id) & get_pcvar_flags(g_FlagVip))
    {  
        if ( cs_get_user_money ( id ) < g_SENTRYCOST_VIP[iSentryCount] )
        {
            client_print(id, print_center, "У тебя не хватает денег! (нужно %d$)", g_SENTRYCOST_VIP[iSentryCount] )
            return
        }
    }
    else
    {  
        if ( cs_get_user_money ( id ) < g_SENTRYCOST[iSentryCount] )
        {
            client_print(id, print_center, "У тебя не хватает денег! (нужно %d$)", g_SENTRYCOST[iSentryCount] )
            return
        }
    }
    new Float:fPlayerOrigin[3], Float:fOrigin[3], Float:fAngle[3]
    pev ( id, pev_origin, fPlayerOrigin )
    pev ( id, pev_angles, fAngle )
    fOrigin = fPlayerOrigin

    fOrigin[0] += floatcos ( fAngle[1], degrees ) * PLACE_RANGE
    fOrigin[1] += floatsin ( fAngle[1], degrees ) * PLACE_RANGE
    fOrigin[0] += floatcos ( fAngle[0], degrees) * PLACE_RANGE
    fOrigin[1] += floatcos ( fAngle[1], degrees )
    fOrigin[0] -= floatsin ( fAngle[1], degrees )
    fOrigin[1] += floatcos ( fAngle[2], degrees )
    fOrigin[1] -= floatsin ( fAngle[2], degrees ) * PLACE_RANGE
    fOrigin[0] -= floatsin ( fAngle[0], degrees ) * PLACE_RANGE
    fOrigin[0] -= PLACE_RANGE

    if ( pev ( id, pev_flags ) & FL_DUCKING )
        fOrigin[2] += 18.0, fPlayerOrigin[2] += 18.0

    new tr = 0, Float:fFraction
    engfunc ( EngFunc_TraceLine, fPlayerOrigin, fOrigin, 0, id, tr )
    get_tr2 ( tr, TR_flFraction, fFraction )

    if ( fFraction != 1.0 )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Здесь не получается установить пушку!" )
        return
    }

    if ( CreateSentryBase ( fOrigin, id ) )
    {
        if (get_user_flags(id) & get_pcvar_flags(g_FlagVip))
        {
            cs_set_user_money ( id, cs_get_user_money ( id ) - g_SENTRYCOST_VIP[iSentryCount] )
        }
        else
        {  
            cs_set_user_money ( id, cs_get_user_money ( id ) - g_SENTRYCOST[iSentryCount] )
        }
        ammo_hud ( id, 0 )
        sentries_num[id] += 1
        ammo_hud ( id, 1 )
    }
    else
    {
        //ChatColor ( id, "^3[^4Информация^3]^1 Пушка не ставится! %f %f %f",fOrigin[0],fOrigin[1],fOrigin[2] )
        ChatColor ( id, "^3[^4Информация^3]^1 Пушка не ставится!" )
    }
}

IncreaseSentryCount ( id, sentry )
{
    g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id]] = sentry
    g_iPlayerSentries[id]++
    new Float:fSentryOrigin[3], iSentryOrigin[3], iPlayerOrigin[3]
    entity_get_vector ( sentry, EV_VEC_origin, fSentryOrigin )
    FVecIVec ( fSentryOrigin, iSentryOrigin )

    new sName[32]
    get_user_name ( id, sName, charsmax ( sName ) )
    new CsTeams:iTeam = cs_get_user_team ( id )

    for ( new i = 1; i <= g_iMaxPlayers; i++ )
    {
        if ( !is_user_connected ( i ) || !is_user_alive ( i ) || cs_get_user_team ( i ) != iTeam || id == i )
            continue

        get_user_origin ( i, iPlayerOrigin )

        client_print ( i, print_center, "%s установил пушку в %d юнитах от вас", sName, get_distance ( iPlayerOrigin, iSentryOrigin ) )

        message_begin ( MSG_ONE_UNRELIABLE, g_msgHostagePos, .player = i )
        write_byte ( i )
        write_byte ( SENTRY_RADAR_TEAMBUILT )
        write_coord ( iSentryOrigin[0] )
        write_coord ( iSentryOrigin[1] )
        write_coord ( iSentryOrigin[2] )
        message_end ()

        message_begin ( MSG_ONE_UNRELIABLE, g_msgHostageK, .player = i )
        write_byte ( SENTRY_RADAR_TEAMBUILT )
        message_end ()
    }
}

DecreaseSentryCount ( id, sentry )
{
    for ( new i; i < g_iPlayerSentries[id]; i++ )
    {
        if ( g_iPlayerSentriesEdicts[id][i] == sentry )
        {
            g_iPlayerSentriesEdicts[id][i] = g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id] - 1]
            g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id] - 1] = 0
            break
        }
    }
    g_iPlayerSentries[id]--
}

stock bool:CreateSentryBase ( Float:origin[3], creator, level = SENTRY_LEVEL_1 )
{
    if ( !CheckLocation ( origin ) )
        return false

    new Float:hitPoint[3], Float:originDown[3]
    originDown = origin
    originDown[2] = -5000.0 // dunno the lowest possible height...
    trace_line(0, origin, originDown, hitPoint)
    new Float:baDistanceFromGround = vector_distance(origin, hitPoint)

    new Float:difference = PLAYERORIGINHEIGHT - baDistanceFromGround
    if (difference < -1 * HEIGHTDIFFERENCEALLOWED || difference > HEIGHTDIFFERENCEALLOWED) return false

    new entbase = create_entity("func_breakable") // func_wall
    if (!entbase)
        return false

    #define SIZE 10.0

    new Float:fTraceEnds[5][3], Float:fTraceHit[3], iType, tr = create_tr2 ()
    fTraceEnds[0][0] = origin[0] - SIZE
    fTraceEnds[0][1] = origin[1] - SIZE
    fTraceEnds[0][2] = origin[2] + SIZE + SIZE
    fTraceEnds[1][0] = origin[0] + SIZE
    fTraceEnds[1][1] = origin[1] - SIZE
    fTraceEnds[1][2] = origin[2] + SIZE + SIZE
    fTraceEnds[2][0] = origin[0] - SIZE
    fTraceEnds[2][1] = origin[1] + SIZE
    fTraceEnds[2][2] = origin[2] + SIZE + SIZE
    fTraceEnds[3][0] = origin[0] + SIZE
    fTraceEnds[3][1] = origin[1] + SIZE
    fTraceEnds[3][2] = origin[2] + SIZE + SIZE
    fTraceEnds[4][0] = origin[0]
    fTraceEnds[4][1] = origin[1]
    fTraceEnds[4][2] = origin[2] + SIZE + SIZE

    for ( new i; i < 5; i++ )
    {
        fTraceHit = fTraceEnds[i]
        fTraceHit[2] += 40.0

        engfunc ( EngFunc_TraceLine, fTraceEnds[i], fTraceHit, 0, 0, tr )
        get_tr2 ( tr, TR_vecEndPos, fTraceHit )

        if ( fTraceHit[2] - fTraceEnds[i][2] != 40.0 )
        {
            iType = 1
            break
        }
    }

    if ( iType )
    {
        fTraceEnds[0][0] = origin[0] - SIZE
        fTraceEnds[0][1] = origin[1] - SIZE
        fTraceEnds[0][2] = origin[2] - SIZE - SIZE
        fTraceEnds[1][0] = origin[0] + SIZE
        fTraceEnds[1][1] = origin[1] - SIZE
        fTraceEnds[1][2] = origin[2] - SIZE - SIZE
        fTraceEnds[2][0] = origin[0] - SIZE
        fTraceEnds[2][1] = origin[1] + SIZE
        fTraceEnds[2][2] = origin[2] - SIZE - SIZE
        fTraceEnds[3][0] = origin[0] + SIZE
        fTraceEnds[3][1] = origin[1] + SIZE
        fTraceEnds[3][2] = origin[2] - SIZE - SIZE
        fTraceEnds[4][0] = origin[0]
        fTraceEnds[4][1] = origin[1]
        fTraceEnds[4][2] = origin[2] - SIZE - SIZE
        new Float:fMinDistance, Float:fDistance
        for ( new i; i < 5; i++ )
        {
            fTraceHit[0] = fTraceEnds[i][0]
            fTraceHit[1] = fTraceEnds[i][1]
            fTraceHit[2] = -8192.0
   
            engfunc ( EngFunc_TraceLine, fTraceEnds[i], fTraceHit, IGNORE_MONSTERS, 0, tr )
            get_tr2 ( tr, TR_vecEndPos, fTraceHit )
   
            fDistance = vector_distance ( fTraceEnds[i], fTraceHit )
   
            if ( fDistance < fMinDistance || fMinDistance <= 0.0 )
            {
                fMinDistance = fDistance
                origin[2] = fTraceHit[2]
            }
        }
    }
   
    new Float:fHighest[3]
    fHighest = origin
    fHighest[2] = 1000.0
    engfunc ( EngFunc_TraceLine, origin, fHighest, DONT_IGNORE_MONSTERS, 0, tr )
    get_tr2 ( tr, TR_vecEndPos, fHighest )
    free_tr2 ( tr )

    new healthstring[16]
    num_to_str(floatround(g_HEALTHS[0]), healthstring, 15)
    DispatchKeyValue(entbase, "health", healthstring)
    DispatchKeyValue(entbase, "material", "6")

    DispatchSpawn(entbase)
    entity_set_string(entbase, EV_SZ_classname, SentryBaseClassName)
    entity_set_model(entbase, "models/error_csdm/sentry/base.mdl") // later set according to level
    SetSentryPeople ( entbase, OWNER, creator )
   
    new Float:mins[3], Float:maxs[3]
    mins[0] = -16.0
    mins[1] = -16.0
    mins[2] = 0.0
    maxs[0] = 16.0
    maxs[1] = 16.0
    maxs[2] = floatclamp ( vector_distance ( origin, fHighest ), 128.0, 1000.0 ) // Set to 16.0 later.
    entity_set_size(entbase, mins, maxs)
    entity_set_origin(entbase, origin)
   
    entity_set_int(entbase, EV_INT_solid, SOLID_SLIDEBOX)
    entity_set_int(entbase, EV_INT_movetype, iType ? MOVETYPE_FLY : MOVETYPE_TOSS) // head flies base falls
    entity_set_int(entbase, BASE_INT_TEAM, _:cs_get_user_team(creator))

    new parms[4]
    parms[0] = entbase
    parms[1] = creator
    parms[2] = level
    parms[3] = iType

    if ( iType ) origin[2] += 16.0

    g_sentryOrigins[creator - 1] = origin
   
    entity_get_vector(creator, EV_VEC_angles, origin)
    origin[0] = 0.0
    entity_set_vector(entbase, EV_VEC_angles, origin)
   
    emit_sound(creator, CHAN_AUTO, "error_csdm/sentry/building.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)

    set_task(0.7, "createsentryhead", 0, parms, 4)
    g_inBuilding[creator] = true

    return true
}

public createsentryhead(parms[4])
{
    new entbase = parms[0]
    new level = parms[2]
    new creator = parms[1]
    new iType = parms[3]

    if ( !is_user_connected ( creator ) || !g_inBuilding[creator] )
    {
        if (is_valid_ent(entbase))
            remove_entity(entbase + 9287)
        sentries_num[creator]--
        return
    }
    new CsTeams:crteam = cs_get_user_team(creator)
    if ( !is_valid_team ( _:crteam ) )
    {
        if (is_valid_ent(entbase))
            remove_entity(entbase + 9287)
        sentries_num[creator]--
        return
    }

    new Float:origin[3]
    origin = g_sentryOrigins[creator - 1]

    new ent = create_entity("func_breakable")
    if (!ent)
    {
        if (is_valid_ent(entbase))
        {
            remove_entity(entbase + 9287)
        }
        return
    }

    new Float:mins[3], Float:maxs[3]
    if (is_valid_ent(entbase)) {
        mins[0] = -5.0
        mins[1] = -5.0
        mins[2] = 0.0
       
        maxs[0] = 5.0
        maxs[1] = 5.0
        maxs[2] = 16.0
        entity_set_size(entbase, mins, maxs)

        entity_set_edict(ent, SENTRY_ENT_BASE, entbase)
        entity_set_edict(entbase, BASE_ENT_SENTRY, ent)
    }
    new healthstring[16]
    num_to_str(floatround(g_HEALTHS[0]), healthstring, 15)
    DispatchKeyValue(ent, "health", healthstring)
    DispatchKeyValue(ent, "material", "6")

    DispatchSpawn(ent)
    entity_set_string(ent, EV_SZ_classname, SentryClassName)
    switch(_:crteam)
    {
        case 1:
        {
            entity_set_model(ent, "models/error_csdm/sentry/te_1.mdl")
        }
        case 2:
        {
            entity_set_model(ent, "models/error_csdm/sentry/ct_1.mdl")
        }
    }
   
   
    mins[0] = -10.0
    mins[1] = -10.0
    mins[2] = 0.0
    maxs[0] = 10.0
    maxs[1] = 10.0
    maxs[2] = 48.0
    entity_set_size(ent, mins, maxs)
    entity_set_origin(ent, origin)
    entity_get_vector(creator, EV_VEC_angles, origin)
    origin[0] = 0.0
    origin[1] += 180.0
    entity_set_float(ent, SENTRY_FL_ANGLE, origin[1])
    origin[2] = 0.0
    entity_set_vector(ent, EV_VEC_angles, origin)
    entity_set_int(ent, EV_INT_solid, SOLID_SLIDEBOX) // SOLID_SLIDEBOX
    entity_set_int(ent, EV_INT_movetype, iType ? MOVETYPE_FLY : MOVETYPE_TOSS) // head flies, base doesn't

    SetSentryPeople(ent, OWNER, creator)
   
    SetSentryTeam ( ent, crteam )
    SetSentryLevel ( ent, level )

    g_teamsentriesNum[_:crteam-1]++
   
    emit_sound(ent, CHAN_AUTO, "error_csdm/sentry/turrset.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)

    IncreaseSentryCount(creator, ent)

    new directions = (random_num(0, 1)<<SENTRY_DIR_CANNON)
    SetSentryPenddir ( ent, directions )

    g_inBuilding[creator] = false

    if (!is_valid_ent(entbase))
        SetSentryFiremode ( ent, SENTRY_FIREMODE_NUTS )

    entity_set_float ( ent, SENTRY_FL_LASTTHINK, get_gametime () + g_THINKFREQUENCIES )
    entity_set_float ( ent, EV_FL_nextthink, get_gametime () + 0.01 )

    /*static bool:bHamRegistred

    if ( !bHamRegistred )
    {
        RegisterHamFromEntity ( Ham_Think, ent, "fw_ThinkSentry", 1 )
        bHamRegistred = true
    }*/
}

stock bool:CheckLocation ( const Float:origin[3] )
{
    if ( engfunc ( EngFunc_PointContents, origin ) != CONTENTS_EMPTY )
        return false

    new tr = create_tr2 ()

    engfunc ( EngFunc_TraceHull, origin, origin, 0, HULL_HEAD/*HUMAN*/, 0, tr )
    if ( !get_tr2 ( tr, TR_InOpen ) || get_tr2 ( tr, TR_StartSolid ) || get_tr2 ( tr, TR_AllSolid ) )
    {
        free_tr2 ( tr )
        return false
    }

    #define SIZE 10.0

    new Float:fTraceEnds[9][3], Float:fTraceHit[3], iHitEnt
    fTraceEnds[0][0] = origin[0]
    fTraceEnds[0][1] = origin[1]
    fTraceEnds[0][2] = origin[2] - SIZE - SIZE
    fTraceEnds[1][0] = origin[0] - SIZE
    fTraceEnds[1][1] = origin[1] - SIZE
    fTraceEnds[1][2] = origin[2] - SIZE - SIZE
    fTraceEnds[2][0] = origin[0] + SIZE
    fTraceEnds[2][1] = origin[1] - SIZE
    fTraceEnds[2][2] = origin[2] - SIZE - SIZE
    fTraceEnds[3][0] = origin[0] - SIZE
    fTraceEnds[3][1] = origin[1] + SIZE
    fTraceEnds[3][2] = origin[2] - SIZE - SIZE
    fTraceEnds[4][0] = origin[0] + SIZE
    fTraceEnds[4][1] = origin[1] + SIZE
    fTraceEnds[4][2] = origin[2] - SIZE - SIZE
    fTraceEnds[5][0] = origin[0] - SIZE
    fTraceEnds[5][1] = origin[1] - SIZE
    fTraceEnds[5][2] = origin[2] + SIZE + SIZE
    fTraceEnds[6][0] = origin[0] + SIZE
    fTraceEnds[6][1] = origin[1] - SIZE
    fTraceEnds[6][2] = origin[2] + SIZE + SIZE
    fTraceEnds[7][0] = origin[0] - SIZE
    fTraceEnds[7][1] = origin[1] + SIZE
    fTraceEnds[7][2] = origin[2] + SIZE + SIZE
    fTraceEnds[8][0] = origin[0] + SIZE
    fTraceEnds[8][1] = origin[1] + SIZE
    fTraceEnds[8][2] = origin[2] + SIZE + SIZE

    for (new i = 0, b = 0; i < 9; i++)
    {
        if ( engfunc ( EngFunc_PointContents, fTraceEnds[i] ) != CONTENTS_EMPTY )
        {
            free_tr2 ( tr )
            return false
        }

        engfunc ( EngFunc_TraceLine, origin, fTraceEnds[i], 0, 0, tr )
        iHitEnt = get_tr2 ( tr, TR_pHit )

        if ( iHitEnt != -1 )
        {
            free_tr2 ( tr )
            return false
        }

        get_tr2 ( tr, TR_vecEndPos, fTraceHit )

        for ( b = 0; b < 3; b++ )
        {
            if ( fTraceEnds[i][b] != fTraceHit[b] )
            {
                free_tr2 ( tr )
                return false
            }
        }
    }
    free_tr2 ( tr )
    return true
}

bool:sentry_pendulum ( sentry )
{
    switch ( GetSentryFiremode ( sentry ) )
    {
        case SENTRY_FIREMODE_NO:
        {
            new Float:fAngles[3]
            entity_get_vector ( sentry, EV_VEC_angles, fAngles )
            new Float:fBaseAngle = entity_get_float ( sentry, SENTRY_FL_ANGLE )
            new iDirections = GetSentryPenddir ( sentry )
           
            if ( iDirections & (1<<SENTRY_DIR_CANNON) )
            {
                fAngles[1] -= ( PENDULUM_INCREMENT * 0.01 )
                if ( fAngles[1] < fBaseAngle - PENDULUM_MAX )
                {
                    fAngles[1] = fBaseAngle - PENDULUM_MAX
                    iDirections &= ~(1<<SENTRY_DIR_CANNON)
                    SetSentryPenddir ( sentry, iDirections )
                }
            }
            else
            {
                fAngles[1] += ( PENDULUM_INCREMENT * 0.01 )
                if ( fAngles[1] > fBaseAngle + PENDULUM_MAX )
                {
                    fAngles[1] = fBaseAngle + PENDULUM_MAX
                    iDirections |= (1<<SENTRY_DIR_CANNON)
                    SetSentryPenddir ( sentry, iDirections )
                }
            }

            entity_set_vector ( sentry, EV_VEC_angles, fAngles )
            return true
        }
        case SENTRY_FIREMODE_NUTS:
        {
            new Float:fAngles[3]
            entity_get_vector ( sentry, EV_VEC_angles, fAngles )

            new Float:fSpinSpeed = entity_get_float ( sentry, SENTRY_FL_SPINSPEED )
            if ( GetSentryPenddir ( sentry ) & (1<<SENTRY_DIR_CANNON) )
            {
                fAngles[1] -= ( fSpinSpeed * 0.01 )
                if ( fAngles[1] < 0.0 )
                    fAngles[1] = 360.0 + fAngles[1]
            }
            else
            {
                fAngles[1] += ( fSpinSpeed * 0.01 )
                if ( fAngles[1] > 360.0 )
                    fAngles[1] = fAngles[1] - 360.0
            }
            entity_set_float ( sentry, SENTRY_FL_SPINSPEED, ( fSpinSpeed += random_float ( 1.0, 2.0 ) ) )

            new Float:fMaxSpin = entity_get_float ( sentry, SENTRY_FL_MAXSPIN )
            if ( fMaxSpin == 0.0 )
            {
                entity_set_float ( sentry, SENTRY_FL_LASTTHINK, 0.5 )
                entity_set_float ( sentry, SENTRY_FL_MAXSPIN, fMaxSpin = random_float ( 500.0, 750.0 ) )
            }
            else if ( fSpinSpeed >= fMaxSpin )
            {
                sentry_detonate ( sentry, false, false )
                return false
            }
            entity_set_vector ( sentry, EV_VEC_angles, fAngles )
            return true
        }
    }
    return true
}

//#define    TE_TRACER            6        // tracer effect from point to point
tracer(Float:start[3], Float:end[3]) {
    new start_[3], end_[3]
    FVecIVec(start, start_)
    FVecIVec(end, end_)
    message_begin(MSG_BROADCAST, SVC_TEMPENTITY) //  MSG_PAS MSG_BROADCAST
    write_byte(TE_TRACER)
    write_coord(start_[0])
    write_coord(start_[1])
    write_coord(start_[2])
    write_coord(end_[0])
    write_coord(end_[1])
    write_coord(end_[2])
    message_end()
}
stock create_explosion(Float:origin_[3]) {
    new origin[3]
    FVecIVec(origin_, origin)

    message_begin(MSG_BROADCAST, SVC_TEMPENTITY, origin) // MSG_PAS not really good here
    write_byte(TE_EXPLOSION)
    write_coord(origin[0])
    write_coord(origin[1])
    write_coord(origin[2])
    write_short(g_sModelIndexFireball)
    write_byte(random_num(0, 20) + 50) // scale * 10 // random_num(0, 20) + 20
    write_byte(12) // framerate
    write_byte(TE_EXPLFLAG_NONE)
    message_end()

    KnockBack ( origin_ )

    new Float:playerOrigin[3], Float:distance, Float:flDmgToDo, Float:dmgbase = 90.0, newHealth
    for (new i = 1; i <= g_iMaxPlayers; i++) {
        if (!is_user_alive(i) || get_user_godmode(i))
            continue

        entity_get_vector(i, EV_VEC_origin, playerOrigin)
        distance = vector_distance(playerOrigin, origin_)
        if (distance <= SENTRYEXPLODERADIUS) {
            flDmgToDo = dmgbase - (dmgbase * (distance / SENTRYEXPLODERADIUS))
            newHealth = get_user_health(i) - floatround(flDmgToDo)
            if (newHealth <= 0) {
                set_task(0.0, "TicketToHell", i)
                continue
            }

            set_user_health(i, newHealth)

            message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, i)
            write_byte(floatround(flDmgToDo))
            write_byte(floatround(flDmgToDo))
            write_long(DMG_BLAST)
            write_coord(origin[0])
            write_coord(origin[1])
            write_coord(origin[2])
            message_end()
        }
    }
}

public TicketToHell(player) {
    if (!is_user_connected(player))
        return
    new frags = get_user_frags(player)
    user_kill(player, 1) // don't decrease frags
    new parms[4]
    parms[0] = player
    parms[1] = frags
    parms[2] = cs_get_user_deaths(player)
    parms[3] = int:cs_get_user_team(player)
    set_task(0.0, "DelayedScoreInfoUpdate", 0, parms, 4)
}

public DelayedScoreInfoUpdate(parms[4]) {
    scoreinfo_update(parms[0], parms[1], parms[2], parms[3])
}

KnockBack ( Float:origin[3] )
{
    new iEntList[32]
    new iEntsFound = find_sphere_class ( 0, "player", SENTRYEXPLODERADIUS, iEntList, g_iMaxPlayers, origin )

    if ( !iEntsFound )
        return

    new Float:fOriginEnt[3]
    new Float:fVelocity[3]
    new Float:fOriginEnd[3]
    new Float:fDistance
    new iPlayer

    for ( new i; i < iEntsFound; i++ )
    {
        iPlayer = iEntList[i]

        if ( !is_user_alive ( iPlayer ) )
            continue

        entity_get_vector ( iPlayer, EV_VEC_origin, fOriginEnt )

        fDistance = vector_distance ( fOriginEnt, origin )

        if ( is_entity_on_ground ( iPlayer ) && fOriginEnt[2] < origin[2] )
            fOriginEnt[2] = origin[2] + fDistance

        entity_get_vector ( iPlayer, EV_VEC_velocity, fVelocity )

        fOriginEnd[0] = ( fOriginEnt[0] - origin[0] ) * SENTRYEXPLODERADIUS / fDistance + origin[0]
        fOriginEnd[1] = ( fOriginEnt[1] - origin[1] ) * SENTRYEXPLODERADIUS / fDistance + origin[1]
        fOriginEnd[2] = ( fOriginEnt[2] - origin[2] ) * SENTRYEXPLODERADIUS / fDistance + origin[2]

        fVelocity[0] += ( fOriginEnd[0] - fOriginEnt[0] ) * SENTRYSHOCKPOWER
        fVelocity[1] += ( fOriginEnd[1] - fOriginEnt[1] ) * SENTRYSHOCKPOWER
        fVelocity[2] += ( fOriginEnd[2] - fOriginEnt[2] ) * SENTRYSHOCKPOWER

        entity_set_vector ( iPlayer, EV_VEC_velocity, fVelocity )
    }
}

public msg_TempEntity ()
{
    if ( get_msg_args () != 15 && get_msg_arg_int ( 1 ) != TE_BREAKMODEL )
        return PLUGIN_CONTINUE

    new ent = -1
    while ((ent = find_ent_by_class(ent, SentryClassName)))
    {
        if (entity_get_float ( ent, EV_FL_health ) <= 0.0) sentry_detonate ( ent, false, false )
    }
    return PLUGIN_CONTINUE
}

public fw_ThinkSentry ( ent )
{
    if ( !is_valid_ent ( ent ) )
        return

    static iOwner; iOwner = GetSentryPeople ( ent, OWNER )

    if ( !is_user_connected ( iOwner ) )
        return

    if ( cs_get_user_team ( iOwner ) == CS_TEAM_SPECTATOR )
    {
        sentry_detonate ( ent, true, false )
        return
    }
   
    if ( !sentry_pendulum ( ent ) )
        return

    static Float:fGameTime; fGameTime = get_gametime ()
    if ( entity_get_float ( ent, SENTRY_FL_LASTTHINK ) <= fGameTime )
    {
        new Float:fOriginSentry[3], Float:fOriginHit[3], iHitEnt
        entity_get_vector ( ent, EV_VEC_origin, fOriginSentry )
        fOriginSentry[2] += CANNONHEIGHTFROMFEET // Move up some, this should be the Y origin of the cannon
   
        new firemode = GetSentryFiremode ( ent )
        new target = GetSentryTarget ( ent, TARGET )
        if ( firemode == SENTRY_FIREMODE_YES && is_valid_ent ( target ) && is_user_alive ( target ) && cs_get_user_team ( target ) != GetSentryTeam ( ent ) && !IsInSphere ( target ))
        {
            new Float:fOriginTarget[3]
            entity_get_vector ( target, EV_VEC_origin, fOriginTarget )
   
            if ( entity_get_int ( target, EV_INT_flags ) & FL_DUCKING )
                fOriginTarget[2] += TARGETUPMODIFIER
   
            iHitEnt = trace_line ( ent, fOriginSentry, fOriginTarget, fOriginHit )
            if ( iHitEnt == entity_get_edict ( ent, SENTRY_ENT_BASE ) )
                iHitEnt = trace_line ( iHitEnt, fOriginHit, fOriginTarget, fOriginHit )
   
            if ( iHitEnt != target && is_user_alive ( iHitEnt ) && GetSentryTeam ( ent ) != cs_get_user_team ( iHitEnt ) && !IsInSphere ( iHitEnt ))
            {
                target = iHitEnt
                SetSentryTarget(ent, TARGET, iHitEnt)
            }
            if ( iHitEnt == target )
            {
                SentryTurnToTarget ( ent, fOriginSentry, fOriginTarget )
                new Float:fHitRatio = random_float ( 0.0, 1.0 ) - g_HITRATIOS // ie 0.5 - 0.7 = -0.2, a hit and 0.8 - 0.7 = a miss by 0.1
                new iLevel = GetSentryLevel(ent)
                if(iLevel == SENTRY_LEVEL_4){
                   -  if(sh_get_hero() == target)
                  +  if(target)
                    {
                        if(get_gametime() > entity_get_float(ent,SENTRY_ROCKET_TIME)){
                            if(entity_range(ent,target) >= RPG_DISTANCE){
                                new data[2]
                                data[0] = ent
                               
                                ShootRockets(data)
                            }
                           
                            entity_set_float(ent,SENTRY_ROCKET_TIME,get_gametime() +  2.0)
                        }
                    }
                } else if(iLevel == SENTRY_LEVEL_5) {
                    FireZevs(fOriginSentry,fOriginTarget);
                    emit_sound ( ent, CHAN_WEAPON, "weapons/tok.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
                }
                else emit_sound ( ent, CHAN_WEAPON, "error_csdm/sentry/fire.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
               
                if ( !get_user_godmode ( target ) && fHitRatio <= 0.0 )
                    sentry_damagetoplayer ( ent, fOriginSentry, target )
                else
                {
                    new Float:fSentryAngle[3] = {0.0, 0.0, 0.0}
   
                    new Float:x = fOriginHit[0] - fOriginSentry[0]
                    new Float:z = fOriginHit[1] - fOriginSentry[1]
                    new Float:radians = floatatan ( z/x, radian )
                    fSentryAngle[1] = radians * g_ONEEIGHTYTHROUGHPI
                    if ( fOriginHit[0] < fOriginSentry[0] )
                        fSentryAngle[1] -= 180.0
   
                    new Float:h = fOriginHit[2] - fOriginSentry[2]
                    new Float:b = vector_distance ( fOriginSentry, fOriginHit )
                    radians = floatatan ( h/b, radian )
                    fSentryAngle[0] = radians * g_ONEEIGHTYTHROUGHPI
   
                    fSentryAngle[0] += random_float ( -10.0 * fHitRatio, 10.0 * fHitRatio ) // aim is a little off here :-)
                    fSentryAngle[1] += random_float ( -10.0 * fHitRatio, 10.0 * fHitRatio ) // aim is a little off here :-)
                    engfunc ( EngFunc_MakeVectors, fSentryAngle )
                    new Float:vector[3]
                    get_global_vector ( GL_v_forward, vector )
                    for ( new i = 0; i < 3; i++ )
                        vector[i] *= 1000
   
                    new Float:traceEnd[3]
                    for ( new i = 0; i < 3; i++ )
                        traceEnd[i] = vector[i] + fOriginSentry[i]
   
                    new iHitEnt2 = ent
                    static lolcheck = 0
                    while ( ( iHitEnt2 = trace_line ( iHitEnt2, fOriginHit, traceEnd, fOriginHit ) ) )
                        if ( lolcheck++ > 700 ) break
   
                }
                tracer ( fOriginSentry, fOriginHit )

                entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + 0.1 )
                entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
                return
            }
            else
            {
                SetSentryFiremode ( ent, SENTRY_FIREMODE_NO )
                new Float:fAngle[3]
                entity_get_vector(ent, EV_VEC_angles, fAngle)
                fAngle[0] = 0.0
                entity_set_vector(ent, EV_VEC_angles, fAngle)
            }
        }
        else if ( firemode == SENTRY_FIREMODE_NUTS )
        {
            new iHitEnt2 = EntViewHitPoint ( ent, fOriginSentry, fOriginHit )
            emit_sound(ent, CHAN_WEAPON, "error_csdm/sentry/fire.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)
            tracer(fOriginSentry, fOriginHit)
   
            if (is_user_connected(iHitEnt2) && is_user_alive(iHitEnt2) && !get_user_godmode(iHitEnt2))
            {
                sentry_damagetoplayer(ent, fOriginSentry, iHitEnt2)
            }
            entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + 0.1 )
            entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
            return
        }
   
        if ( random_num ( 0, 99 ) < 10 )
            emit_sound ( ent, CHAN_AUTO, "error_csdm/sentry/turridle.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )

        new closestTarget = 0, Float:closestDistance, Float:distance, Float:closestOrigin[3], Float:playerOrigin[3], CsTeams:sentryTeam = GetSentryTeam ( ent )
        for ( new i = 1; i <= g_iMaxPlayers; i++ )
        {
            if ( !is_user_connected ( i ) || !is_user_alive ( i ) || cs_get_user_team ( i ) == sentryTeam || IsInSphere ( i ) )
                continue
   
            entity_get_vector ( i, EV_VEC_origin, playerOrigin )
   
            if ( entity_get_int ( i, EV_INT_flags ) & FL_DUCKING )
                playerOrigin[2] += TARGETUPMODIFIER
   
            iHitEnt = trace_line ( ent, fOriginSentry, playerOrigin, fOriginHit )
            if ( iHitEnt == entity_get_edict ( ent, SENTRY_ENT_BASE ) )
                iHitEnt = trace_line(iHitEnt, fOriginHit, playerOrigin, fOriginHit)
   
            if ( iHitEnt == i )
            {
                distance = vector_distance ( fOriginSentry, playerOrigin )
                closestOrigin = playerOrigin
   
                if ( distance < closestDistance || closestTarget == 0 )
                {
                    closestTarget = i
                    closestDistance = distance
                }
            }
        }
   
        if ( closestTarget )
        {
            emit_sound ( ent, CHAN_AUTO, "error_csdm/sentry/turrspot.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
            SentryTurnToTarget ( ent, fOriginSentry, closestOrigin )
   
            SetSentryFiremode ( ent, SENTRY_FIREMODE_YES )
            SetSentryTarget ( ent, TARGET, closestTarget )
        }
        else
        {
            SetSentryFiremode ( ent, SENTRY_FIREMODE_NO )
            new Float:fAngle[3]
            entity_get_vector(ent, EV_VEC_angles, fAngle)
            fAngle[0] = 0.0
            entity_set_vector(ent, EV_VEC_angles, fAngle)
        }

        entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + g_THINKFREQUENCIES )
    }
    entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
}

public think_sentrybase(sentrybase) {
    sentrybase_broke(sentrybase)
    return PLUGIN_CONTINUE
}

sentrybase_broke(sentrybase) {
    new sentry = entity_get_edict(sentrybase, BASE_ENT_SENTRY)
    if (is_valid_ent(sentrybase))
        remove_entity(sentrybase)

    if (sentry == 0)
        return

    SetSentryFiremode ( sentry, SENTRY_FIREMODE_NUTS )
}

sentry_detonate(sentry, bool:quiet, bool:isIndex) {
    new i
    if (isIndex)
    {
        i = sentry
        sentry = g_sentries[sentry]
        if (!is_valid_ent(sentry))
            return
    }
    else
    {
        if (!is_valid_ent(sentry))
            return

        for (new j = 0; j < g_sentriesNum; j++) {
            if (g_sentries[j] == sentry) {
                i = j
                break
            }
        }
    }
    //entity_set_float ( sentry, EV_FL_nextthink, 0.0 )

    new owner = GetSentryPeople(sentry, OWNER)

    if (!quiet) {
        new Float:origin[3]
        entity_get_vector(sentry, EV_VEC_origin, origin)
        create_explosion(origin)
        ChatColor ( owner, "^3[^4Действие^3]^1 Твоя пушка взорвана!")
        ammo_hud(owner, 0)
        sentries_num[owner] -= 1
        ammo_hud(owner, 1)      
    }
    DecreaseSentryCount(owner, sentry)

    // Remove base first
    if (GetSentryFiremode ( sentry ) != SENTRY_FIREMODE_NUTS)
        set_task ( 0.1, "DelayRemoveEntity", entity_get_edict ( sentry, SENTRY_ENT_BASE ) )
        //remove_entity(entity_get_edict(sentry, SENTRY_ENT_BASE))

    new CsTeams:iSentryTeam = GetSentryTeam ( sentry )

    set_task ( 0.1, "DelayRemoveEntity", sentry )
    //remove_entity(sentry)
    // Put the last sentry in the deleted entity's place
    if(0 > (g_sentriesNum - 1) > MAXSENTRIES) return
    g_sentries[i] = g_sentries[g_sentriesNum - 1]
   
    g_teamsentriesNum[_:iSentryTeam-1]--
}

public DelayRemoveEntity ( ent )
{
    if ( is_valid_ent ( ent ) )
        remove_entity ( ent )
}

sentry_detonate_by_owner(owner, bool:quiet = false) {
    if(sentries_num[owner]>0)
    {
        static ent = -1
        while ((ent = find_ent_by_class(ent, SentryClassName)))
        {      
            if( GetSentryPeople(ent,OWNER) != owner) continue;
            sentry_detonate ( ent, quiet, false )
        }
    }
}

public client_disconnect(id) {
    sentry_detonate_by_owner ( id )
}

// урон игроку
stock sentry_damagetoplayer(sentry, Float:sentryOrigin[3], target) {
    new newHealth = get_user_health(target) - g_DMG

    if (newHealth <= 0) {
        new targetFrags = get_user_frags(target) + 1
        new owner = GetSentryPeople(sentry, OWNER)
       
        if(!is_user_connected(owner))
            return
       
        new ownerFrags = get_user_frags(owner) + 1
        set_user_frags(target, targetFrags) // otherwise frags are subtracted from victim for dying (!!)
        set_user_frags(owner, ownerFrags)
       
        new contributors[5]
        contributors[0] = owner
        contributors[1] = GetSentryPeople(sentry, UPGRADER_1)
        contributors[2] = GetSentryPeople(sentry, UPGRADER_2)
        contributors[3] = GetSentryPeople(sentry, UPGRADER_3)
        contributors[4] = GetSentryPeople(sentry, UPGRADER_4)
       
        for(new i ; i < sizeof contributors ; i++){
            if(!contributors[i])
                continue
               
            if(!is_user_connected(contributors[i]) || get_user_team(contributors[i]) != get_user_team(contributors[0])){
                switch(i){ // yao face
                    case 1: SetSentryPeople(sentry,UPGRADER_1,0)
                    case 2: SetSentryPeople(sentry,UPGRADER_2,0)
                    case 3: SetSentryPeople(sentry,UPGRADER_3,0)
                    case 4: SetSentryPeople(sentry,UPGRADER_4,0)
                }
               
                continue
            }
           
            // izvini 4yvak, no menya nakrilo
            cs_set_user_money(contributors[i], cs_get_user_money(contributors[i]) + (i == 0 ? SENTRYOWNERAWARD : SENTRYASSISTAWARD))
        }

        // ny ebatb kakoy frag
        message_begin(MSG_ALL, g_msgDeathMsg, {0, 0, 0} ,0)
        write_byte(owner)
        write_byte(target)
        write_byte(0)
        write_string("sentry gun")
        message_end()
       
        //add_user_exp ( owner )
        scoreinfo_update(owner, ownerFrags, cs_get_user_deaths(owner), int:cs_get_user_team(owner))
        set_msg_block(g_msgDeathMsg, BLOCK_ONCE)
    }

    set_user_health(target, newHealth)

    message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, target)
    write_byte(g_DMG)
    write_byte(g_DMG)
    write_long(DMG_BULLET)
    write_coord(floatround(sentryOrigin[0]))
    write_coord(floatround(sentryOrigin[1]))
    write_coord(floatround(sentryOrigin[2]))
    message_end()
}

scoreinfo_update(id, frags, deaths, team) {
    message_begin(MSG_ALL, g_msgScoreInfo)
    write_byte(id)
    write_short(frags)
    write_short(deaths)
    write_short(0)
    write_short(team)
    message_end()
}

/* SentryTurnToTarget ( ent, Float:sentry_origin[3], Float:closest_origin[3] )
{
    new Float:fAngle[3]
    entity_get_vector ( ent, EV_VEC_angles, fAngle )
    new Float:x = closest_origin[0] - sentry_origin[0]
    new Float:z = closest_origin[1] - sentry_origin[1]

    new Float:fRadians = floatatan ( z/x, radian )
    fAngle[1] = fRadians * g_ONEEIGHTYTHROUGHPI
    if ( closest_origin[0] < sentry_origin[0] )
        fAngle[1] -= 180.0

    entity_set_float ( ent, SENTRY_FL_ANGLE, fAngle[1] )
    entity_set_vector ( ent, EV_VEC_angles, fAngle )
} */

SentryTurnToTarget (ent, Float:sentryOrigin[3], Float:closestOrigin[3]) {
    new Float:newAngle[3]
    entity_get_vector(ent, EV_VEC_angles, newAngle)
    new Float:x = closestOrigin[0] - sentryOrigin[0]
    new Float:z = closestOrigin[1] - sentryOrigin[1]
    new Float:y = closestOrigin[2] - sentryOrigin[2]

    newAngle[1] = floatasin(z/floatsqroot(x*x+z*z), degrees)
   
    new Float:radians = floatatan(z/x, radian)
    newAngle[1] = radians * g_ONEEIGHTYTHROUGHPI
    if (closestOrigin[0] < sentryOrigin[0])
        newAngle[1] -= 180.0

    entity_set_float(ent, SENTRY_FL_ANGLE, newAngle[1])

    new Float:h = closestOrigin[2] - sentryOrigin[2]
    new Float:b = vector_distance(sentryOrigin, closestOrigin)
    radians = floatatan(h/b, radian)
    newAngle[0] = floatasin(x/floatsqroot(x*x+y*y), degrees)
    newAngle[0] = radians * g_ONEEIGHTYTHROUGHPI
   
    entity_set_vector(ent, EV_VEC_angles, newAngle)
}

AimingAtSentry ( id )
{
    if ( !is_user_alive ( id ) )
        return 0

    new hitEnt, bodyPart
    if (get_user_aiming(id, hitEnt, bodyPart) == 0.0)
        return 0

    if ( is_valid_ent ( hitEnt ) )
    {
        new classname[32], l_sentry
        entity_get_string(hitEnt, EV_SZ_classname, classname, 31)
        if (equal(classname, "sentry_base"))
            l_sentry = entity_get_edict(hitEnt, BASE_ENT_SENTRY)
        else if (equal(classname, SentryClassName))
            l_sentry = hitEnt
        else
            l_sentry = 0

        return l_sentry
    }
    return 0
}

// улучшение уровня пушки
bool:SentryUpgrade ( id, sentry )
{
    if(!is_user_alive(id)||!is_user_connected(id))
        return false
   
    if ( GetSentryFiremode ( sentry ) == SENTRY_FIREMODE_NUTS )
        return false
   
    new iLevel = GetSentryLevel ( sentry )
    if ( iLevel == SENTRY_LEVEL_5 )
        return false
   
    if ( cs_get_user_team ( id ) != GetSentryTeam ( sentry ))
    {
        return false
    }

    if ( cs_get_user_team ( GetSentryPeople ( sentry, OWNER ) ) == CS_TEAM_SPECTATOR )
        return false

    // e ron don don
    if (get_user_flags(id) & get_pcvar_flags(g_FlagVip))
    {
    }
    else
    {
        // e ron don don
        if(GetSentryPeople(sentry,UPGRADER_1) == id ||
            GetSentryPeople(sentry,UPGRADER_2) == id ||
            GetSentryPeople(sentry,UPGRADER_3) == id ||
            GetSentryPeople(sentry,UPGRADER_4) == id
        )
            return false
    }
   
    iLevel++

    if (get_user_flags(id) & get_pcvar_flags(g_FlagVip))
    {
        if ( cs_get_user_money ( id ) - g_COST_VIP[iLevel] < 0 )
        {
            client_print(id, print_center, "У тебя не хватает денег (нужно %d$)", g_COST_VIP[iLevel])
            return false
        }
    }
    else
    {
        if ( cs_get_user_money ( id ) - g_COST[iLevel] < 0 )
        {
            client_print(id, print_center, "У тебя не хватает денег (нужно %d$)", g_COST[iLevel])
            return false
        }
    }

    if (get_user_flags(id) &get_pcvar_flags(g_FlagVip))
    {
        cs_set_user_money ( id, cs_get_user_money ( id ) - g_COST_VIP[iLevel] )
    }
    else
    {
        cs_set_user_money ( id, cs_get_user_money ( id ) - g_COST[iLevel] )
    }
   
    new iTeam = _:cs_get_user_team ( id ), iUpgraderField
    switch ( iLevel )
    {
        // this kod is very zaebisb
        case SENTRY_LEVEL_2:
        {
            switch ( iTeam )
            {
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_2.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_2.mdl" )
            }
            iUpgraderField = UPGRADER_1
        }
        case SENTRY_LEVEL_3:
        {
            switch ( iTeam )
            {
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            iUpgraderField = UPGRADER_2
        }
        case SENTRY_LEVEL_4:{
            switch(iTeam){
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            iUpgraderField = UPGRADER_3
        }
        case SENTRY_LEVEL_5:{
            switch(iTeam){
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            // entity_set_byte(sentry,EV_BYTE_controller2,120)
            // entity_set_byte(sentry,EV_BYTE_controller3,120)
            iUpgraderField = UPGRADER_4
        }
    }

    new Float:fMins[3], Float:fMaxs[3]
    fMins[0] = -10.0
    fMins[1] = -10.0
    fMins[2] = 0.0
    fMaxs[0] = 10.0
    fMaxs[1] = 10.0
    fMaxs[2] = 40.0 // 4.0
   
    entity_set_size ( sentry, fMins, fMaxs )
    emit_sound ( sentry, CHAN_AUTO, "error_csdm/sentry/turrset.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
    SetSentryLevel ( sentry, iLevel )
    entity_set_float ( sentry, EV_FL_health, g_HEALTHS[iLevel] )
    entity_set_float ( entity_get_edict ( sentry, SENTRY_ENT_BASE ), EV_FL_health, g_HEALTHS[0] )
    SetSentryPeople ( sentry, iUpgraderField, id )
   
    new sName[32]
    get_user_name ( id, sName, charsmax ( sName ) )
    client_print ( GetSentryPeople ( sentry, OWNER ), print_center, "%s прокачал твою пушку до уровня %d", sName, iLevel + 1 )
    return true
}

stock EntViewHitPoint ( index, Float:origin[3], Float:hitorigin[3] )
{
    if ( !is_valid_ent ( index ) )
        return 0

    new Float:angle[3], Float:vec[3], Float:f_dest[3]

    entity_get_vector(index, EV_VEC_angles, angle)

    engfunc(EngFunc_AngleVectors, angle, vec, 0, 0)

    f_dest[0] = origin[0] + vec[0] * 9999
    f_dest[1] = origin[1] + vec[1] * 9999
    f_dest[2] = origin[2] + vec[2] * 9999

    return trace_line(index, origin, f_dest, hitorigin)
}

public fw_PlayerSpawn_Post ( id )
{
    g_inBuilding[id] = false
    sentry_detonate_by_owner ( id )
    ammo_hud ( id, 0 )
    sentries_num[id] = 0
}

public fw_TraceLine_Post ( Float:start[3], Float:end[3], noMonsters, id )
{
    if ( !is_valid_player ( id ) || !is_user_alive ( id ) )
        return FMRES_IGNORED

    new iHitEnt = get_tr ( TR_pHit )

    if ( iHitEnt <= g_iMaxPlayers )
        return FMRES_IGNORED

    new sClassName[11], sentry, base

    pev ( iHitEnt, pev_classname, sClassName, charsmax ( sClassName ) )

    if ( equal ( sClassName, "sentrybase" ) )
    {
        base = iHitEnt
        sentry = entity_get_edict ( iHitEnt, BASE_ENT_SENTRY )
    }
    else if ( equal ( sClassName, SentryClassName ) )
    {
        sentry = iHitEnt
        base = entity_get_edict ( sentry, SENTRY_ENT_BASE )
    }

    if ( !is_valid_ent ( sentry ) || !base )
        return FMRES_IGNORED
       
    if ( GetSentryFiremode ( sentry ) == SENTRY_FIREMODE_NUTS )
             return FMRES_IGNORED
       
    new Float:health = entity_get_float ( sentry, EV_FL_health )

    if ( health <= 0 )
        return FMRES_IGNORED

    new Float:basehealth = entity_get_float ( base, EV_FL_health )

    if ( basehealth <= 0 )
        return FMRES_IGNORED

    new CsTeams:team = GetSentryTeam ( sentry )

    if ( team != cs_get_user_team ( id ) )
        return FMRES_IGNORED

    new level = GetSentryLevel ( sentry )

    static tempStatusBuffer[192], tempStatusBuffer2[192]

    new OwnName[33]
    get_user_name ( GetSentryPeople ( sentry, OWNER ), OwnName, 32 )
   
    formatex ( tempStatusBuffer, charsmax ( tempStatusBuffer ), "Установил: %s^nЗдоровье: %d/%d",OwnName, floatround(health), floatround(g_HEALTHS[level]) )
    formatex ( tempStatusBuffer2, charsmax ( tempStatusBuffer2 ), "^n^nЗдоровье основания: %d/%d^nУровень: %d", floatround(basehealth), floatround(g_HEALTHS[0]), level + 1 )

    set_dhudmessage ( 255, 255, 255, -1.0, 0.75, 0, 0.0, 0.6, 0.0, 0.0 )
    show_dhudmessage(id, tempStatusBuffer)
    show_dhudmessage(id, tempStatusBuffer2)

    return FMRES_IGNORED
}

// прикосновение к пушке игрока
new Float:flood_touch[33]
public fw_TouchSentry ( sentry, player )
{
    static Float:time;time=get_gametime()
    if(flood_touch[player] < time)
    SentryUpgrade ( player, sentry );
    flood_touch[player] = time+1.0
    return 1
}

ammo_hud(id, sw)
{
    if(is_user_bot(id)||!is_user_alive(id)||!is_user_connected(id))
            return

    new s_sprite[33]
    format(s_sprite, 32, "number_%d", sentries_num[id])
    if(sw)
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 1 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }
    else
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 0 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }
    if(sentries_num[id] <= 0)
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 0 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }  
}

stock ChatColor(const id, const input[], any:...)
{
        new count = 1, players[32]
        static msg[191]
        vformat(msg, 190, input, 3)
     
        replace_all(msg, 190, "!g", "^4") // Green Color
        replace_all(msg, 190, "!y", "^1") // Default Color
        replace_all(msg, 190, "!team", "^3") // Team Color
        replace_all(msg, 190, "!team2", "^0") // Team2 Color
     
        if (id) players[0] = id; else get_players(players, count, "ch")
        {
                for (new i = 0; i < count; i++)
                {
                        if (is_user_connected(players[i]))
                        {
                                message_begin(MSG_ONE_UNRELIABLE, get_user_msgid("SayText"), _, players[i])
                                write_byte(players[i]);
                                write_string(msg);
                                message_end();
                        }
                }
        }
}

bool:IsInSphere ( id )
{
    if ( !is_user_alive ( id ) )
        return false

    new ent = -1
    while ( ( ent = engfunc ( EngFunc_FindEntityByString, ent, "classname", "campo_grenade_forze" ) ) > 0 )
    {
        new iOwner = pev ( ent, pev_owner )

        if ( cs_get_user_team ( id ) != cs_get_user_team ( iOwner ) )
            continue

        new Float:fOrigin[3]
        pev ( ent, pev_origin, fOrigin )
        new iPlayer = -1
        while ( ( iPlayer = engfunc ( EngFunc_FindEntityInSphere, iPlayer, fOrigin, 68.0 ) ) != 0 )
        {
            if ( iPlayer == id )
                return true
        }
    }
    return false
}

//
// Launch rocket from 4lvl sentry
//    data[2]
//        0 - sentry id
//        1 - side    // 0 - right, 1 - left
//
public ShootRockets(data[2]){
    new sentry = data[0]
    new side = data[1]
   
    new Float:rocketOrigin[3],Float:rocketAngles[3]
   
    entity_get_vector(sentry,EV_VEC_angles,rocketAngles)
    engfunc(EngFunc_MakeVectors,rocketAngles)
   
    new Float:vecForward[3],Float:vecRight[3],Float:vecUp[3]
   
    get_global_vector(GL_v_forward,vecForward)
    xs_vec_mul_scalar(vecForward,20.0,vecForward)
   
    get_global_vector(GL_v_right,vecRight)
    xs_vec_mul_scalar(vecRight,side ? 8.0 : -8.0,vecRight) // right or left rocket
   
    get_global_vector(GL_v_up,vecUp)
    xs_vec_mul_scalar(vecUp,30.0,vecUp)
   
    entity_get_vector(sentry,EV_VEC_origin,rocketOrigin)
    xs_vec_add(rocketOrigin,vecForward,rocketOrigin)
    xs_vec_add(rocketOrigin,vecRight,rocketOrigin)
    xs_vec_add(rocketOrigin,vecUp,rocketOrigin)
   
    // shot rocket
    CreateRocket(sentry,rocketOrigin,rocketAngles,GetSentryTarget(sentry,TARGET))
   
    data[1] = 1
   
    if(!side) // shot left rocket
        set_task(0.2,"ShootRockets",_,data,sizeof data)
}

//
// Launch RPG rocket
//    sentry - sentry id
//    origin - rocket origin
//    angles - sentry angles
//    traget - rocket target id
//
CreateRocket(sentry,Float:origin[3],Float:angles[3],target){
    new rocket = create_entity("info_target")
   
    entity_set_string(rocket,EV_SZ_classname,"rpg_rocket")
   
    entity_set_int(rocket,EV_INT_movetype,MOVETYPE_FLY)
    entity_set_int(rocket,EV_INT_solid,SOLID_BBOX)
   
    entity_set_edict(rocket,EV_ENT_owner,sentry)
    entity_set_edict(rocket,EV_ENT_euser4,GetSentryPeople(sentry,OWNER))
   
    entity_set_size(rocket,Float:{-2.0,-2.0,-2.0},Float:{2.0,2.0,2.0})
    entity_set_origin(rocket,origin)
   
    new Float:targetOrigin[3]
    entity_get_vector(target,EV_VEC_origin,targetOrigin)
    angles[0] = -GetAngleOrigins(origin,targetOrigin)
    entity_set_model(rocket,"models/error_csdm/sentry/rpgrocket.mdl")
    entity_set_vector(rocket,EV_VEC_angles,angles)
   
    engfunc(EngFunc_MakeVectors,angles)
   
    new Float:vecVelocity[3]
    get_global_vector(GL_v_forward,vecVelocity)
    xs_vec_mul_scalar(vecVelocity,1000.0,vecVelocity)
    entity_set_vector(rocket,EV_VEC_velocity,vecVelocity)
   
    entity_set_int(rocket,EV_INT_effects,entity_get_int(rocket,EV_INT_effects) | EF_LIGHT)
   
    // rocket trail
    message_begin(MSG_BROADCAST,SVC_TEMPENTITY)
    write_byte(TE_BEAMFOLLOW)
    write_short(rocket)
    write_short(m_iTrail)
    write_byte(10)
    write_byte(5)
    write_byte(224)
    write_byte(224)
    write_byte(255)
    write_byte(255)
    message_end()
   
    emit_sound(rocket,CHAN_VOICE,"weapons/rocket1.wav",1.0,0.5,0,PITCH_NORM)
}

public fw_RpgTouch(rocket,ent){
    new Float:origin[3],Float:angles[3],Float:vecPlaneNormal[3]
    entity_get_vector(rocket,EV_VEC_origin,origin)
    entity_get_vector(rocket,EV_VEC_angles,angles)
   
    engfunc(EngFunc_MakeVectors,angles)
    get_global_vector(GL_v_forward,angles)
    xs_vec_mul_scalar(angles,9999.0,angles)
    xs_vec_add(origin,angles,angles)
    engfunc(EngFunc_TraceLine,origin,angles,0,rocket,0)
   
    get_tr2(0,TR_vecEndPos,origin)
   
    message_begin_f(MSG_BROADCAST,SVC_TEMPENTITY,origin,0)
    write_byte(TE_WORLDDECAL)
    write_coord_f(origin[0])
    write_coord_f(origin[1])
    write_coord_f(origin[2])
    write_byte(expDecal)
    message_end()
   
    get_tr2(0,TR_vecPlaneNormal,vecPlaneNormal)
   
    xs_vec_mul_scalar(vecPlaneNormal,8.0,vecPlaneNormal)
    xs_vec_add(origin,vecPlaneNormal,origin)
   
    message_begin_f(MSG_PVS,SVC_TEMPENTITY,origin,0)
    write_byte(TE_EXPLOSION)
    write_coord_f(origin[0])
    write_coord_f(origin[1])
    write_coord_f(origin[2])
    write_short(g_sModelIndexFireball)
    write_byte(20)
    write_byte(15)
    write_byte(0)
    message_end()
    shit_radiusdamage(rocket,origin)
    emit_sound(rocket,CHAN_VOICE,"weapons/rocket1.wav",0.0,ATTN_NORM,SND_STOP,0)
}

// this very bad method
stock shit_radiusdamage(rocket,Float:origin_[3]) {
    new origin[3]
    FVecIVec(origin_, origin)
   
    new attacker = entity_get_edict(rocket,EV_ENT_euser4)
   
    if(!is_user_connected(attacker))
        return


    new Float:playerOrigin[3], Float:distance, Float:flDmgToDo, Float:dmgbase = RPG_DAMAGE
    new sentry = entity_get_edict(rocket,EV_ENT_owner)
    for (new i = 1; i <= g_iMaxPlayers; i++) {
        if (!is_user_alive(i) || get_user_godmode(i) || get_user_team(i) == get_user_team(attacker))
            continue

        entity_get_vector(i, EV_VEC_origin, playerOrigin)
        distance = vector_distance(playerOrigin, origin_)
        if (distance <= RPG_RADIUS) {
            flDmgToDo = dmgbase - (dmgbase * (distance / RPG_RADIUS))
           
            // zemletryasenie!!111
            Util_ScreenShake(i,0.5,16.0,16.0)
            rocket_damagetoplayer(sentry,origin_,i,flDmgToDo)
        }
    }
    remove_entity(rocket)
}
// ScreenShake
stock Util_ScreenShake(id, Float:duration, Float:frequency, Float:amplitude)
{
    static ScreenShake = 0;
    if( !ScreenShake )
    {
        ScreenShake = get_user_msgid("ScreenShake");
    }
    message_begin( id ? MSG_ONE_UNRELIABLE : MSG_BROADCAST, ScreenShake, _, id);
    write_short( FixedUnsigned16( amplitude, 1<<12 ) ); // shake amount
    write_short( FixedUnsigned16( duration, 1<<12 ) ); // shake lasts this long
    write_short( FixedUnsigned16( frequency, 1<<8 ) ); // shake noise frequency
    message_end();
}

// урон игроку
stock rocket_damagetoplayer(sentry, Float:sentryOrigin[3], target, Float:dmg) {
    new newHealth = get_user_health(target) - floatround(dmg)

    if (newHealth <= 0) {
        new targetFrags = get_user_frags(target) + 1
        new owner = GetSentryPeople(sentry, OWNER)
       
        if(!is_user_connected(owner))
            return
       
        new ownerFrags = get_user_frags(owner) + 1
        set_user_frags(target, targetFrags) // otherwise frags are subtracted from victim for dying (!!)
        set_user_frags(owner, ownerFrags)
       
        new contributors[5]
        contributors[0] = owner
        contributors[1] = GetSentryPeople(sentry, UPGRADER_1)
        contributors[2] = GetSentryPeople(sentry, UPGRADER_2)
        contributors[3] = GetSentryPeople(sentry, UPGRADER_3)
        contributors[4] = GetSentryPeople(sentry, UPGRADER_4)
       
        for(new i ; i < sizeof contributors ; i++){
            if(!contributors[i])
                continue
               
            if(!is_user_connected(contributors[i]) || get_user_team(contributors[i]) != get_user_team(contributors[0])){
                switch(i){ // yao face
                    case 1: SetSentryPeople(sentry,UPGRADER_1,0)
                    case 2: SetSentryPeople(sentry,UPGRADER_2,0)
                    case 3: SetSentryPeople(sentry,UPGRADER_3,0)
                    case 4: SetSentryPeople(sentry,UPGRADER_4,0)
                }
               
                continue
            }
           
            // izvini 4yvak, no menya nakrilo
            cs_set_user_money(contributors[i],
                clamp(
                    cs_get_user_money(contributors[i]) + (i == 0 ? SENTRYOWNERAWARD : SENTRYASSISTAWARD),
                    0,
                    16000
                )
            )
        }

        // ny ebatb kakoy frag
        message_begin(MSG_ALL, g_msgDeathMsg, {0, 0, 0} ,0)
        write_byte(owner)
        write_byte(target)
        write_byte(0)
        write_string("sentry gun")
        message_end()

        scoreinfo_update(owner, ownerFrags, cs_get_user_deaths(owner), int:cs_get_user_team(owner))
        set_msg_block(g_msgDeathMsg, BLOCK_ONCE)
    }

    set_user_health(target, newHealth)

    message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, target)
    write_byte(g_DMG)
    write_byte(g_DMG)
    write_long(DMG_BLAST)
    write_coord(floatround(sentryOrigin[0]))
    write_coord(floatround(sentryOrigin[1]))
    write_coord(floatround(sentryOrigin[2]))
    message_end()
}

FireZevs(Float:starting[3], Float:ending[3])
{
    new OriginStr[3],OriginEnd[3];
    //Еффект луча от пушки до цели.
    FVecIVec(starting,OriginStr)
    FVecIVec(ending,OriginEnd)
    message_begin(MSG_BROADCAST, SVC_TEMPENTITY);
    write_byte(TE_BEAMPOINTS);
    write_coord(OriginStr[0]);
    write_coord(OriginStr[1]);
    write_coord(OriginStr[2]);
    write_coord(OriginEnd[0]);
    write_coord(OriginEnd[1]);
    write_coord(OriginEnd[2]);
    write_short(g_Sprt_Tok);
    write_byte(0);
    write_byte(1);
    write_byte(1);
    write_byte(random_num(10, 50));
    write_byte(random_num(10, 25));
    write_byte(random_num(0, 255));
    write_byte(random_num(0, 255));
    write_byte(random_num(0, 255));
    write_byte(1000);
    write_byte(0);
    message_end();
}

stock FixedUnsigned16( Float:value, scale )
{
    new output;

    output = floatround(value * scale);
    if ( output < 0 )
        output = 0;
    if ( output > 0xFFFF )
        output = 0xFFFF;

    return output;
}

Float: GetAngleOrigins(Float:fOrigin1[3], Float:fOrigin2[3] )
{
    new Float:fVector[3];
    new Float:fAngle[3];
    new Float:fLineAngle;
   
    xs_vec_sub(fOrigin2, fOrigin1, fVector);
    vector_to_angle(fVector, fAngle);
   
    if( fAngle[0] > 90.0 )
        fLineAngle = -(360.0 - fAngle[0]);
    else
        fLineAngle = fAngle[0];
   
    return fLineAngle;
}

Код:
#include <amxmodx>
#include <engine>
#include <fun>
#include <cstrike>
#include <fakemeta>
#include <hamsandwich>
#include <xs>

#define is_valid_player(%1) ( 1 <= %1 <= g_iMaxPlayers )
#define is_valid_team(%1) ( 0 < %1 < 3 )

#define is_entity_on_ground(%1) ( entity_get_int ( %1, EV_INT_flags ) & FL_ONGROUND )

#define MAXUPGRADERANGE                75.0
#define SENTRYEXPLODERADIUS            250.0            // радиус отброса при взрыве
#define SENTRYTILTRADIUS            830.0            // Насколько точно пушка будет нацелена на цель вертикально (вверх/вниз, просто для взглядов, цель рассчитывается по-разному)

#define SENTRYMINDISTANCE    256.0
#define MAXSENTRIES         1000
#define TASK_GODMODE         114455

#define SENTRY_INT_TARGET        EV_INT_iuser3
#define SENTRY_TARGET_BITS        6
#define TARGET                    0
#define MASK_TARGET                0xFFFFFFC0 // 11111111111111111111111111000000
new const MASKS_TARGET[1] = {MASK_TARGET}

GetSentryTarget(const SENTRY, const WHO) {
    new data = entity_get_int(SENTRY, SENTRY_INT_TARGET)
    data |= MASKS_TARGET[WHO]
    data ^= MASKS_TARGET[WHO]
    data = (data>>(WHO*SENTRY_TARGET_BITS))
    return data
}

SetSentryTarget(const SENTRY, const WHO, const IS) {
    new data = entity_get_int(SENTRY, SENTRY_INT_TARGET)
    data &= MASKS_TARGET[WHO] // nullify the setting
    data |= (IS<<(WHO*SENTRY_TARGET_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_TARGET, data) // store
}

#define SENTRY_INT_PEOPLE        EV_INT_iuser2 // max 5 users using 6 bits!
#define SENTRY_PEOPLE_BITS        6
#define OWNER                    0
#define UPGRADER_1                1
#define UPGRADER_2                2
#define UPGRADER_3                3
#define UPGRADER_4                4
#define MASK_OWNER                0xFFFFFFC0 // 11111111111111111111111111000000
#define MASK_UPGRADER_1            0xFFFFF03F // 11111111111111111111000000111111
#define MASK_UPGRADER_2            0xFFFC0FFF // 11111111111111000000111111111111
#define MASK_UPGRADER_3            0xFF03FFFF // 11111111000000111111111111111111
#define MASK_UPGRADER_4            0xC0FFFFFF // 11000000111111111111111111111111

new const MASKS_PEOPLE[5] = {MASK_OWNER, MASK_UPGRADER_1, MASK_UPGRADER_2, MASK_UPGRADER_3, MASK_UPGRADER_4}

GetSentryPeople(const SENTRY, const WHO) {
    new data = entity_get_int(SENTRY, SENTRY_INT_PEOPLE)
    data |= MASKS_PEOPLE[WHO]
    data ^= MASKS_PEOPLE[WHO]
    data = (data>>(WHO*SENTRY_PEOPLE_BITS))
    return data
}
SetSentryPeople(const SENTRY, const WHO, const IS) {
    new data = entity_get_int(SENTRY, SENTRY_INT_PEOPLE)
    data &= MASKS_PEOPLE[WHO] // nullify the setting
    data |= (IS<<(WHO*SENTRY_PEOPLE_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_PEOPLE, data) // store
}

#define SENTRY_INT_SETTINGS        EV_INT_iuser1
#define SENTRY_ROCKET_TIME        EV_FL_teleport_time
#define SENTRY_SETTINGS_BITS    3
#define SENTRY_SETTING_FIREMODE    0
#define SENTRY_SETTING_TEAM        1
#define SENTRY_SETTING_LEVEL    2
#define SENTRY_SETTING_PENDDIR    3
#define MASK_FIREMODE            0xFFFFFFF8 // 11111111111111111111111111111000 = FFFFFFFC
#define MASK_TEAM                0xFFFFFFC7 // 11111111111111111111111111000111 = FFFFFFF3
#define MASK_LEVEL                0xFFFFFE3F // 11111111111111111111111000111111 = FFFFFFCF
#define MASK_PENDDIR            0xFFFFF1FF // 11111111111111111111000111111111 = FFFFFF3F
new const MASKS_SETTINGS[4] = {MASK_FIREMODE, MASK_TEAM, MASK_LEVEL, MASK_PENDDIR}

GetSentrySettings(const SENTRY, const SETTING) {
    new data = entity_get_int(SENTRY, SENTRY_INT_SETTINGS)
    data |= MASKS_SETTINGS[SETTING]
    data ^= MASKS_SETTINGS[SETTING]
    //data = (data>>(SETTING*SENTRY_SETTINGS_BITS))
    return (data>>(SETTING*SENTRY_SETTINGS_BITS))
}
SetSentrySettings(const SENTRY, const SETTING, const VALUE) {
    new data = entity_get_int(SENTRY, SENTRY_INT_SETTINGS)
    data &= MASKS_SETTINGS[SETTING] // nullify the setting
    //data |= (VALUE<<(SETTING*SENTRY_SETTINGS_BITS)) // set the setting
    entity_set_int(SENTRY, SENTRY_INT_SETTINGS, data | (VALUE<<(SETTING*SENTRY_SETTINGS_BITS))) // store
}

GetSentryFiremode(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_FIREMODE)
}
SetSentryFiremode(const SENTRY, const MODE) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_FIREMODE, MODE)
}
CsTeams:GetSentryTeam(const SENTRY) {
    return CsTeams:GetSentrySettings(SENTRY, SENTRY_SETTING_TEAM)
}
SetSentryTeam(const SENTRY, const CsTeams:TEAM) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_TEAM, int:TEAM)
}
GetSentryLevel(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_LEVEL)
}
SetSentryLevel(const SENTRY, const LEVEL) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_LEVEL, LEVEL)
}
GetSentryPenddir(const SENTRY) {
    return GetSentrySettings(SENTRY, SENTRY_SETTING_PENDDIR)
}
SetSentryPenddir(const SENTRY, const PENDDIR) {
    SetSentrySettings(SENTRY, SENTRY_SETTING_PENDDIR, PENDDIR)
}

#define SENTRY_ENT_BASE            EV_ENT_euser1

#define SENTRY_FL_ANGLE            EV_FL_fuser1
#define SENTRY_FL_SPINSPEED        EV_FL_fuser2
#define SENTRY_FL_MAXSPIN        EV_FL_fuser3
#define SENTRY_FL_LASTTHINK        EV_FL_fuser4

#define SENTRY_DIR_CANNON        0

#define BASE_ENT_SENTRY            EV_ENT_euser1
#define BASE_INT_TEAM            EV_INT_iuser1

#define SENTRY_LEVEL_1            0
#define SENTRY_LEVEL_2            1
#define SENTRY_LEVEL_3            2
#define SENTRY_LEVEL_4            3
#define SENTRY_LEVEL_5            4
#define SENTRY_FIREMODE_NO        0
#define SENTRY_FIREMODE_YES        1
#define SENTRY_FIREMODE_NUTS    2
#define TARGETUPMODIFIER        18.0         //     if player ducks on ground, traces don't hit...
#define DMG_BULLET                (1<<1)        //    Выстрел
#define DMG_BLAST                (1<<6)        //     Урон от взрыва
#define TE_EXPLFLAG_NONE        0
#define TE_EXPLOSION            3
#define TE_TRACER                6
#define TE_BREAKMODEL            108
#define PENDULUM_MAX            45.0         //     Как далеко сторожевая башня поворачивается в каждом направлении на холостом ходу, прежде чем повернуть назад
#define PENDULUM_INCREMENT        10.0         //    Скорость поворота башни
#define SENTRYSHOCKPOWER        3.0         // multiplier, increase to make exploding sentries throw stuff further away
#define CANNONHEIGHTFROMFEET    20.0         // tweakable to make tracer originate from the same height as the sentry's cannon. Also traces rely on this Y-wise offset.
#define PLAYERORIGINHEIGHT        36.0         // this is the distance from a player's EV_VEC_origin to ground, if standing up
#define HEIGHTDIFFERENCEALLOWED    20.0         // increase value to allow building in slopes with higher angles. You can set to 0.0 and you will only be able to build on exact flat ground. note: mostly applies to downhill building, uphill is still likely to "collide" with ground...

#define PLACE_RANGE 45.0

#define SENTRY_RADAR            20             // use as high as possible but should still be working (ie be able to see sentries plotted on radar while in menu, too high values doesn't seem to work)
#define SENTRY_RADAR_TEAMBUILT    21             // same as above

#define RPG_RADIUS    250.0
#define RPG_DAMAGE    150.0
#define RPG_DISTANCE    400.0

#define FLAG_VIP     ADMIN_LEVEL_H            // Флаг VIP игрока
#define MAX_VIP     2                        // Максимальное кол-во пушек для VIP игрока [ FLAG_VIP ]
#define    MAX_PL        1                        // Максимальное кол-во пушек для игрока
#define NAGRADA        2000                    // Награда за разрушение пушки

new const szModels[][] =
{
    "models/error_csdm/sentry/base.mdl",
    "models/error_csdm/sentry/te_1.mdl",
    "models/error_csdm/sentry/te_2.mdl",
    "models/error_csdm/sentry/te_3.mdl",
   
    "models/error_csdm/sentry/ct_1.mdl",
    "models/error_csdm/sentry/ct_2.mdl",
    "models/error_csdm/sentry/ct_3.mdl",
    "models/error_csdm/sentry/rpgrocket.mdl",
    "models/computergibs.mdl"
}

new const szSounds[][] =
{
    "debris/bustmetal1.wav",
    "debris/bustmetal2.wav",
    "debris/metal1.wav",
    "debris/metal3.wav",
    "error_csdm/sentry/turridle.wav",
    "error_csdm/sentry/turrset.wav",
    "error_csdm/sentry/turrspot.wav",
    "error_csdm/sentry/building.wav",
    "error_csdm/sentry/fire.wav",
    "weapons/rocket1.wav",
    "weapons/tok.wav"
}

new expDecal

#define SENTRYOWNERAWARD    300
#define SENTRYASSISTAWARD    150

#define g_DMG 30                                                                 // количество урона от пушки в зависимости от ее уровня
#define g_THINKFREQUENCIES 0.7                                                     // через сколько захватывается цель
#define g_HITRATIOS 0.7                                                         // разброс

new const g_SENTRYCOST[5] = {3500, 4000, 7000, 8000, 9000}                           // стоимость установки
new const g_SENTRYCOST_VIP[5] = {200,300,400,700,800}                            // стоимость установки
new const g_COST_VIP[5] = {300, 100, 150, 850, 1000}                            // стоимость улучшения пушек VIP
new const Float:g_HEALTHS[5] = {3500.0, 4500.0, 5500.0, 6500.0, 7500.0}            // сколько хп у пушки в зависимости от ее уровня (верхняя часть)
new const g_COST[5] = {3500, 1500, 2500, 10000, 12000}                            // стоимость улучшения пушек

#define g_sentriesNum (g_teamsentriesNum[0]+g_teamsentriesNum[1])
new g_teamsentriesNum[2]  
new g_sentries[MAXSENTRIES]
new g_iPlayerSentries[33]
new g_iPlayerSentriesEdicts[33][5]
new g_sModelIndexFireball
new g_msgDamage
new g_msgDeathMsg
new g_msgScoreInfo
new g_msgHostagePos
new g_msgHostageK
new g_iMaxPlayers
new Float:g_ONEEIGHTYTHROUGHPI
new Float:g_sentryOrigins[32][3]
new bool:g_inBuilding[33]
new sentries_num[33]
new gMsgID
new m_iTrail
new g_Sprt_Tok
enum _:menuStatus {
    CURRENT_SENTRYID
}
#define SentryClassName "sentry"
#define SentryBaseClassName "sentrybase"
new attach_viewmod[33]

public plugin_init() {

    register_plugin("Sentry Guns", "1.0", "pro100web")

    register_event ( "Spectator", "ev_Spectation", "a" )
    register_clcmd("sentry_build", "cmd_CreateSentry", 0, "- build a sentry gun where you are")
   
    RegisterHam ( Ham_Spawn, "player", "fw_PlayerSpawn_Post", 1 )
    RegisterHam ( Ham_TakeDamage, "func_breakable", "fw_TakeDamage" )

    register_forward ( FM_TraceLine, "fw_TraceLine_Post", 1 )

    register_touch ( SentryClassName, "player", "fw_TouchSentry" )
    register_touch("rpg_rocket","*","fw_RpgTouch")
   
    register_message ( 23, "msg_TempEntity" )
    register_think("sentrybase", "think_sentrybase")
    register_think ( SentryClassName, "fw_ThinkSentry" )

    g_msgDamage = get_user_msgid("Damage")
    g_msgDeathMsg = get_user_msgid("DeathMsg")
    g_msgScoreInfo = get_user_msgid("ScoreInfo")
    g_msgHostagePos = get_user_msgid("HostagePos")
    g_msgHostageK = get_user_msgid("HostageK")
    gMsgID = get_user_msgid("StatusIcon")
   
    g_iMaxPlayers = get_global_int(GL_maxClients)
    g_ONEEIGHTYTHROUGHPI = 180.0 / 3.141592654
   
    expDecal = get_decal_index("{scorch1")
   
    set_task(0.2,"informer",_,_,_,"b");                                        // Информер, сколько пушек КТ и ТТ
}

public DelayRemoveAttachView(idoff){
    new id = idoff+9250
    attach_viewEx(id,id)
}
stock attach_viewEx(id,target){
    attach_view(id,target)
    if(id == target) attach_viewmod[id] = 0, remove_task(id-100)
    else attach_viewmod[id] = target, set_task ( 1.0, "set_taskdhudmessagebg", id-100, .flags = "b" )
}

public set_taskdhudmessagebg(idoff){
    new id = idoff+100
    if(!attach_viewmod[id]) remove_task(idoff)
    new Float:health = entity_get_float ( attach_viewmod[id], EV_FL_health )
    set_dhudmessage ( 255, 255, 255, 0.0, 0.75, 0, 0.0, 0.99, 0.0, 0.0 )
    static tempStatusBuffer[192]
    new level = GetSentryLevel ( attach_viewmod[id] )
    new OwnName[33]
    get_user_name ( GetSentryPeople ( attach_viewmod[id], OWNER ), OwnName, 32 )
    formatex ( tempStatusBuffer, charsmax ( tempStatusBuffer ), "Установил: %s^nЗдоровье: %d/%d^n", OwnName, floatround(health), floatround(g_HEALTHS[level]))
    show_dhudmessage(id, tempStatusBuffer)
}

public plugin_precache() {
    for(new i=0;i<sizeof(szModels);i++)
        precache_model(szModels[i])
    for(new i=0;i<sizeof(szSounds);i++)
        precache_sound(szSounds[i])
   
    g_sModelIndexFireball = precache_model("sprites/zerogxplode.spr")
    m_iTrail = precache_model("sprites/smoke.spr")
    g_Sprt_Tok = precache_model("sprites/tok.spr")
}

public plugin_natives ()
{
    register_native ("get_sentry_people", "native_get_sentry_people", 1)
    register_native ("get_sentry_team", "native_get_sentry_team", 1)
    register_native ("get_SentryCount", "nativegetSentryCount", 1)
   
    register_native ("get_SentryLevel", "nativeget_SentryLevel", 1)
    register_native ("set_freezSentry", "native_CreateSentry", 1)
    register_native ("set_sentrGrand", "native_CreateSentry", 1)
    register_native ("native_create_sentry", "native_CreateSentry", 1)
    register_native ("get_leader", "native_get_leader", 1)
}

public native_get_leader() return 1
public native_CreateSentry(id) return 0

public nativeget_SentryLevel(sentry)
{
    return GetSentryLevel ( sentry )
}

public nativegetSentryCount(id)
{
    return sentries_num[id];
}

public native_get_sentry_people ( sentry, who )
{
    return GetSentryPeople ( sentry, who )
}

public CsTeams:native_get_sentry_team ( sentry )
{
    return GetSentryTeam ( sentry )
}

public informer(id)
{
    set_dhudmessage(255, 0, 0, 0.41, 0.0, 0 , 0.0, 0.1,0.1,0.1);
    show_dhudmessage(id, "Пушек ТТ [%d] ",g_teamsentriesNum[0]);
   
    set_dhudmessage(29, 224, 215, 0.54, 0.0, 0, 0.0, 0.1,0.1,0.1);
    show_dhudmessage(id, "[%d] Пушек КТ",g_teamsentriesNum[1]);
}

public ev_Spectation()
{
    new id = read_data (1)

    if(is_user_connected(id) && cs_get_user_team(id) == CS_TEAM_SPECTATOR)
        sentry_detonate_by_owner(id)
}

public fw_TakeDamage ( ent, idinflictor, idattacker, Float:damage, damagebits)
{
    if ( !is_valid_ent ( ent ) )
        return HAM_IGNORED
   
    new sClassname[11]
    pev ( ent, pev_classname, sClassname, charsmax ( sClassname ) )
   
    if ( equal ( sClassname, SentryClassName ) || equal ( sClassname, "sentrybase" ) )
    {
        if ( sClassname[6] == 'b' )
            ent = entity_get_edict(ent, BASE_ENT_SENTRY)
       
        if ( is_valid_ent ( ent ) )
        {
            new iOwner = GetSentryPeople ( ent, OWNER )
           
            if ( !is_user_connected ( iOwner ) || !is_valid_player ( iOwner ) || !is_user_connected ( idattacker ) || !is_valid_player ( idattacker ) )
                return HAM_SUPERCEDE
       
            if ( cs_get_user_team ( iOwner ) == cs_get_user_team ( idattacker ) && idattacker != iOwner )
                return HAM_SUPERCEDE
           
            if ( pev ( ent, pev_health ) - damage <= 0.0 )
            {
                ChatColor ( idattacker, "^3[^4Информация^3]^1 Вы получили ^4%i^1$ за разрушение пушки", NAGRADA)
                cs_set_user_money ( idattacker, cs_get_user_money ( idattacker ) + NAGRADA)
            }
            cs_set_user_money(idattacker,cs_get_user_money(idattacker) + (floatround(damage)-10))
        }
    }
    return HAM_IGNORED  
}

public cmd_CreateSentry ( id )
{
    new iSentry = AimingAtSentry ( id )

    if ( iSentry && entity_range ( iSentry, id ) <= MAXUPGRADERANGE )
        SentryUpgrade ( id, iSentry )
    else
        SentryBuild ( id )

    return PLUGIN_HANDLED
}

public SentryBuild ( id )
{
    if ( !is_user_alive ( id ) )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Мертвым нельзя ставить пушку!" )
        return
    }
    new iSentryCount = sentries_num[id]
    if (get_user_flags(id) & FLAG_VIP)
    {
        if ( iSentryCount >= MAX_VIP)
        {
            ChatColor ( id, "^3[^4Информация^3]^1 Увы, ты уже построил %d пушки!", MAX_VIP)
            return
        }
    }
    else
    {
        if ( iSentryCount >= MAX_PL)
        {
            ChatColor ( id, "^3[^4Информация^3]^1 Нельзя установить более %d пушки!", MAX_PL)
            return
        }
    }
    if ( g_inBuilding[id] )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Эй, не так быстро..." )
        return
    }
    if ( !is_entity_on_ground ( id ) )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Встань на землю, чтобы установить пушку!" )
        return
    }
   
    if (get_user_flags(id) & FLAG_VIP)
    {  
        if ( cs_get_user_money ( id ) < g_SENTRYCOST_VIP[iSentryCount] )
        {
            client_print(id, print_center, "У тебя не хватает денег! (нужно %d$)", g_SENTRYCOST_VIP[iSentryCount] )
            return
        }
    }
    else
    {  
        if ( cs_get_user_money ( id ) < g_SENTRYCOST[iSentryCount] )
        {
            client_print(id, print_center, "У тебя не хватает денег! (нужно %d$)", g_SENTRYCOST[iSentryCount] )
            return
        }
    }
    new Float:fPlayerOrigin[3], Float:fOrigin[3], Float:fAngle[3]
    pev ( id, pev_origin, fPlayerOrigin )
    pev ( id, pev_angles, fAngle )
    fOrigin = fPlayerOrigin

    fOrigin[0] += floatcos ( fAngle[1], degrees ) * PLACE_RANGE
    fOrigin[1] += floatsin ( fAngle[1], degrees ) * PLACE_RANGE
    fOrigin[0] += floatcos ( fAngle[0], degrees) * PLACE_RANGE
    fOrigin[1] += floatcos ( fAngle[1], degrees )
    fOrigin[0] -= floatsin ( fAngle[1], degrees )
    fOrigin[1] += floatcos ( fAngle[2], degrees )
    fOrigin[1] -= floatsin ( fAngle[2], degrees ) * PLACE_RANGE
    fOrigin[0] -= floatsin ( fAngle[0], degrees ) * PLACE_RANGE
    fOrigin[0] -= PLACE_RANGE

    if ( pev ( id, pev_flags ) & FL_DUCKING )
        fOrigin[2] += 18.0, fPlayerOrigin[2] += 18.0

    new tr = 0, Float:fFraction
    engfunc ( EngFunc_TraceLine, fPlayerOrigin, fOrigin, 0, id, tr )
    get_tr2 ( tr, TR_flFraction, fFraction )

    if ( fFraction != 1.0 )
    {
        ChatColor ( id, "^3[^4Информация^3]^1 Здесь не получается установить пушку!" )
        return
    }

    if ( CreateSentryBase ( fOrigin, id ) )
    {
        if (get_user_flags(id) & FLAG_VIP)
        {
            cs_set_user_money ( id, cs_get_user_money ( id ) - g_SENTRYCOST_VIP[iSentryCount] )
        }
        else
        {  
            cs_set_user_money ( id, cs_get_user_money ( id ) - g_SENTRYCOST[iSentryCount] )
        }
        ammo_hud ( id, 0 )
        sentries_num[id] += 1
        ammo_hud ( id, 1 )
    }
    else
    {
        //ChatColor ( id, "^3[^4Информация^3]^1 Пушка не ставится! %f %f %f",fOrigin[0],fOrigin[1],fOrigin[2] )
        ChatColor ( id, "^3[^4Информация^3]^1 Пушка не ставится!" )
    }
}

IncreaseSentryCount ( id, sentry )
{
    g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id]] = sentry
    g_iPlayerSentries[id]++
    new Float:fSentryOrigin[3], iSentryOrigin[3], iPlayerOrigin[3]
    entity_get_vector ( sentry, EV_VEC_origin, fSentryOrigin )
    FVecIVec ( fSentryOrigin, iSentryOrigin )

    new sName[32]
    get_user_name ( id, sName, charsmax ( sName ) )
    new CsTeams:iTeam = cs_get_user_team ( id )

    for ( new i = 1; i <= g_iMaxPlayers; i++ )
    {
        if ( !is_user_connected ( i ) || !is_user_alive ( i ) || cs_get_user_team ( i ) != iTeam || id == i )
            continue

        get_user_origin ( i, iPlayerOrigin )

        client_print ( i, print_center, "%s установил пушку в %d юнитах от вас", sName, get_distance ( iPlayerOrigin, iSentryOrigin ) )

        message_begin ( MSG_ONE_UNRELIABLE, g_msgHostagePos, .player = i )
        write_byte ( i )
        write_byte ( SENTRY_RADAR_TEAMBUILT )
        write_coord ( iSentryOrigin[0] )
        write_coord ( iSentryOrigin[1] )
        write_coord ( iSentryOrigin[2] )
        message_end ()

        message_begin ( MSG_ONE_UNRELIABLE, g_msgHostageK, .player = i )
        write_byte ( SENTRY_RADAR_TEAMBUILT )
        message_end ()
    }
}

DecreaseSentryCount ( id, sentry )
{
    for ( new i; i < g_iPlayerSentries[id]; i++ )
    {
        if ( g_iPlayerSentriesEdicts[id][i] == sentry )
        {
            g_iPlayerSentriesEdicts[id][i] = g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id] - 1]
            g_iPlayerSentriesEdicts[id][g_iPlayerSentries[id] - 1] = 0
            break
        }
    }
    g_iPlayerSentries[id]--
}

stock bool:CreateSentryBase ( Float:origin[3], creator, level = SENTRY_LEVEL_1 )
{
    if ( !CheckLocation ( origin ) )
        return false

    new Float:hitPoint[3], Float:originDown[3]
    originDown = origin
    originDown[2] = -5000.0 // dunno the lowest possible height...
    trace_line(0, origin, originDown, hitPoint)
    new Float:baDistanceFromGround = vector_distance(origin, hitPoint)

    new Float:difference = PLAYERORIGINHEIGHT - baDistanceFromGround
    if (difference < -1 * HEIGHTDIFFERENCEALLOWED || difference > HEIGHTDIFFERENCEALLOWED) return false

    new entbase = create_entity("func_breakable") // func_wall
    if (!entbase)
        return false

    #define SIZE 10.0

    new Float:fTraceEnds[5][3], Float:fTraceHit[3], iType, tr = create_tr2 ()
    fTraceEnds[0][0] = origin[0] - SIZE
    fTraceEnds[0][1] = origin[1] - SIZE
    fTraceEnds[0][2] = origin[2] + SIZE + SIZE
    fTraceEnds[1][0] = origin[0] + SIZE
    fTraceEnds[1][1] = origin[1] - SIZE
    fTraceEnds[1][2] = origin[2] + SIZE + SIZE
    fTraceEnds[2][0] = origin[0] - SIZE
    fTraceEnds[2][1] = origin[1] + SIZE
    fTraceEnds[2][2] = origin[2] + SIZE + SIZE
    fTraceEnds[3][0] = origin[0] + SIZE
    fTraceEnds[3][1] = origin[1] + SIZE
    fTraceEnds[3][2] = origin[2] + SIZE + SIZE
    fTraceEnds[4][0] = origin[0]
    fTraceEnds[4][1] = origin[1]
    fTraceEnds[4][2] = origin[2] + SIZE + SIZE

    for ( new i; i < 5; i++ )
    {
        fTraceHit = fTraceEnds[i]
        fTraceHit[2] += 40.0

        engfunc ( EngFunc_TraceLine, fTraceEnds[i], fTraceHit, 0, 0, tr )
        get_tr2 ( tr, TR_vecEndPos, fTraceHit )

        if ( fTraceHit[2] - fTraceEnds[i][2] != 40.0 )
        {
            iType = 1
            break
        }
    }

    if ( iType )
    {
        fTraceEnds[0][0] = origin[0] - SIZE
        fTraceEnds[0][1] = origin[1] - SIZE
        fTraceEnds[0][2] = origin[2] - SIZE - SIZE
        fTraceEnds[1][0] = origin[0] + SIZE
        fTraceEnds[1][1] = origin[1] - SIZE
        fTraceEnds[1][2] = origin[2] - SIZE - SIZE
        fTraceEnds[2][0] = origin[0] - SIZE
        fTraceEnds[2][1] = origin[1] + SIZE
        fTraceEnds[2][2] = origin[2] - SIZE - SIZE
        fTraceEnds[3][0] = origin[0] + SIZE
        fTraceEnds[3][1] = origin[1] + SIZE
        fTraceEnds[3][2] = origin[2] - SIZE - SIZE
        fTraceEnds[4][0] = origin[0]
        fTraceEnds[4][1] = origin[1]
        fTraceEnds[4][2] = origin[2] - SIZE - SIZE
        new Float:fMinDistance, Float:fDistance
        for ( new i; i < 5; i++ )
        {
            fTraceHit[0] = fTraceEnds[i][0]
            fTraceHit[1] = fTraceEnds[i][1]
            fTraceHit[2] = -8192.0
   
            engfunc ( EngFunc_TraceLine, fTraceEnds[i], fTraceHit, IGNORE_MONSTERS, 0, tr )
            get_tr2 ( tr, TR_vecEndPos, fTraceHit )
   
            fDistance = vector_distance ( fTraceEnds[i], fTraceHit )
   
            if ( fDistance < fMinDistance || fMinDistance <= 0.0 )
            {
                fMinDistance = fDistance
                origin[2] = fTraceHit[2]
            }
        }
    }
   
    new Float:fHighest[3]
    fHighest = origin
    fHighest[2] = 1000.0
    engfunc ( EngFunc_TraceLine, origin, fHighest, DONT_IGNORE_MONSTERS, 0, tr )
    get_tr2 ( tr, TR_vecEndPos, fHighest )
    free_tr2 ( tr )

    new healthstring[16]
    num_to_str(floatround(g_HEALTHS[0]), healthstring, 15)
    DispatchKeyValue(entbase, "health", healthstring)
    DispatchKeyValue(entbase, "material", "6")

    DispatchSpawn(entbase)
    entity_set_string(entbase, EV_SZ_classname, SentryBaseClassName)
    entity_set_model(entbase, "models/error_csdm/sentry/base.mdl") // later set according to level
    SetSentryPeople ( entbase, OWNER, creator )
   
    new Float:mins[3], Float:maxs[3]
    mins[0] = -16.0
    mins[1] = -16.0
    mins[2] = 0.0
    maxs[0] = 16.0
    maxs[1] = 16.0
    maxs[2] = floatclamp ( vector_distance ( origin, fHighest ), 128.0, 1000.0 ) // Set to 16.0 later.
    entity_set_size(entbase, mins, maxs)
    entity_set_origin(entbase, origin)
   
    entity_set_int(entbase, EV_INT_solid, SOLID_SLIDEBOX)
    entity_set_int(entbase, EV_INT_movetype, iType ? MOVETYPE_FLY : MOVETYPE_TOSS) // head flies base falls
    entity_set_int(entbase, BASE_INT_TEAM, _:cs_get_user_team(creator))

    new parms[4]
    parms[0] = entbase
    parms[1] = creator
    parms[2] = level
    parms[3] = iType

    if ( iType ) origin[2] += 16.0

    g_sentryOrigins[creator - 1] = origin
   
    entity_get_vector(creator, EV_VEC_angles, origin)
    origin[0] = 0.0
    entity_set_vector(entbase, EV_VEC_angles, origin)
   
    emit_sound(creator, CHAN_AUTO, "error_csdm/sentry/building.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)

    set_task(0.7, "createsentryhead", 0, parms, 4)
    g_inBuilding[creator] = true

    return true
}

public createsentryhead(parms[4])
{
    new entbase = parms[0]
    new level = parms[2]
    new creator = parms[1]
    new iType = parms[3]

    if ( !is_user_connected ( creator ) || !g_inBuilding[creator] )
    {
        if (is_valid_ent(entbase))
            remove_entity(entbase + 9287)
        sentries_num[creator]--
        return
    }
    new CsTeams:crteam = cs_get_user_team(creator)
    if ( !is_valid_team ( _:crteam ) )
    {
        if (is_valid_ent(entbase))
            remove_entity(entbase + 9287)
        sentries_num[creator]--
        return
    }

    new Float:origin[3]
    origin = g_sentryOrigins[creator - 1]

    new ent = create_entity("func_breakable")
    if (!ent)
    {
        if (is_valid_ent(entbase))
        {
            remove_entity(entbase + 9287)
        }
        return
    }

    new Float:mins[3], Float:maxs[3]
    if (is_valid_ent(entbase)) {
        mins[0] = -5.0
        mins[1] = -5.0
        mins[2] = 0.0
       
        maxs[0] = 5.0
        maxs[1] = 5.0
        maxs[2] = 16.0
        entity_set_size(entbase, mins, maxs)

        entity_set_edict(ent, SENTRY_ENT_BASE, entbase)
        entity_set_edict(entbase, BASE_ENT_SENTRY, ent)
    }
    new healthstring[16]
    num_to_str(floatround(g_HEALTHS[0]), healthstring, 15)
    DispatchKeyValue(ent, "health", healthstring)
    DispatchKeyValue(ent, "material", "6")

    DispatchSpawn(ent)
    entity_set_string(ent, EV_SZ_classname, SentryClassName)
    switch(_:crteam)
    {
        case 1:
        {
            entity_set_model(ent, "models/error_csdm/sentry/te_1.mdl")
        }
        case 2:
        {
            entity_set_model(ent, "models/error_csdm/sentry/ct_1.mdl")
        }
    }
   
   
    mins[0] = -10.0
    mins[1] = -10.0
    mins[2] = 0.0
    maxs[0] = 10.0
    maxs[1] = 10.0
    maxs[2] = 48.0
    entity_set_size(ent, mins, maxs)
    entity_set_origin(ent, origin)
    entity_get_vector(creator, EV_VEC_angles, origin)
    origin[0] = 0.0
    origin[1] += 180.0
    entity_set_float(ent, SENTRY_FL_ANGLE, origin[1])
    origin[2] = 0.0
    entity_set_vector(ent, EV_VEC_angles, origin)
    entity_set_int(ent, EV_INT_solid, SOLID_SLIDEBOX) // SOLID_SLIDEBOX
    entity_set_int(ent, EV_INT_movetype, iType ? MOVETYPE_FLY : MOVETYPE_TOSS) // head flies, base doesn't

    SetSentryPeople(ent, OWNER, creator)
   
    SetSentryTeam ( ent, crteam )
    SetSentryLevel ( ent, level )

    g_teamsentriesNum[_:crteam-1]++
   
    emit_sound(ent, CHAN_AUTO, "error_csdm/sentry/turrset.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)

    IncreaseSentryCount(creator, ent)

    new directions = (random_num(0, 1)<<SENTRY_DIR_CANNON)
    SetSentryPenddir ( ent, directions )

    g_inBuilding[creator] = false

    if (!is_valid_ent(entbase))
        SetSentryFiremode ( ent, SENTRY_FIREMODE_NUTS )

    entity_set_float ( ent, SENTRY_FL_LASTTHINK, get_gametime () + g_THINKFREQUENCIES )
    entity_set_float ( ent, EV_FL_nextthink, get_gametime () + 0.01 )

    /*static bool:bHamRegistred

    if ( !bHamRegistred )
    {
        RegisterHamFromEntity ( Ham_Think, ent, "fw_ThinkSentry", 1 )
        bHamRegistred = true
    }*/
}

stock bool:CheckLocation ( const Float:origin[3] )
{
    if ( engfunc ( EngFunc_PointContents, origin ) != CONTENTS_EMPTY )
        return false

    new tr = create_tr2 ()

    engfunc ( EngFunc_TraceHull, origin, origin, 0, HULL_HEAD/*HUMAN*/, 0, tr )
    if ( !get_tr2 ( tr, TR_InOpen ) || get_tr2 ( tr, TR_StartSolid ) || get_tr2 ( tr, TR_AllSolid ) )
    {
        free_tr2 ( tr )
        return false
    }

    #define SIZE 10.0

    new Float:fTraceEnds[9][3], Float:fTraceHit[3], iHitEnt
    fTraceEnds[0][0] = origin[0]
    fTraceEnds[0][1] = origin[1]
    fTraceEnds[0][2] = origin[2] - SIZE - SIZE
    fTraceEnds[1][0] = origin[0] - SIZE
    fTraceEnds[1][1] = origin[1] - SIZE
    fTraceEnds[1][2] = origin[2] - SIZE - SIZE
    fTraceEnds[2][0] = origin[0] + SIZE
    fTraceEnds[2][1] = origin[1] - SIZE
    fTraceEnds[2][2] = origin[2] - SIZE - SIZE
    fTraceEnds[3][0] = origin[0] - SIZE
    fTraceEnds[3][1] = origin[1] + SIZE
    fTraceEnds[3][2] = origin[2] - SIZE - SIZE
    fTraceEnds[4][0] = origin[0] + SIZE
    fTraceEnds[4][1] = origin[1] + SIZE
    fTraceEnds[4][2] = origin[2] - SIZE - SIZE
    fTraceEnds[5][0] = origin[0] - SIZE
    fTraceEnds[5][1] = origin[1] - SIZE
    fTraceEnds[5][2] = origin[2] + SIZE + SIZE
    fTraceEnds[6][0] = origin[0] + SIZE
    fTraceEnds[6][1] = origin[1] - SIZE
    fTraceEnds[6][2] = origin[2] + SIZE + SIZE
    fTraceEnds[7][0] = origin[0] - SIZE
    fTraceEnds[7][1] = origin[1] + SIZE
    fTraceEnds[7][2] = origin[2] + SIZE + SIZE
    fTraceEnds[8][0] = origin[0] + SIZE
    fTraceEnds[8][1] = origin[1] + SIZE
    fTraceEnds[8][2] = origin[2] + SIZE + SIZE

    for (new i = 0, b = 0; i < 9; i++)
    {
        if ( engfunc ( EngFunc_PointContents, fTraceEnds[i] ) != CONTENTS_EMPTY )
        {
            free_tr2 ( tr )
            return false
        }

        engfunc ( EngFunc_TraceLine, origin, fTraceEnds[i], 0, 0, tr )
        iHitEnt = get_tr2 ( tr, TR_pHit )

        if ( iHitEnt != -1 )
        {
            free_tr2 ( tr )
            return false
        }

        get_tr2 ( tr, TR_vecEndPos, fTraceHit )

        for ( b = 0; b < 3; b++ )
        {
            if ( fTraceEnds[i][b] != fTraceHit[b] )
            {
                free_tr2 ( tr )
                return false
            }
        }
    }
    free_tr2 ( tr )
    return true
}

bool:sentry_pendulum ( sentry )
{
    switch ( GetSentryFiremode ( sentry ) )
    {
        case SENTRY_FIREMODE_NO:
        {
            new Float:fAngles[3]
            entity_get_vector ( sentry, EV_VEC_angles, fAngles )
            new Float:fBaseAngle = entity_get_float ( sentry, SENTRY_FL_ANGLE )
            new iDirections = GetSentryPenddir ( sentry )
           
            if ( iDirections & (1<<SENTRY_DIR_CANNON) )
            {
                fAngles[1] -= ( PENDULUM_INCREMENT * 0.01 )
                if ( fAngles[1] < fBaseAngle - PENDULUM_MAX )
                {
                    fAngles[1] = fBaseAngle - PENDULUM_MAX
                    iDirections &= ~(1<<SENTRY_DIR_CANNON)
                    SetSentryPenddir ( sentry, iDirections )
                }
            }
            else
            {
                fAngles[1] += ( PENDULUM_INCREMENT * 0.01 )
                if ( fAngles[1] > fBaseAngle + PENDULUM_MAX )
                {
                    fAngles[1] = fBaseAngle + PENDULUM_MAX
                    iDirections |= (1<<SENTRY_DIR_CANNON)
                    SetSentryPenddir ( sentry, iDirections )
                }
            }

            entity_set_vector ( sentry, EV_VEC_angles, fAngles )
            return true
        }
        case SENTRY_FIREMODE_NUTS:
        {
            new Float:fAngles[3]
            entity_get_vector ( sentry, EV_VEC_angles, fAngles )

            new Float:fSpinSpeed = entity_get_float ( sentry, SENTRY_FL_SPINSPEED )
            if ( GetSentryPenddir ( sentry ) & (1<<SENTRY_DIR_CANNON) )
            {
                fAngles[1] -= ( fSpinSpeed * 0.01 )
                if ( fAngles[1] < 0.0 )
                    fAngles[1] = 360.0 + fAngles[1]
            }
            else
            {
                fAngles[1] += ( fSpinSpeed * 0.01 )
                if ( fAngles[1] > 360.0 )
                    fAngles[1] = fAngles[1] - 360.0
            }
            entity_set_float ( sentry, SENTRY_FL_SPINSPEED, ( fSpinSpeed += random_float ( 1.0, 2.0 ) ) )

            new Float:fMaxSpin = entity_get_float ( sentry, SENTRY_FL_MAXSPIN )
            if ( fMaxSpin == 0.0 )
            {
                entity_set_float ( sentry, SENTRY_FL_LASTTHINK, 0.5 )
                entity_set_float ( sentry, SENTRY_FL_MAXSPIN, fMaxSpin = random_float ( 500.0, 750.0 ) )
            }
            else if ( fSpinSpeed >= fMaxSpin )
            {
                sentry_detonate ( sentry, false, false )
                return false
            }
            entity_set_vector ( sentry, EV_VEC_angles, fAngles )
            return true
        }
    }
    return true
}

//#define    TE_TRACER            6        // tracer effect from point to point
tracer(Float:start[3], Float:end[3]) {
    new start_[3], end_[3]
    FVecIVec(start, start_)
    FVecIVec(end, end_)
    message_begin(MSG_BROADCAST, SVC_TEMPENTITY) //  MSG_PAS MSG_BROADCAST
    write_byte(TE_TRACER)
    write_coord(start_[0])
    write_coord(start_[1])
    write_coord(start_[2])
    write_coord(end_[0])
    write_coord(end_[1])
    write_coord(end_[2])
    message_end()
}
stock create_explosion(Float:origin_[3]) {
    new origin[3]
    FVecIVec(origin_, origin)

    message_begin(MSG_BROADCAST, SVC_TEMPENTITY, origin) // MSG_PAS not really good here
    write_byte(TE_EXPLOSION)
    write_coord(origin[0])
    write_coord(origin[1])
    write_coord(origin[2])
    write_short(g_sModelIndexFireball)
    write_byte(random_num(0, 20) + 50) // scale * 10 // random_num(0, 20) + 20
    write_byte(12) // framerate
    write_byte(TE_EXPLFLAG_NONE)
    message_end()

    KnockBack ( origin_ )

    new Float:playerOrigin[3], Float:distance, Float:flDmgToDo, Float:dmgbase = 90.0, newHealth
    for (new i = 1; i <= g_iMaxPlayers; i++) {
        if (!is_user_alive(i) || get_user_godmode(i))
            continue

        entity_get_vector(i, EV_VEC_origin, playerOrigin)
        distance = vector_distance(playerOrigin, origin_)
        if (distance <= SENTRYEXPLODERADIUS) {
            flDmgToDo = dmgbase - (dmgbase * (distance / SENTRYEXPLODERADIUS))
            newHealth = get_user_health(i) - floatround(flDmgToDo)
            if (newHealth <= 0) {
                set_task(0.0, "TicketToHell", i)
                continue
            }

            set_user_health(i, newHealth)

            message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, i)
            write_byte(floatround(flDmgToDo))
            write_byte(floatround(flDmgToDo))
            write_long(DMG_BLAST)
            write_coord(origin[0])
            write_coord(origin[1])
            write_coord(origin[2])
            message_end()
        }
    }
}

public TicketToHell(player) {
    if (!is_user_connected(player))
        return
    new frags = get_user_frags(player)
    user_kill(player, 1) // don't decrease frags
    new parms[4]
    parms[0] = player
    parms[1] = frags
    parms[2] = cs_get_user_deaths(player)
    parms[3] = int:cs_get_user_team(player)
    set_task(0.0, "DelayedScoreInfoUpdate", 0, parms, 4)
}

public DelayedScoreInfoUpdate(parms[4]) {
    scoreinfo_update(parms[0], parms[1], parms[2], parms[3])
}

KnockBack ( Float:origin[3] )
{
    new iEntList[32]
    new iEntsFound = find_sphere_class ( 0, "player", SENTRYEXPLODERADIUS, iEntList, g_iMaxPlayers, origin )

    if ( !iEntsFound )
        return

    new Float:fOriginEnt[3]
    new Float:fVelocity[3]
    new Float:fOriginEnd[3]
    new Float:fDistance
    new iPlayer

    for ( new i; i < iEntsFound; i++ )
    {
        iPlayer = iEntList[i]

        if ( !is_user_alive ( iPlayer ) )
            continue

        entity_get_vector ( iPlayer, EV_VEC_origin, fOriginEnt )

        fDistance = vector_distance ( fOriginEnt, origin )

        if ( is_entity_on_ground ( iPlayer ) && fOriginEnt[2] < origin[2] )
            fOriginEnt[2] = origin[2] + fDistance

        entity_get_vector ( iPlayer, EV_VEC_velocity, fVelocity )

        fOriginEnd[0] = ( fOriginEnt[0] - origin[0] ) * SENTRYEXPLODERADIUS / fDistance + origin[0]
        fOriginEnd[1] = ( fOriginEnt[1] - origin[1] ) * SENTRYEXPLODERADIUS / fDistance + origin[1]
        fOriginEnd[2] = ( fOriginEnt[2] - origin[2] ) * SENTRYEXPLODERADIUS / fDistance + origin[2]

        fVelocity[0] += ( fOriginEnd[0] - fOriginEnt[0] ) * SENTRYSHOCKPOWER
        fVelocity[1] += ( fOriginEnd[1] - fOriginEnt[1] ) * SENTRYSHOCKPOWER
        fVelocity[2] += ( fOriginEnd[2] - fOriginEnt[2] ) * SENTRYSHOCKPOWER

        entity_set_vector ( iPlayer, EV_VEC_velocity, fVelocity )
    }
}

public msg_TempEntity ()
{
    if ( get_msg_args () != 15 && get_msg_arg_int ( 1 ) != TE_BREAKMODEL )
        return PLUGIN_CONTINUE

    new ent = -1
    while ((ent = find_ent_by_class(ent, SentryClassName)))
    {
        if (entity_get_float ( ent, EV_FL_health ) <= 0.0) sentry_detonate ( ent, false, false )
    }
    return PLUGIN_CONTINUE
}

public fw_ThinkSentry ( ent )
{
    if ( !is_valid_ent ( ent ) )
        return

    static iOwner; iOwner = GetSentryPeople ( ent, OWNER )

    if ( !is_user_connected ( iOwner ) )
        return

    if ( cs_get_user_team ( iOwner ) == CS_TEAM_SPECTATOR )
    {
        sentry_detonate ( ent, true, false )
        return
    }
   
    if ( !sentry_pendulum ( ent ) )
        return

    static Float:fGameTime; fGameTime = get_gametime ()
    if ( entity_get_float ( ent, SENTRY_FL_LASTTHINK ) <= fGameTime )
    {
        new Float:fOriginSentry[3], Float:fOriginHit[3], iHitEnt
        entity_get_vector ( ent, EV_VEC_origin, fOriginSentry )
        fOriginSentry[2] += CANNONHEIGHTFROMFEET // Move up some, this should be the Y origin of the cannon
   
        new firemode = GetSentryFiremode ( ent )
        new target = GetSentryTarget ( ent, TARGET )
        if ( firemode == SENTRY_FIREMODE_YES && is_valid_ent ( target ) && is_user_alive ( target ) && cs_get_user_team ( target ) != GetSentryTeam ( ent ) && !IsInSphere ( target ))
        {
            new Float:fOriginTarget[3]
            entity_get_vector ( target, EV_VEC_origin, fOriginTarget )
   
            if ( entity_get_int ( target, EV_INT_flags ) & FL_DUCKING )
                fOriginTarget[2] += TARGETUPMODIFIER
   
            iHitEnt = trace_line ( ent, fOriginSentry, fOriginTarget, fOriginHit )
            if ( iHitEnt == entity_get_edict ( ent, SENTRY_ENT_BASE ) )
                iHitEnt = trace_line ( iHitEnt, fOriginHit, fOriginTarget, fOriginHit )
   
            if ( iHitEnt != target && is_user_alive ( iHitEnt ) && GetSentryTeam ( ent ) != cs_get_user_team ( iHitEnt ) && !IsInSphere ( iHitEnt ))
            {
                target = iHitEnt
                SetSentryTarget(ent, TARGET, iHitEnt)
            }
            if ( iHitEnt == target )
            {
                SentryTurnToTarget ( ent, fOriginSentry, fOriginTarget )
                new Float:fHitRatio = random_float ( 0.0, 1.0 ) - g_HITRATIOS // ie 0.5 - 0.7 = -0.2, a hit and 0.8 - 0.7 = a miss by 0.1
                new iLevel = GetSentryLevel(ent)
                if(iLevel == SENTRY_LEVEL_4){
                    if(target)
                    {
                        if(get_gametime() > entity_get_float(ent,SENTRY_ROCKET_TIME)){
                            if(entity_range(ent,target) >= RPG_DISTANCE){
                                new data[2]
                                data[0] = ent
                               
                                ShootRockets(data)
                            }
                           
                            entity_set_float(ent,SENTRY_ROCKET_TIME,get_gametime() +  2.0)
                        }
                    }
                } else if(iLevel == SENTRY_LEVEL_5) {
                    FireZevs(fOriginSentry,fOriginTarget);
                    emit_sound ( ent, CHAN_WEAPON, "weapons/tok.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
                }
                else emit_sound ( ent, CHAN_WEAPON, "error_csdm/sentry/fire.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
               
                if ( !get_user_godmode ( target ) && fHitRatio <= 0.0 )
                    sentry_damagetoplayer ( ent, fOriginSentry, target )
                else
                {
                    new Float:fSentryAngle[3] = {0.0, 0.0, 0.0}
   
                    new Float:x = fOriginHit[0] - fOriginSentry[0]
                    new Float:z = fOriginHit[1] - fOriginSentry[1]
                    new Float:radians = floatatan ( z/x, radian )
                    fSentryAngle[1] = radians * g_ONEEIGHTYTHROUGHPI
                    if ( fOriginHit[0] < fOriginSentry[0] )
                        fSentryAngle[1] -= 180.0
   
                    new Float:h = fOriginHit[2] - fOriginSentry[2]
                    new Float:b = vector_distance ( fOriginSentry, fOriginHit )
                    radians = floatatan ( h/b, radian )
                    fSentryAngle[0] = radians * g_ONEEIGHTYTHROUGHPI
   
                    fSentryAngle[0] += random_float ( -10.0 * fHitRatio, 10.0 * fHitRatio ) // aim is a little off here :-)
                    fSentryAngle[1] += random_float ( -10.0 * fHitRatio, 10.0 * fHitRatio ) // aim is a little off here :-)
                    engfunc ( EngFunc_MakeVectors, fSentryAngle )
                    new Float:vector[3]
                    get_global_vector ( GL_v_forward, vector )
                    for ( new i = 0; i < 3; i++ )
                        vector[i] *= 1000
   
                    new Float:traceEnd[3]
                    for ( new i = 0; i < 3; i++ )
                        traceEnd[i] = vector[i] + fOriginSentry[i]
   
                    new iHitEnt2 = ent
                    static lolcheck = 0
                    while ( ( iHitEnt2 = trace_line ( iHitEnt2, fOriginHit, traceEnd, fOriginHit ) ) )
                        if ( lolcheck++ > 700 ) break
   
                }
                tracer ( fOriginSentry, fOriginHit )

                entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + 0.1 )
                entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
                return
            }
            else
            {
                SetSentryFiremode ( ent, SENTRY_FIREMODE_NO )
                new Float:fAngle[3]
                entity_get_vector(ent, EV_VEC_angles, fAngle)
                fAngle[0] = 0.0
                entity_set_vector(ent, EV_VEC_angles, fAngle)
            }
        }
        else if ( firemode == SENTRY_FIREMODE_NUTS )
        {
            new iHitEnt2 = EntViewHitPoint ( ent, fOriginSentry, fOriginHit )
            emit_sound(ent, CHAN_WEAPON, "error_csdm/sentry/fire.wav", 1.0, ATTN_NORM, 0, PITCH_NORM)
            tracer(fOriginSentry, fOriginHit)
   
            if (is_user_connected(iHitEnt2) && is_user_alive(iHitEnt2) && !get_user_godmode(iHitEnt2))
            {
                sentry_damagetoplayer(ent, fOriginSentry, iHitEnt2)
            }
            entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + 0.1 )
            entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
            return
        }
   
        if ( random_num ( 0, 99 ) < 10 )
            emit_sound ( ent, CHAN_AUTO, "error_csdm/sentry/turridle.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )

        new closestTarget = 0, Float:closestDistance, Float:distance, Float:closestOrigin[3], Float:playerOrigin[3], CsTeams:sentryTeam = GetSentryTeam ( ent )
        for ( new i = 1; i <= g_iMaxPlayers; i++ )
        {
            if ( !is_user_connected ( i ) || !is_user_alive ( i ) || cs_get_user_team ( i ) == sentryTeam || IsInSphere ( i ) )
                continue
   
            entity_get_vector ( i, EV_VEC_origin, playerOrigin )
   
            if ( entity_get_int ( i, EV_INT_flags ) & FL_DUCKING )
                playerOrigin[2] += TARGETUPMODIFIER
   
            iHitEnt = trace_line ( ent, fOriginSentry, playerOrigin, fOriginHit )
            if ( iHitEnt == entity_get_edict ( ent, SENTRY_ENT_BASE ) )
                iHitEnt = trace_line(iHitEnt, fOriginHit, playerOrigin, fOriginHit)
   
            if ( iHitEnt == i )
            {
                distance = vector_distance ( fOriginSentry, playerOrigin )
                closestOrigin = playerOrigin
   
                if ( distance < closestDistance || closestTarget == 0 )
                {
                    closestTarget = i
                    closestDistance = distance
                }
            }
        }
   
        if ( closestTarget )
        {
            emit_sound ( ent, CHAN_AUTO, "error_csdm/sentry/turrspot.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
            SentryTurnToTarget ( ent, fOriginSentry, closestOrigin )
   
            SetSentryFiremode ( ent, SENTRY_FIREMODE_YES )
            SetSentryTarget ( ent, TARGET, closestTarget )
        }
        else
        {
            SetSentryFiremode ( ent, SENTRY_FIREMODE_NO )
            new Float:fAngle[3]
            entity_get_vector(ent, EV_VEC_angles, fAngle)
            fAngle[0] = 0.0
            entity_set_vector(ent, EV_VEC_angles, fAngle)
        }

        entity_set_float ( ent, SENTRY_FL_LASTTHINK, fGameTime + g_THINKFREQUENCIES )
    }
    entity_set_float ( ent, EV_FL_nextthink, fGameTime + 0.01 )
}

public think_sentrybase(sentrybase) {
    sentrybase_broke(sentrybase)
    return PLUGIN_CONTINUE
}

sentrybase_broke(sentrybase) {
    new sentry = entity_get_edict(sentrybase, BASE_ENT_SENTRY)
    if (is_valid_ent(sentrybase))
        remove_entity(sentrybase)

    if (sentry == 0)
        return

    SetSentryFiremode ( sentry, SENTRY_FIREMODE_NUTS )
}

sentry_detonate(sentry, bool:quiet, bool:isIndex) {
    new i
    if (isIndex)
    {
        i = sentry
        sentry = g_sentries[sentry]
        if (!is_valid_ent(sentry))
            return
    }
    else
    {
        if (!is_valid_ent(sentry))
            return

        for (new j = 0; j < g_sentriesNum; j++) {
            if (g_sentries[j] == sentry) {
                i = j
                break
            }
        }
    }
    //entity_set_float ( sentry, EV_FL_nextthink, 0.0 )

    new owner = GetSentryPeople(sentry, OWNER)

    if (!quiet) {
        new Float:origin[3]
        entity_get_vector(sentry, EV_VEC_origin, origin)
        create_explosion(origin)
        ChatColor ( owner, "^3[^4Действие^3]^1 Твоя пушка взорвана!")
        ammo_hud(owner, 0)
        sentries_num[owner] -= 1
        ammo_hud(owner, 1)      
    }
    DecreaseSentryCount(owner, sentry)

    // Remove base first
    if (GetSentryFiremode ( sentry ) != SENTRY_FIREMODE_NUTS)
        set_task ( 0.1, "DelayRemoveEntity", entity_get_edict ( sentry, SENTRY_ENT_BASE ) )
        //remove_entity(entity_get_edict(sentry, SENTRY_ENT_BASE))

    new CsTeams:iSentryTeam = GetSentryTeam ( sentry )

    set_task ( 0.1, "DelayRemoveEntity", sentry )
    //remove_entity(sentry)
    // Put the last sentry in the deleted entity's place
    if(0 > (g_sentriesNum - 1) > MAXSENTRIES) return
    g_sentries[i] = g_sentries[g_sentriesNum - 1]
   
    g_teamsentriesNum[_:iSentryTeam-1]--
}

public DelayRemoveEntity ( ent )
{
    if ( is_valid_ent ( ent ) )
        remove_entity ( ent )
}

sentry_detonate_by_owner(owner, bool:quiet = false) {
    if(sentries_num[owner]>0)
    {
        static ent = -1
        while ((ent = find_ent_by_class(ent, SentryClassName)))
        {      
            if( GetSentryPeople(ent,OWNER) != owner) continue;
            sentry_detonate ( ent, quiet, false )
        }
    }
}

public client_disconnected(id) {
    sentry_detonate_by_owner ( id )
}

// урон игроку
stock sentry_damagetoplayer(sentry, Float:sentryOrigin[3], target) {
    new newHealth = get_user_health(target) - g_DMG

    if (newHealth <= 0) {
        new targetFrags = get_user_frags(target) + 1
        new owner = GetSentryPeople(sentry, OWNER)
       
        if(!is_user_connected(owner))
            return
       
        new ownerFrags = get_user_frags(owner) + 1
        set_user_frags(target, targetFrags) // otherwise frags are subtracted from victim for dying (!!)
        set_user_frags(owner, ownerFrags)
       
        new contributors[5]
        contributors[0] = owner
        contributors[1] = GetSentryPeople(sentry, UPGRADER_1)
        contributors[2] = GetSentryPeople(sentry, UPGRADER_2)
        contributors[3] = GetSentryPeople(sentry, UPGRADER_3)
        contributors[4] = GetSentryPeople(sentry, UPGRADER_4)
       
        for(new i ; i < sizeof contributors ; i++){
            if(!contributors[i])
                continue
               
            if(!is_user_connected(contributors[i]) || get_user_team(contributors[i]) != get_user_team(contributors[0])){
                switch(i){ // yao face
                    case 1: SetSentryPeople(sentry,UPGRADER_1,0)
                    case 2: SetSentryPeople(sentry,UPGRADER_2,0)
                    case 3: SetSentryPeople(sentry,UPGRADER_3,0)
                    case 4: SetSentryPeople(sentry,UPGRADER_4,0)
                }
               
                continue
            }
           
            // izvini 4yvak, no menya nakrilo
            cs_set_user_money(contributors[i], cs_get_user_money(contributors[i]) + (i == 0 ? SENTRYOWNERAWARD : SENTRYASSISTAWARD))
        }

        // ny ebatb kakoy frag
        message_begin(MSG_ALL, g_msgDeathMsg, {0, 0, 0} ,0)
        write_byte(owner)
        write_byte(target)
        write_byte(0)
        write_string("sentry gun")
        message_end()
       
        //add_user_exp ( owner )
        scoreinfo_update(owner, ownerFrags, cs_get_user_deaths(owner), int:cs_get_user_team(owner))
        set_msg_block(g_msgDeathMsg, BLOCK_ONCE)
    }

    set_user_health(target, newHealth)

    message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, target)
    write_byte(g_DMG)
    write_byte(g_DMG)
    write_long(DMG_BULLET)
    write_coord(floatround(sentryOrigin[0]))
    write_coord(floatround(sentryOrigin[1]))
    write_coord(floatround(sentryOrigin[2]))
    message_end()
}

scoreinfo_update(id, frags, deaths, team) {
    message_begin(MSG_ALL, g_msgScoreInfo)
    write_byte(id)
    write_short(frags)
    write_short(deaths)
    write_short(0)
    write_short(team)
    message_end()
}

/* SentryTurnToTarget ( ent, Float:sentry_origin[3], Float:closest_origin[3] )
{
    new Float:fAngle[3]
    entity_get_vector ( ent, EV_VEC_angles, fAngle )
    new Float:x = closest_origin[0] - sentry_origin[0]
    new Float:z = closest_origin[1] - sentry_origin[1]

    new Float:fRadians = floatatan ( z/x, radian )
    fAngle[1] = fRadians * g_ONEEIGHTYTHROUGHPI
    if ( closest_origin[0] < sentry_origin[0] )
        fAngle[1] -= 180.0

    entity_set_float ( ent, SENTRY_FL_ANGLE, fAngle[1] )
    entity_set_vector ( ent, EV_VEC_angles, fAngle )
} */

SentryTurnToTarget (ent, Float:sentryOrigin[3], Float:closestOrigin[3]) {
    new Float:newAngle[3]
    entity_get_vector(ent, EV_VEC_angles, newAngle)
    new Float:x = closestOrigin[0] - sentryOrigin[0]
    new Float:z = closestOrigin[1] - sentryOrigin[1]
    new Float:y = closestOrigin[2] - sentryOrigin[2]

    newAngle[1] = floatasin(z/floatsqroot(x*x+z*z), degrees)
   
    new Float:radians = floatatan(z/x, radian)
    newAngle[1] = radians * g_ONEEIGHTYTHROUGHPI
    if (closestOrigin[0] < sentryOrigin[0])
        newAngle[1] -= 180.0

    entity_set_float(ent, SENTRY_FL_ANGLE, newAngle[1])

    new Float:h = closestOrigin[2] - sentryOrigin[2]
    new Float:b = vector_distance(sentryOrigin, closestOrigin)
    radians = floatatan(h/b, radian)
    newAngle[0] = floatasin(x/floatsqroot(x*x+y*y), degrees)
    newAngle[0] = radians * g_ONEEIGHTYTHROUGHPI
   
    entity_set_vector(ent, EV_VEC_angles, newAngle)
}

AimingAtSentry ( id )
{
    if ( !is_user_alive ( id ) )
        return 0

    new hitEnt, bodyPart
    if (get_user_aiming(id, hitEnt, bodyPart) == 0.0)
        return 0

    if ( is_valid_ent ( hitEnt ) )
    {
        new classname[32], l_sentry
        entity_get_string(hitEnt, EV_SZ_classname, classname, 31)
        if (equal(classname, "sentry_base"))
            l_sentry = entity_get_edict(hitEnt, BASE_ENT_SENTRY)
        else if (equal(classname, SentryClassName))
            l_sentry = hitEnt
        else
            l_sentry = 0

        return l_sentry
    }
    return 0
}

// улучшение уровня пушки
bool:SentryUpgrade ( id, sentry )
{
    if(!is_user_alive(id)||!is_user_connected(id))
        return false
   
    if ( GetSentryFiremode ( sentry ) == SENTRY_FIREMODE_NUTS )
        return false
   
    new iLevel = GetSentryLevel ( sentry )
    if ( iLevel == SENTRY_LEVEL_5 )
        return false
   
    if ( cs_get_user_team ( id ) != GetSentryTeam ( sentry ))
    {
        return false
    }

    if ( cs_get_user_team ( GetSentryPeople ( sentry, OWNER ) ) == CS_TEAM_SPECTATOR )
        return false

    // e ron don don
    if (get_user_flags(id) & FLAG_VIP)
    {
    }
    else
    {
        // e ron don don
        if(GetSentryPeople(sentry,UPGRADER_1) == id ||
            GetSentryPeople(sentry,UPGRADER_2) == id ||
            GetSentryPeople(sentry,UPGRADER_3) == id ||
            GetSentryPeople(sentry,UPGRADER_4) == id
        )
            return false
    }
   
    iLevel++

    if (get_user_flags(id) & FLAG_VIP)
    {
        if ( cs_get_user_money ( id ) - g_COST_VIP[iLevel] < 0 )
        {
            client_print(id, print_center, "У тебя не хватает денег (нужно %d$)", g_COST_VIP[iLevel])
            return false
        }
    }
    else
    {
        if ( cs_get_user_money ( id ) - g_COST[iLevel] < 0 )
        {
            client_print(id, print_center, "У тебя не хватает денег (нужно %d$)", g_COST[iLevel])
            return false
        }
    }

    if (get_user_flags(id) & FLAG_VIP)
    {
        cs_set_user_money ( id, cs_get_user_money ( id ) - g_COST_VIP[iLevel] )
    }
    else
    {
        cs_set_user_money ( id, cs_get_user_money ( id ) - g_COST[iLevel] )
    }
   
    new iTeam = _:cs_get_user_team ( id ), iUpgraderField
    switch ( iLevel )
    {
        // this kod is very zaebisb
        case SENTRY_LEVEL_2:
        {
            switch ( iTeam )
            {
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_2.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_2.mdl" )
            }
            iUpgraderField = UPGRADER_1
        }
        case SENTRY_LEVEL_3:
        {
            switch ( iTeam )
            {
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            iUpgraderField = UPGRADER_2
        }
        case SENTRY_LEVEL_4:{
            switch(iTeam){
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            iUpgraderField = UPGRADER_3
        }
        case SENTRY_LEVEL_5:{
            switch(iTeam){
                case 1:entity_set_model ( sentry, "models/error_csdm/sentry/te_3.mdl" )
                case 2:entity_set_model ( sentry, "models/error_csdm/sentry/ct_3.mdl" )
            }
            // entity_set_byte(sentry,EV_BYTE_controller2,120)
            // entity_set_byte(sentry,EV_BYTE_controller3,120)
            iUpgraderField = UPGRADER_4
        }
    }

    new Float:fMins[3], Float:fMaxs[3]
    fMins[0] = -10.0
    fMins[1] = -10.0
    fMins[2] = 0.0
    fMaxs[0] = 10.0
    fMaxs[1] = 10.0
    fMaxs[2] = 40.0 // 4.0
   
    entity_set_size ( sentry, fMins, fMaxs )
    emit_sound ( sentry, CHAN_AUTO, "error_csdm/sentry/turrset.wav", 1.0, ATTN_NORM, 0, PITCH_NORM )
    SetSentryLevel ( sentry, iLevel )
    entity_set_float ( sentry, EV_FL_health, g_HEALTHS[iLevel] )
    entity_set_float ( entity_get_edict ( sentry, SENTRY_ENT_BASE ), EV_FL_health, g_HEALTHS[0] )
    SetSentryPeople ( sentry, iUpgraderField, id )
   
    new sName[32]
    get_user_name ( id, sName, charsmax ( sName ) )
    client_print ( GetSentryPeople ( sentry, OWNER ), print_center, "%s прокачал твою пушку до уровня %d", sName, iLevel + 1 )
    return true
}

stock EntViewHitPoint ( index, Float:origin[3], Float:hitorigin[3] )
{
    if ( !is_valid_ent ( index ) )
        return 0

    new Float:angle[3], Float:vec[3], Float:f_dest[3]

    entity_get_vector(index, EV_VEC_angles, angle)

    engfunc(EngFunc_AngleVectors, angle, vec, 0, 0)

    f_dest[0] = origin[0] + vec[0] * 9999
    f_dest[1] = origin[1] + vec[1] * 9999
    f_dest[2] = origin[2] + vec[2] * 9999

    return trace_line(index, origin, f_dest, hitorigin)
}

public fw_PlayerSpawn_Post ( id )
{
    g_inBuilding[id] = false
    sentry_detonate_by_owner ( id )
    ammo_hud ( id, 0 )
    sentries_num[id] = 0
}

public fw_TraceLine_Post ( Float:start[3], Float:end[3], noMonsters, id )
{
    if ( !is_valid_player ( id ) || !is_user_alive ( id ) )
        return FMRES_IGNORED

    new iHitEnt = get_tr ( TR_pHit )

    if ( iHitEnt <= g_iMaxPlayers )
        return FMRES_IGNORED

    new sClassName[11], sentry, base

    pev ( iHitEnt, pev_classname, sClassName, charsmax ( sClassName ) )

    if ( equal ( sClassName, "sentrybase" ) )
    {
        base = iHitEnt
        sentry = entity_get_edict ( iHitEnt, BASE_ENT_SENTRY )
    }
    else if ( equal ( sClassName, SentryClassName ) )
    {
        sentry = iHitEnt
        base = entity_get_edict ( sentry, SENTRY_ENT_BASE )
    }

    if ( !is_valid_ent ( sentry ) || !base )
        return FMRES_IGNORED
       
    if ( GetSentryFiremode ( sentry ) == SENTRY_FIREMODE_NUTS )
             return FMRES_IGNORED
       
    new Float:health = entity_get_float ( sentry, EV_FL_health )

    if ( health <= 0 )
        return FMRES_IGNORED

    new Float:basehealth = entity_get_float ( base, EV_FL_health )

    if ( basehealth <= 0 )
        return FMRES_IGNORED

    new CsTeams:team = GetSentryTeam ( sentry )

    if ( team != cs_get_user_team ( id ) )
        return FMRES_IGNORED

    new level = GetSentryLevel ( sentry )

    static tempStatusBuffer[192], tempStatusBuffer2[192]

    new OwnName[33]
    get_user_name ( GetSentryPeople ( sentry, OWNER ), OwnName, 32 )
   
    formatex ( tempStatusBuffer, charsmax ( tempStatusBuffer ), "Установил: %s^nЗдоровье: %d/%d",OwnName, floatround(health), floatround(g_HEALTHS[level]) )
    formatex ( tempStatusBuffer2, charsmax ( tempStatusBuffer2 ), "^n^nЗдоровье основания: %d/%d^nУровень: %d", floatround(basehealth), floatround(g_HEALTHS[0]), level + 1 )

    set_dhudmessage ( 255, 255, 255, -1.0, 0.75, 0, 0.0, 0.6, 0.0, 0.0 )
    show_dhudmessage(id, tempStatusBuffer)
    show_dhudmessage(id, tempStatusBuffer2)

    return FMRES_IGNORED
}

// прикосновение к пушке игрока
new Float:flood_touch[33]
public fw_TouchSentry ( sentry, player )
{
    static Float:time;time=get_gametime()
    if(flood_touch[player] < time)
    SentryUpgrade ( player, sentry );
    flood_touch[player] = time+1.0
    return 1
}

ammo_hud(id, sw)
{
    if(is_user_bot(id)||!is_user_alive(id)||!is_user_connected(id))
            return

    new s_sprite[33]
    format(s_sprite, 32, "number_%d", sentries_num[id])
    if(sw)
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 1 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }
    else
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 0 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }
    if(sentries_num[id] <= 0)
    {
        message_begin( MSG_ONE, gMsgID, {0,0,0}, id )
        write_byte( 0 ) // status
        write_string( s_sprite ) // sprite name
        write_byte( 250 ) // red
        write_byte( 250 ) // green
        write_byte( 250 ) // blue
        message_end()
    }  
}

stock ChatColor(const id, const input[], any:...)
{
        new count = 1, players[32]
        static msg[191]
        vformat(msg, 190, input, 3)
     
        replace_all(msg, 190, "!g", "^4") // Green Color
        replace_all(msg, 190, "!y", "^1") // Default Color
        replace_all(msg, 190, "!team", "^3") // Team Color
        replace_all(msg, 190, "!team2", "^0") // Team2 Color
     
        if (id) players[0] = id; else get_players(players, count, "ch")
        {
                for (new i = 0; i < count; i++)
                {
                        if (is_user_connected(players[i]))
                        {
                                message_begin(MSG_ONE_UNRELIABLE, get_user_msgid("SayText"), _, players[i])
                                write_byte(players[i]);
                                write_string(msg);
                                message_end();
                        }
                }
        }
}

bool:IsInSphere ( id )
{
    if ( !is_user_alive ( id ) )
        return false

    new ent = -1
    while ( ( ent = engfunc ( EngFunc_FindEntityByString, ent, "classname", "campo_grenade_forze" ) ) > 0 )
    {
        new iOwner = pev ( ent, pev_owner )

        if ( cs_get_user_team ( id ) != cs_get_user_team ( iOwner ) )
            continue

        new Float:fOrigin[3]
        pev ( ent, pev_origin, fOrigin )
        new iPlayer = -1
        while ( ( iPlayer = engfunc ( EngFunc_FindEntityInSphere, iPlayer, fOrigin, 68.0 ) ) != 0 )
        {
            if ( iPlayer == id )
                return true
        }
    }
    return false
}

//
// Launch rocket from 4lvl sentry
//    data[2]
//        0 - sentry id
//        1 - side    // 0 - right, 1 - left
//
public ShootRockets(data[2]){
    new sentry = data[0]
    new side = data[1]
   
    new Float:rocketOrigin[3],Float:rocketAngles[3]
   
    entity_get_vector(sentry,EV_VEC_angles,rocketAngles)
    engfunc(EngFunc_MakeVectors,rocketAngles)
   
    new Float:vecForward[3],Float:vecRight[3],Float:vecUp[3]
   
    get_global_vector(GL_v_forward,vecForward)
    xs_vec_mul_scalar(vecForward,20.0,vecForward)
   
    get_global_vector(GL_v_right,vecRight)
    xs_vec_mul_scalar(vecRight,side ? 8.0 : -8.0,vecRight) // right or left rocket
   
    get_global_vector(GL_v_up,vecUp)
    xs_vec_mul_scalar(vecUp,30.0,vecUp)
   
    entity_get_vector(sentry,EV_VEC_origin,rocketOrigin)
    xs_vec_add(rocketOrigin,vecForward,rocketOrigin)
    xs_vec_add(rocketOrigin,vecRight,rocketOrigin)
    xs_vec_add(rocketOrigin,vecUp,rocketOrigin)
   
    // shot rocket
    CreateRocket(sentry,rocketOrigin,rocketAngles,GetSentryTarget(sentry,TARGET))
   
    data[1] = 1
   
    if(!side) // shot left rocket
        set_task(0.2,"ShootRockets",_,data,sizeof data)
}

//
// Launch RPG rocket
//    sentry - sentry id
//    origin - rocket origin
//    angles - sentry angles
//    traget - rocket target id
//
CreateRocket(sentry,Float:origin[3],Float:angles[3],target){
    new rocket = create_entity("info_target")
   
    entity_set_string(rocket,EV_SZ_classname,"rpg_rocket")
   
    entity_set_int(rocket,EV_INT_movetype,MOVETYPE_FLY)
    entity_set_int(rocket,EV_INT_solid,SOLID_BBOX)
   
    entity_set_edict(rocket,EV_ENT_owner,sentry)
    entity_set_edict(rocket,EV_ENT_euser4,GetSentryPeople(sentry,OWNER))
   
    entity_set_size(rocket,Float:{-2.0,-2.0,-2.0},Float:{2.0,2.0,2.0})
    entity_set_origin(rocket,origin)
   
    new Float:targetOrigin[3]
    entity_get_vector(target,EV_VEC_origin,targetOrigin)
    angles[0] = -GetAngleOrigins(origin,targetOrigin)
    entity_set_model(rocket,"models/error_csdm/sentry/rpgrocket.mdl")
    entity_set_vector(rocket,EV_VEC_angles,angles)
   
    engfunc(EngFunc_MakeVectors,angles)
   
    new Float:vecVelocity[3]
    get_global_vector(GL_v_forward,vecVelocity)
    xs_vec_mul_scalar(vecVelocity,1000.0,vecVelocity)
    entity_set_vector(rocket,EV_VEC_velocity,vecVelocity)
   
    entity_set_int(rocket,EV_INT_effects,entity_get_int(rocket,EV_INT_effects) | EF_LIGHT)
   
    // rocket trail
    message_begin(MSG_BROADCAST,SVC_TEMPENTITY)
    write_byte(TE_BEAMFOLLOW)
    write_short(rocket)
    write_short(m_iTrail)
    write_byte(10)
    write_byte(5)
    write_byte(224)
    write_byte(224)
    write_byte(255)
    write_byte(255)
    message_end()
   
    emit_sound(rocket,CHAN_VOICE,"weapons/rocket1.wav",1.0,0.5,0,PITCH_NORM)
}

public fw_RpgTouch(rocket,ent){
    new Float:origin[3],Float:angles[3],Float:vecPlaneNormal[3]
    entity_get_vector(rocket,EV_VEC_origin,origin)
    entity_get_vector(rocket,EV_VEC_angles,angles)
   
    engfunc(EngFunc_MakeVectors,angles)
    get_global_vector(GL_v_forward,angles)
    xs_vec_mul_scalar(angles,9999.0,angles)
    xs_vec_add(origin,angles,angles)
    engfunc(EngFunc_TraceLine,origin,angles,0,rocket,0)
   
    get_tr2(0,TR_vecEndPos,origin)
   
    message_begin_f(MSG_BROADCAST,SVC_TEMPENTITY,origin,0)
    write_byte(TE_WORLDDECAL)
    write_coord_f(origin[0])
    write_coord_f(origin[1])
    write_coord_f(origin[2])
    write_byte(expDecal)
    message_end()
   
    get_tr2(0,TR_vecPlaneNormal,vecPlaneNormal)
   
    xs_vec_mul_scalar(vecPlaneNormal,8.0,vecPlaneNormal)
    xs_vec_add(origin,vecPlaneNormal,origin)
   
    message_begin_f(MSG_PVS,SVC_TEMPENTITY,origin,0)
    write_byte(TE_EXPLOSION)
    write_coord_f(origin[0])
    write_coord_f(origin[1])
    write_coord_f(origin[2])
    write_short(g_sModelIndexFireball)
    write_byte(20)
    write_byte(15)
    write_byte(0)
    message_end()
    shit_radiusdamage(rocket,origin)
    emit_sound(rocket,CHAN_VOICE,"weapons/rocket1.wav",0.0,ATTN_NORM,SND_STOP,0)
}

// this very bad method
stock shit_radiusdamage(rocket,Float:origin_[3]) {
    new origin[3]
    FVecIVec(origin_, origin)
   
    new attacker = entity_get_edict(rocket,EV_ENT_euser4)
   
    if(!is_user_connected(attacker))
        return


    new Float:playerOrigin[3], Float:distance, Float:flDmgToDo, Float:dmgbase = RPG_DAMAGE
    new sentry = entity_get_edict(rocket,EV_ENT_owner)
    for (new i = 1; i <= g_iMaxPlayers; i++) {
        if (!is_user_alive(i) || get_user_godmode(i) || get_user_team(i) == get_user_team(attacker))
            continue

        entity_get_vector(i, EV_VEC_origin, playerOrigin)
        distance = vector_distance(playerOrigin, origin_)
        if (distance <= RPG_RADIUS) {
            flDmgToDo = dmgbase - (dmgbase * (distance / RPG_RADIUS))
           
            // zemletryasenie!!111
            Util_ScreenShake(i,0.5,16.0,16.0)
            rocket_damagetoplayer(sentry,origin_,i,flDmgToDo)
        }
    }
    remove_entity(rocket)
}
// ScreenShake
stock Util_ScreenShake(id, Float:duration, Float:frequency, Float:amplitude)
{
    static ScreenShake = 0;
    if( !ScreenShake )
    {
        ScreenShake = get_user_msgid("ScreenShake");
    }
    message_begin( id ? MSG_ONE_UNRELIABLE : MSG_BROADCAST, ScreenShake, _, id);
    write_short( FixedUnsigned16( amplitude, 1<<12 ) ); // shake amount
    write_short( FixedUnsigned16( duration, 1<<12 ) ); // shake lasts this long
    write_short( FixedUnsigned16( frequency, 1<<8 ) ); // shake noise frequency
    message_end();
}

// урон игроку
stock rocket_damagetoplayer(sentry, Float:sentryOrigin[3], target, Float:dmg) {
    new newHealth = get_user_health(target) - floatround(dmg)

    if (newHealth <= 0) {
        new targetFrags = get_user_frags(target) + 1
        new owner = GetSentryPeople(sentry, OWNER)
       
        if(!is_user_connected(owner))
            return
       
        new ownerFrags = get_user_frags(owner) + 1
        set_user_frags(target, targetFrags) // otherwise frags are subtracted from victim for dying (!!)
        set_user_frags(owner, ownerFrags)
       
        new contributors[5]
        contributors[0] = owner
        contributors[1] = GetSentryPeople(sentry, UPGRADER_1)
        contributors[2] = GetSentryPeople(sentry, UPGRADER_2)
        contributors[3] = GetSentryPeople(sentry, UPGRADER_3)
        contributors[4] = GetSentryPeople(sentry, UPGRADER_4)
       
        for(new i ; i < sizeof contributors ; i++){
            if(!contributors[i])
                continue
               
            if(!is_user_connected(contributors[i]) || get_user_team(contributors[i]) != get_user_team(contributors[0])){
                switch(i){ // yao face
                    case 1: SetSentryPeople(sentry,UPGRADER_1,0)
                    case 2: SetSentryPeople(sentry,UPGRADER_2,0)
                    case 3: SetSentryPeople(sentry,UPGRADER_3,0)
                    case 4: SetSentryPeople(sentry,UPGRADER_4,0)
                }
               
                continue
            }
           
            // izvini 4yvak, no menya nakrilo
            cs_set_user_money(contributors[i],
                clamp(
                    cs_get_user_money(contributors[i]) + (i == 0 ? SENTRYOWNERAWARD : SENTRYASSISTAWARD),
                    0,
                    16000
                )
            )
        }

        // ny ebatb kakoy frag
        message_begin(MSG_ALL, g_msgDeathMsg, {0, 0, 0} ,0)
        write_byte(owner)
        write_byte(target)
        write_byte(0)
        write_string("sentry gun")
        message_end()

        scoreinfo_update(owner, ownerFrags, cs_get_user_deaths(owner), int:cs_get_user_team(owner))
        set_msg_block(g_msgDeathMsg, BLOCK_ONCE)
    }

    set_user_health(target, newHealth)

    message_begin(MSG_ONE_UNRELIABLE, g_msgDamage, {0,0,0}, target)
    write_byte(g_DMG)
    write_byte(g_DMG)
    write_long(DMG_BLAST)
    write_coord(floatround(sentryOrigin[0]))
    write_coord(floatround(sentryOrigin[1]))
    write_coord(floatround(sentryOrigin[2]))
    message_end()
}

FireZevs(Float:starting[3], Float:ending[3])
{
    new OriginStr[3],OriginEnd[3];
    //Еффект луча от пушки до цели.
    FVecIVec(starting,OriginStr)
    FVecIVec(ending,OriginEnd)
    message_begin(MSG_BROADCAST, SVC_TEMPENTITY);
    write_byte(TE_BEAMPOINTS);
    write_coord(OriginStr[0]);
    write_coord(OriginStr[1]);
    write_coord(OriginStr[2]);
    write_coord(OriginEnd[0]);
    write_coord(OriginEnd[1]);
    write_coord(OriginEnd[2]);
    write_short(g_Sprt_Tok);
    write_byte(0);
    write_byte(1);
    write_byte(1);
    write_byte(random_num(10, 50));
    write_byte(random_num(10, 25));
    write_byte(random_num(0, 255));
    write_byte(random_num(0, 255));
    write_byte(random_num(0, 255));
    write_byte(1000);
    write_byte(0);
    message_end();
}

stock FixedUnsigned16( Float:value, scale )
{
    new output;

    output = floatround(value * scale);
    if ( output < 0 )
        output = 0;
    if ( output > 0xFFFF )
        output = 0xFFFF;

    return output;
}

Float: GetAngleOrigins(Float:fOrigin1[3], Float:fOrigin2[3])
{
    new Float:fVector[3];
    new Float:fAngle[3];
    new Float:fLineAngle;
   
    xs_vec_sub(fOrigin2, fOrigin1, fVector);
    vector_to_angle(fVector, fAngle);
   
    if( fAngle[0] > 90.0 )
        fLineAngle = -(360.0 - fAngle[0]);
    else
        fLineAngle = fAngle[0];
   
    return fLineAngle;
}
 

Download all Attachments

Сообщения
173
Реакции
5
Пока ничего не понял.
Где ты менял? Что ты менял?
 

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

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