KlasterK

Система скинов через стандартный выбор класса

В теме 3 сообщения

Стандартная система выбора класса хороша, но в ней можно сделать только определённое количество скинов/классов. А ведь на сервере может быть выбор скина при регистрации, в магазине во фракции... Здесь я постараюсь наглядно обьяснить суть работы.

 

В первую очередь, нужно добавить 3 абсолютно любых класса.

Нужно создать массив нужных нам скинов, с их количеством в нулевом элементе. Например: static const bitchSkins[][] = { 4, 78, 79, 134, 135 };

В ваш енум игроков добавьте pClassID и добавьте нужную вам переменную для хранения выбора скина(например, pRegSkin).
Вставьте в ваш мод следующий код:

Спойлер



// pinfo - enum игроков
// skinid - id скина(индекс массива скинов)
// wasclass - id предыдущего класса
// skins - массив скинов

// В то место, откуда вы хотите начать выбор скина:
    pinfo[playerid][skinid] = 1;

    // Эта строка бесполезная если вы делаете выбор скина сразу после подключения игрока(регистрация)
    ForceClassSelection(playerid);

public OnPlayerRequestClass(playerid, classid) // Когда игрок нажимает кнопки '<<' или '>>'
{
    #define was pinfo[playerid][wasclass] // Макрос на прошлый id класса
    #define now classid // Макрос на текущий id класса

    if(pinfo[playerid][skinid] > 0// Проверка на выбор скина(мы на старте установили skinid в 1)
    {
        // Вправо: если было 0 и стало 1 или было 1 и стало 2 или было 2 и стало 0
        if((was == 0 && now == 1) || (was == 1 && now == 2) || (was == 2 && now == 0))
        {
            pInfo[playerid][skinid]++; // Инкремент счетчика
            if(pinfo[playerid][skinid] > skins[0]) // Если индекс скина дошел до края(0 ячейка - размер)
                pInfo[playerid][skinid] = 1// то листнуть индекс скина до первого скина в массиве
            SetPlayerSkin(playerid, skins[ pInfo[playerid][skinid] ]); // Поставить скин из массива игроку
            was = now; // Установить прошлый класс в текущий
        }
        
        // Влево: если было 0 и стало 2 или было 1 и стало 0 или было 2 и стало 1
        else if((was == 0 && now == 2) || (was == 1 && now == 0) || (was == 2 && now == 1))
        {
            pInfo[playerid][skinid]--; // Декремент счетчика
            if(pInfo[playerid][skinid] == 0// Если счетчик скина дошёл до начала
                pInfo[playerid][skinid] = bitchSkins[0]; // то листнуть счетчик до последнего скина в массиве(0 ячейка - размер)
            SetPlayerSkin(playerid, skins[ pInfo[playerid][skinid] ]); // Поставить скин из массива игроку
            was = now; // Установить прошлый класс в текущий
        }
    }
}

public OnPlayerRequestSpawn(playerid) // Когда игрок нажимает кнопку 'Spawn'
{
    // В пунктах с защитой от F4: я это не проверял, мб будет работать
    if(pinfo[playerid][skinid] > 0// если игрок выбирает себе скин
    {
        // SetSpawnInfo
        SpawnPlayer(playerid); // Спавним игрока чтобы сбросить выбор класса
        // SetPlayerPos/FacingAngle/Interior/VirtualWorld/...
        return 0// Если вы хотите защиты от F4 :: Отменяем спавн игрока
    }
    return 0// Не забудьте вернуть 0 чтобы паблик не заспавнил ничего сам по себе

    // SetSpawnInfo
    return 1// Если вы хотите защиту от F4 :: Если игрок нажал F4, то он может просто заспавниться по нажатию кнопки
    // SetPlayerPos/FacingAngle/Interior/VirtualWorld/...
}


 

 

* Завтра добавлю пример *

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Пример: выбор скина при регистрации + смена скина в магазине

Спойлер

#include <a_samp>
#include <dc_cmd>

enum pInfo_t { pClassID, pRegSkin, pShopSkin } // Енум только со всем необходимым

static pInfo[MAX_PLAYERS][pInfo_t]; // Массив енума

static const bitchSkins[] = {               // Список скинов бомжей
    16,                                     // Размер
    78,  79,  134135137212230239// Мужские
    77,  10,  31,  39,  54,  64,  89,  151// Женские
};

static const pedSkins[] = {         // Список скинов педов
    16,                             // Размер
    1,  2,  3,  5,  6,  7,  8,  14// Мужские
    9,  10111213314041// Женские
};

public OnGameModeInit()
{
    // Три обязательных класса
    AddPlayerClass(00000000000);
    AddPlayerClass(00000000000);
    AddPlayerClass(00000000000);
}

public OnPlayerConnect(playerid)
{
    SendClientMessage(playerid, -1"Выберите скин! Если захотите купить себе новый, введите команду /newskin");
    pInfo[playerid][pRegSkin] = 1// Включаем игроку выбор скина бомжа
    GivePlayerMoney(playerid, 1000); // Даем денег
}

CMD:newskin(playerid, params[])
{
    pInfo[playerid][pShopSkin] = 1;  // Включаям игроку выбор скина педа
    GivePlayerMoney(playerid, -100); // Отнимаем деньги
    ForceClassSelection(playerid);   // Включаем выбор класса
    return SendClientMessage(playerid, -1"Выберите скин! С вас снято 100$");
}

public OnPlayerRequestClass(playerid, classid)
{
    #define now    classid                    // Макрос для текущего класса
    #define was    pInfo[playerid][pClassID]  // Макрос для предыдущего класса
    #define regsk  pInfo[playerid][pRegSkin]  // Макрос для счетчика регистрационного скина
    #define shopsk pInfo[playerid][pShopSkin] // Макрос для счетчика магазинного скина
    #define left   (was == 0 && now == 1) || (was == 1 && now == 2) || (was == 2 && now == 0// Макрос для проверки лева
    #define right  (was == 0 && now == 2) || (was == 1 && now == 0) || (was == 2 && now == 1// Макрос для проверки права
    if(regsk && right) // Если выбирается скин бомжа и кнопка вправо
    {
        regsk++; // Инкремент счетчика
        if(regsk > bitchSkins[0]) regsk = 1// Если счетчик дошел до края то устанавливаем его в другой край
        SetPlayerSkin(playerid, bitchSkins[regsk]); // Ставим скин
        was = now; // Меняем переменные
    }
    else if(regsk && left) // Если выбирается скин бомжа и кнопка влево
    {
        regsk--; // Декремент счетчика
        if(regsk == 0) regsk = bitchSkins[0]; // Если счетчик дошел до края то устанавливаем его в другой край
        SetPlayerSkin(playerid, bitchSkins[regsk]); // Ставим скин
        was = now; // Меняем переменные
    }
    else if(shopsk && right) // Если выбирается скин педа и кнопка вправо
    {
        shopsk++; // Инкремент счетчика
        if(shopsk > pedSkins[0]) shopsk = 1// Если счетчик дошел до края то устанавливаем его в другой край
        SetPlayerSkin(playerid, bitchSkins[shopsk]); // Ставим скин
        was = now; // Меняем переменные
    }
    else if(shopsk && left) // Если выбирается скин педа и кнопка влево
    {
        shopsk--; // Декремент счетчика
        if(shopsk == 0) shopsk = bitchSkins[0]; // Если счетчик дошел до края то устанавливаем его в другой край
        SetPlayerSkin(playerid, bitchSkins[shopsk]); // Ставим скин
        was = now; // Меняем переменные
    }
}

public OnPlayerRequestSpawn(playerid)
{
    if(shopsk) shopsk = 0// Устанавливаем счетчик скина бомжа в ноль
    if(regsk)  regsk  = 0// Устанавливаем счетчик скина педа  в ноль

    SpawnPlayer(playerid); // Спавним игрока
    SetPlayerVirtualWorld(playerid, 0);
    SetPlayerInterior(playerid, 0);
    SetPlayerPos(playerid, 1275.0, -1665.020.0);
    SetPlayerFacingAngle(playerid, 180.0);
}

 

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты
4 часа назад, KlasterK сказал:

#define now    classid                    // Макрос для текущего класса
#define was    pInfo[playerid][pClassID]  // Макрос для предыдущего класса
#define regsk  pInfo[playerid][pRegSkin]  // Макрос для счетчика регистрационного скина
#define shopsk pInfo[playerid][pShopSkin] // Макрос для счетчика магазинного скина
#define left   (was == 0 && now == 1) || (was == 1 && now == 2) || (was == 2 && now == 0// Макрос для проверки лева
#define right  (was == 0 && now == 2) || (was == 1 && now == 0) || (was == 2 && now == 1) // Макрос для проверки права

 

Еще бы избавились от объявленных директив, и цены бы вам не было.

#undef now
#undef was
#undef regsk
#undef shopsk
#undef left
#undef right

 

Поделиться сообщением


Ссылка на сообщение
Поделиться на другие сайты

Для публикации сообщений создайте учётную запись или авторизуйтесь

Вы должны быть пользователем, чтобы оставить комментарий

Создать учетную запись

Зарегистрируйте новую учётную запись в нашем сообществе. Это очень просто!


Регистрация нового пользователя

Войти

Уже есть аккаунт? Войти в систему.


Войти

  • Последние посетители   0 пользователей онлайн

    Ни одного зарегистрированного пользователя не просматривает данную страницу

  • Похожий контент

    • Chopick
      От Chopick
      Всем здравствуйте! Делаю систему домов/бизнесов по туториалу knox know с плагином GoodArea. Есть проблема, если заходишь в интерьер бизнеса, то когда выходишь спавнит почему-то на пикап дома.

      Вот енумы:
       
      #define MAX_HOUSES 1000 #define MAX_BIZS 1000 enum house {         hID,         STREAMER_TAG_PICKUP:hPICKUP,         STREAMER_TAG_AREA:hAREA,         hOWNER[MAX_PLAYER_NAME],         STREAMER_TAG_3D_TEXT_LABEL:hTEXT[256],         STREAMER_TAG_MAP_ICON:hICON,         hPRICE,         hCLASS,         hINTERIOR,         Float:hX,         Float:hY,         Float:hZ,         Float:hA,         hCONDITION // Закрыт/Открыт } new House[MAX_HOUSES][house]; new Houses; new HouseGroup; enum biz {         bID,         STREAMER_TAG_PICKUP:bPICKUP,         STREAMER_TAG_AREA:bAREA,         bOWNER[MAX_PLAYER_NAME],         STREAMER_TAG_3D_TEXT_LABEL:bTEXT[256],         STREAMER_TAG_MAP_ICON:bICON,         bNAME[25],         bPRICE,         bTYPE,         bINTERIOR,         Float:bX,         Float:bY,         Float:bZ,         Float:bA } new Biz[MAX_BIZS][biz]; new Bizs; new BizGroup; enum interior_info {         intID,         intNAME[41],         intINTERIOR,         Float:intX,         Float:intY,         Float:intZ,         Float:intA,         STREAMER_TAG_AREA:intAREA } new Interior[MAX_INTERIORS][interior_info]; new Interiors; new InteriorGroup; Вот что у меня в OnGameModeInit:
      HouseGroup = CreateGroupGoodAreas(GoodArea:HouseEnter); InteriorGroup = CreateGroupGoodAreas(GoodArea:InteriorExit); BizGroup = CreateGroupGoodAreas(GoodArea:BizArea);  
      Вот стоки:
      GAResponse:HouseEnter(playerid, response, key, index) {         switch(response)         {             case GA_RESPONSE_PRESS_KEY:             {                 if(key & KEY_WALK)                 {                 if(House[index][hCONDITION] > 0 && House[index][hOWNER] == player_info[playerid][NAME] || House[index][hCONDITION] == 0)                 {                                         GoToInterior(playerid, House[index][hINTERIOR]);                                         SetPlayerVirtualWorld(playerid, index);                                 }                                 else if(House[index][hCONDITION] > 0 && House[index][hOWNER] != player_info[playerid][NAME]) return SCM(playerid, COLOR_RED, "[Ошибка]{ffffff}Этот дом закрыт на ключ!");                         }                 }                 case GA_RESPONSE_ENTER:                 {                     if(House[index][hOWNER] == 0)                     {                         CreateNoOwnerHouseTD(playerid);                         for(new i = 0; i < sizeof NoOwnerTextDraw_PTD[]; i++)                         {                             PlayerTextDrawShow(playerid, NoOwnerTextDraw_PTD[playerid][i]);                                 }                                 new price[8];                                 format(price, sizeof(price), "%d$", House[index][hPRICE]);                                 PlayerTextDrawSetString(playerid, NoOwnerTextDraw_PTD[playerid][2], price);                                 new id[9];                                 format(id, sizeof(id), "%d", House[index][hID]);                                 PlayerTextDrawSetString(playerid, NoOwnerTextDraw_PTD[playerid][0], id);                                 switch(House[index][hCLASS])                                 {                                     case 1: PlayerTextDrawSetString(playerid, NoOwnerTextDraw_PTD[playerid][1], "A");                                     case 2: PlayerTextDrawSetString(playerid, NoOwnerTextDraw_PTD[playerid][1], "B");                                     case 3: PlayerTextDrawSetString(playerid, NoOwnerTextDraw_PTD[playerid][1], "C");                                 }                                 for(new i = 0; i < sizeof NoOwnerTextDraw_TD; i++)                         {                             TextDrawShowForPlayer(playerid, NoOwnerTextDraw_TD[i]);                                 }                         }                 }                 case GA_RESPONSE_LEAVE:                 {                 for(new i = 0; i < sizeof NoOwnerTextDraw_PTD[]; i++)                 {                     PlayerTextDrawHide(playerid, NoOwnerTextDraw_PTD[playerid][i]);                         }                         for(new i = 0; i < sizeof NoOwnerTextDraw_TD; i++)                 {                     TextDrawHideForPlayer(playerid, NoOwnerTextDraw_TD[i]);                         }                 }         }         return 1; } GAResponse:BizArea(playerid, response, key, index) {         switch(response)         {             case GA_RESPONSE_PRESS_KEY:             {                 if(key & KEY_WALK)                 {                     GoToInteriorBiz(playerid, Biz[index][bINTERIOR]);                                 SetPlayerVirtualWorld(playerid, index);                 }                 }         }         return 1; } GAResponse:InteriorExit(playerid, response, key, index) {         switch(response)         {             case GA_RESPONSE_PRESS_KEY:             {                 new world = GetPlayerVirtualWorld(playerid);                 if(key & KEY_WALK)                 {                     if(world < MAX_HOUSES)                     {                         SetPlayerInterior(playerid, 0);                             SetPlayerVirtualWorld(playerid, 0);                             SetPlayerPos(playerid, House[world][hX], House[world][hY], House[world][hZ]);                             SetPlayerFacingAngle(playerid, House[world][hA]);                                 }                                 else if(world < MAX_HOUSES + MAX_BIZS)                     {                         world -= MAX_HOUSES;                                         printf("%d index world", world);                         SetPlayerInterior(playerid, 0);                             SetPlayerVirtualWorld(playerid, 0);                             SetPlayerPos(playerid, Biz[world][bX], Biz[world][bY], Biz[world][bZ]);                             SetPlayerFacingAngle(playerid, Biz[world][bA]);                                 }                         }                         if(key & KEY_CTRL_BACK)                         {                             if(player_info[playerid][HOUSE] != House[world][hID]) return SCM(playerid, COLOR_RED, "[Ошибка]{ffffff}Вы не владелец данного дома!");                             new dialog[256];                                 format(dialog, sizeof(dialog),                                         "{ffd900}[1]{ffffff}Информация о доме\n\                                         {ffd900}[2]{ffffff}%s дом",                                 (House[world][hCONDITION] == 1) ? ("{00ff00}Открыть") : ("{ff0000}Закрыть"));                             SPD(playerid, DLG_HMENU, DIALOG_STYLE_LIST, "{ffd900}Меню дома", dialog, "Выбрать", "Закрыть");                             return 1;                         }                 }         }         return 1; } stock GoToInterior(playerid, interior) {     for(new i = 0; i < Interiors; i++)         {             if(Interior[i][intID] != interior) continue;             SetPlayerInterior(playerid, Interior[i][intINTERIOR]);             SetPlayerPos(playerid, Interior[i][intX], Interior[i][intY], Interior[i][intZ]);             SetPlayerFacingAngle(playerid, Interior[i][intA]);             SetPlayerCheckpoint(playerid, Interior[i][intX], Interior[i][intY], Interior[i][intZ], 1.0);             new str[128];             format(str, sizeof(str),                         "Нажмите \"ALT\" для выхода\n\                         Нажмите \"H\", чтобы открыть меню дома");             Create3DTextLabel(str, -1, Interior[i][intX], Interior[i][intY], Interior[i][intZ], 15.0, 0, 1);             return 1;         }         return 0; } stock GoToInteriorBiz(playerid, interior) {     for(new i = 0; i < Interiors; i++)         {             if(Interior[i][intID] != interior) continue;             SetPlayerInterior(playerid, Interior[i][intINTERIOR]);             SetPlayerPos(playerid, Interior[i][intX], Interior[i][intY], Interior[i][intZ]);             SetPlayerFacingAngle(playerid, Interior[i][intA]);             SetPlayerCheckpoint(playerid, Interior[i][intX], Interior[i][intY], Interior[i][intZ], 1.0);             new str[128];             format(str, sizeof(str),                         "Нажмите \"ALT\" для выхода");             Create3DTextLabel(str, -1, Interior[i][intX], Interior[i][intY], Interior[i][intZ], 15.0, 0, 1);             return 1;         }         return 0; }  
       
      Что мне делать, подскажите пожалуйста?
    • Jdjdjd
      От Jdjdjd
      Как сделать открытие закрытие ворот в Радмир рп на копии,и куд а нужно вставлять код?