Sign in to follow this  
Followers 0
KlasterK

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

3 posts in this topic

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

 

В первую очередь, нужно добавить 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/...
}


 

 

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

Share this post


Link to post
Share on other sites

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

Спойлер

#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);
}

 

 

Share this post


Link to post
Share on other sites
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

 

Share this post


Link to post
Share on other sites

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!


Register a new account

Sign in

Already have an account? Sign in here.


Sign In Now
Sign in to follow this  
Followers 0

  • Recently Browsing   0 members

    No registered users viewing this page.

  • Similar Content

    • Wiggins
      By Wiggins
      Доброго времени суток, нуждаюсь в помощи. Если есть люди, которые шарят в pawno, нужна помощь. Короче, мод ART RP, там в донате есть строка "Уникальные аксессуары", когда на него нажмешь, открывается TextDraw менью, нужно переделать это на обычную менью (/mm например, там обычная менью), буду рад, если поможете, за ранее спасибо.

      Если не ошибаюсь, строка текстдрав аксессуаров начинается с 1416 до 1443.
      art4.pwn
    • Danny_Marcelo
      By Danny_Marcelo
      Приветствую всех. Сегодня я хочу поделиться системой создания объектов как на Advance RP. Пока нигде не встречал. Да, есть аналоги, порой весьма неплохие. Особенно скрипт SOE. Но тут именно конкретная копия проекта Advance RP. Администрация Advance RP старается максимально скрывать админ информацию главной админки, но благодаря ютуберам какая то информация всё таки просочилась в общий доступ. Благодаря этому мне и удалось воссоздать почти на 100% копию данной системы. Для начала распишу суть. К никам Andrey Ringo, Smart_Jakson и прочим спец.админам привязана команда /lego. Она даёт доступ к другим командам, которые используются для создания и редактирования объектов. Как я понял объекты находятся непосредственно в базе данных MariaDB, а не на сервере, создатель боится что если всем подряд давать команду, те могут положить сервер (случайно), как например ютубер MalFoy чуть этого не сделал:

      Ну в основном она привязана на каждом сервере индивидуально. К примеру на грине по мимо Ринго и Смарта еще и к главному администратору. На Блу аналогично.
      Ну что же, меньше слов, больше дела. Приступим:
      Перейдём к нашим define:
      //========================= [ Привязка к нику ] ================================ #define FullAccessOne                    !strcmp(PlayerInfo[playerid][pName], "Andrey_Ringo", false) #define FullAccessTwo                   !strcmp(PlayerInfo[playerid][pName], "Smart_Jakson", false) #define FullAccessThree                  !strcmp(PlayerInfo[playerid][pName], "Danny_Marcelo", false) #define MAX_LEGO_OBJECTS                 100 // Максимальное количество объектов К нашим переменным:
      //========================  [ LEGO ] =========================================== new LegoObjects[MAX_PLAYERS][MAX_LEGO_OBJECTS]; // Массив для хранения объектов new LegoObjectCount[MAX_PLAYERS]; // Счетчик объектов для каждого игрока new bool:LegoMode[MAX_PLAYERS]; // Режим LEGO new Float:LegoObjectOriginalPos[MAX_PLAYERS][MAX_LEGO_OBJECTS][6]; // [playerid][object_index][x, y, z, rotX, rotY, rotZ] new Float:ServerObjectOriginalPos[MAX_OBJECTS][6]; // [objectid][x, y, z, rotX, rotY, rotZ] new bool:ServerObjectMoved[MAX_OBJECTS]; // Флаг, указывающий, был ли объект перемещён enum {     EDIT_TYPE_NONE,     EDIT_TYPE_GLOBAL,     EDIT_TYPE_PLAYER }; new EditingObjectType[MAX_PLAYERS]; // Тип редактируемого объекта (серверный или игровой) new EditingObjectID[MAX_PLAYERS] = { -1, ... }; // ID редактируемого объекта // ============================================================================= В public OnPlayerConnect(playerid)
      LegoMode[playerid] = false;     LegoObjectCount[playerid] = 0;     EditingObjectID[playerid] = -1;     for(new i = 0; i < MAX_LEGO_OBJECTS; i++)     {         LegoObjects[playerid][i] = INVALID_OBJECT_ID;     } В public OnPlayerDisconnect(playerid, reason)
      for(new i = 0; i < LegoObjectCount[playerid]; i++)     {         if(IsValidObject(LegoObjects[playerid][i]))         {             DestroyObject(LegoObjects[playerid][i]);         }     }     LegoObjectCount[playerid] = 0;     EditingObjectID[playerid] = -1;     DestroyPlayerLogo(playerid); В OnDialogResponse:
      case 272: // Диалог подтверждения для серверных объектов, не забудьте заменить на свой         {             if(response) // Если игрок нажал "Да"             {                 EditObject(playerid, EditingObjectID[playerid]);                 SendClientMessage(playerid, 0xFFFF00AA, "[LEGO] Выберите объект для редактирования");             }         }
      После стандартных public
      public OnPlayerSelectObject(playerid, type, objectid, modelid, Float:fX, Float:fY, Float:fZ) {     if(type == SELECT_OBJECT_GLOBAL_OBJECT) // Если выбран серверный объект     {         ShowPlayerDialog(playerid, 272, DIALOG_STYLE_MSGBOX, !"{ED7A19}Предупреждение",         !"{FFFFFF}Это не LEGO объект. Его перемещение нежелательно.\n\         Вы всё-таки хотите передвинуть его?", !"Да", !"Нет"); // Не забудьте ID диалога заменить аналогично OnDialogResponse         EditingObjectID[playerid] = objectid;         EditingObjectType[playerid] = EDIT_TYPE_GLOBAL;     }     else if(type == SELECT_OBJECT_PLAYER_OBJECT) // Если выбран LEGO-объект     {         for(new i = 0; i < LegoObjectCount[playerid]; i++)         {             if(LegoObjects[playerid][i] == objectid)             {                 EditingObjectID[playerid] = i;                 EditingObjectType[playerid] = EDIT_TYPE_PLAYER;                 EditPlayerObject(playerid, LegoObjects[playerid][i]);                 new string[128];                 format(string, sizeof(string), "[LEGO] Редактирование объекта {3E8A33}%d", i + 1);                 SendClientMessage(playerid, 0xFFFF00AA, string);                 return 1;             }         }     }     return 1; } и еще один
      public OnPlayerEditObject(playerid, playerobject, objectid, response, Float:fX, Float:fY, Float:fZ, Float:fRotX, Float:fRotY, Float:fRotZ) {     if(response == EDIT_RESPONSE_FINAL)     {         if(EditingObjectType[playerid] == EDIT_TYPE_GLOBAL) // Если это серверный объект         {             GetObjectPos(objectid, ServerObjectOriginalPos[objectid][0], ServerObjectOriginalPos[objectid][1], ServerObjectOriginalPos[objectid][2]);             GetObjectRot(objectid, ServerObjectOriginalPos[objectid][3], ServerObjectOriginalPos[objectid][4], ServerObjectOriginalPos[objectid][5]);             ServerObjectMoved[objectid] = true;             SetObjectPos(objectid, fX, fY, fZ);             SetObjectRot(objectid, fRotX, fRotY, fRotZ);             new string[110];             format(string, sizeof(string), "Позиция объекта сохранена {E5A938}(%d/100){66AF0B}. Используйте /moveobj для перемещения на исходное место", LegoObjectCount[playerid]);             SendClientMessage(playerid, 0x66AF0BFF, string);         }         else if(EditingObjectType[playerid] == EDIT_TYPE_PLAYER) // Если это LEGO объект         {             SetPlayerObjectPos(playerid, objectid, fX, fY, fZ);             SetPlayerObjectRot(playerid, objectid, fRotX, fRotY, fRotZ);         }     }     return 1; } Ну и наконец перейдем в наши команды. Я для примера буду использовать pawncmd:
      Команда для доступа к остальным командам редактирвоания:
      CMD:lego(playerid, params[]) {     if (!FullAccessOne && !FullAccessTwo && !FullAccessThree) return 1;     {         LegoMode[playerid] = !LegoMode[playerid];         if(LegoMode[playerid])         {             SendClientMessage(playerid, 0xFFFFFFFF, !"Режим {F7CB31}LEGO {FFFFFF}включён");             SendClientMessage(playerid, 0xFFFFFFFF, !"Команды: /newobj /dellast /delall /objedit /repeat");             SendClientMessage(playerid, 0xFF0000FF, !"Внимательно вводите id объекта! Неверный id может вызвать падение сервера!");         }         else         {             SendClientMessage(playerid, 0xFFFFFFFF, !"Режим {F7CB31}LEGO {FFFFFF}выключен");         }     }     return 1; } Команда для создания объектов:
      CMD:newobj(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, !"Включи режим LEGO: {F7CB31}/lego");         SendClientMessage(playerid, 0xE3871EFF, !"Внимательно вводите id объекта! Неверный id может вызвать падение сервера!");         if(LegoObjectCount[playerid] >= MAX_LEGO_OBJECTS) return SendClientMessage(playerid, 0xFF0000FF, "Достигнут лимит объектов");         new objectid;         if(sscanf(params, "d", objectid)) return SendClientMessage(playerid, 0xCECECEFF, !"Используйте: /newobj [id объекта]");         new Float:x, Float:y, Float:z;         GetPlayerPos(playerid, x, y, z);         LegoObjects[playerid][LegoObjectCount[playerid]] = CreatePlayerObject(playerid, objectid, x, y, z, 0.0, 0.0, 0.0, 300.0);         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][0] = x;         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][1] = y;         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][2] = z;         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][3] = 0.0;         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][4] = 0.0;         LegoObjectOriginalPos[playerid][LegoObjectCount[playerid]][5] = 0.0;         LegoObjectCount[playerid]++; // Увеличиваем счетчик объектов         new string[59];         format(string, sizeof(string), "[LEGO] Создан объект. Всего объектов: {35BD39}%d/%d", LegoObjectCount[playerid], MAX_LEGO_OBJECTS);         SendClientMessage(playerid, 0xFFFF00AA, string);     }     return 1; } Команда для редактирования объектов, включая серверных.
      CMD:objedit(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, !"Включи режим LEGO: {F7CB31}/lego");         SendClientMessage(playerid, 0xFFFF00AA, !"[LEGO] Выберите объект для редактирования");         SelectObject(playerid);     }     return 1; } Данная команда, вернёт передвинутые объекты вашего сервера в исходное состояние
      CMD:moveobj(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, !"Включи режим LEGO: {F7CB31}/lego");         new movedObjects = 0;         for(new i = 0; i < MAX_OBJECTS; i++)         {             if(ServerObjectMoved[i])             {                 SetObjectPos(i, ServerObjectOriginalPos[i][0], ServerObjectOriginalPos[i][1], ServerObjectOriginalPos[i][2]);                 SetObjectRot(i, ServerObjectOriginalPos[i][3], ServerObjectOriginalPos[i][4], ServerObjectOriginalPos[i][5]);                 ServerObjectMoved[i] = false;                 movedObjects++;             }         }         new string[92];         format(string, sizeof(string), "Стандартные объекты со смещённой позицией возвращены на свои места {E8AC32}(%d шт.)", movedObjects);         SendClientMessage(playerid, 0x78A82CFF, string);     }     return 1; } Эта команда удаляет последний созданный объект через /newobj
      CMD:dellast(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, "Включи режим LEGO: {F7CB31}/lego");         if(LegoObjectCount[playerid] == 0) return SendClientMessage(playerid, 0xCECECEFF, "Нет объектов для удаления.");         DestroyPlayerObject(playerid, LegoObjects[playerid][LegoObjectCount[playerid] - 1]);         LegoObjectCount[playerid]--;         new string[37];         format(string, sizeof(string), "[LEGO] Удалён объект {F74214}%d/100", LegoObjectCount[playerid]);         SendClientMessage(playerid, 0xFFFF00AA, string);     }     return 1; } Эта команда удалит все созданные вами объекты.
      CMD:delall(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, "Включи режим LEGO: {F7CB31}/lego");         if(LegoObjectCount[playerid] == 0) return SendClientMessage(playerid, 0xCECECEFF, "Нет объектов для удаления.");         for(new i = 0; i < LegoObjectCount[playerid]; i++)         {             DestroyPlayerObject(playerid, LegoObjects[playerid][i]);         }         LegoObjectCount[playerid] = 0;         SendClientMessage(playerid, 0xFFFF00AA, !"[LEGO] Все объекты удалены");     }     return 1; } Ну и эта команда повторит действие /newobj [id игрока]
      CMD:repeat(playerid, params[]) {     if(PlayerInfo[playerid][pAdmin] < 5) return 1;     {         if(!LegoMode[playerid]) return SendClientMessage(playerid, 0xFFFFFFFF, "Включи режим LEGO: {F7CB31}/lego");         if(LegoObjectCount[playerid] == 0) return SendClientMessage(playerid, 0xCECECEFF, "Нет объектов для повторения.");         new Float:x, Float:y, Float:z;         GetPlayerPos(playerid, x, y, z);         new lastObjectID = LegoObjects[playerid][LegoObjectCount[playerid] - 1];         new modelid = GetPlayerObjectModel(playerid, lastObjectID);         LegoObjects[playerid][LegoObjectCount[playerid]] = CreatePlayerObject(playerid, modelid, x, y, z, 0.0, 0.0, 0.0, 300.0);         LegoObjectCount[playerid]++;         new string[39];         format(string, sizeof(string), "[LEGO] Создан объект {35BD39}%d/%d", LegoObjectCount[playerid], MAX_LEGO_OBJECTS);         SendClientMessage(playerid, 0xFFFF00AA, string);     }     return 1; } Переменные
      enum pInfo {     pName[MAX_PLAYER_NAME+1],     pAdmin,  new PlayerInfo[MAX_PLAYERS][pInfo]; заменяйте на свои из вашего мода. Цвета я использовал как коды, чтобы у вас не было ошибок error 017: можете заменить на свои дефайны цвета.
      Система полностью тщательно протестирована, она полностью работает. 
      Скриншоты, как выглядит это всё:
       
       
      P.S. Если редактируете серверные объекты, то сразу имейте ввиду, что createobject и стандартные объекты карты GTA San Andreas не редактируются. А вот уже те, которые через inckude <streamer>, то есть CreateDynamicObject, они редактируются. Будьте осторожны при изменениях и добавлениях объектов. Не вбивайте рандомные числа в /newobj. ID объектов SAMP полно в интернете. 
      Спасибо за внимание!