• Уважаемые гости и новички, приветствуем Вас на нашем форуме
    Здесь вы можете найти ответы практически на все свои вопросы о серии игр «Готика» (в том числе различных модах на нее), «Ведьмак», «Ризен», «Древние свитки», «Эра дракона» и о многих других играх. Можете также узнать свежие новости о разработке новых проектов, восхититься творчеством наших форумчан, либо самим показать, что вы умеете. Ну и наконец, можете обсудить общие увлечения или просто весело пообщаться с посетителями «Таверны».

    Чтобы получить возможность писать на форуме, оставьте сообщение в этой теме.
    Удачи!
  • Друзья, доброго времени суток!
    Стартовал новый литературный конкурс от "Ордена Хранителей" - "Пираты Миртанского моря".
    Каждый может принять в нём участие и снискать славу и уважение, а в случае занятия призового места ещё и получить награду. Дерзайте

Вопросы по union

Jr13San


Модостроитель
Регистрация
1 Апр 2010
Сообщения
473
Благодарности
307
Баллы
230
Выходит что pWorld->Release() освобождает память?
Да, внутри этой функции заключён механизм удаления объекта.
Ну т.е. когда создаётся новый объект, он имеет лишь одну ссылку (zCObject::refCtr = 1). Если при этом вызвать "Release()", то объект сразу же удалится.
Почему? Потому что Release() - освобождает одну ссылку на объект, а там как раз была одна, тогда остаётся ноль. А отсутствие ссылок на объект (zCObject::refCtr = 0), говорит системе о том, что данный экземпляр больше никому не нужен и его можно удалить, чтобы освободить память для других объектов или переменных.

Также иногда бывают ситуации, когда объект создаётся в одной функции и "передаётся во владение" другой функции, которая периодически обращается к этому объекту.
Либо этот объект может быть "облачным объектом", без каких-то опознавательных знаков и с ним постоянно работают из разных мест. Причём из каждого места работы с объектом, "сообщили системе", что он ещё нужен, с помощью функции "zCObject::AddRef()".
Функция AddRef() - увеличивает число ссылок на объект на одну. Или по-другому, резервирует ещё одну доп. линию для обращения к объекту.

В случае, если в каком-то месте работы с объектом, он больше не нужен, вызывается Release(). И если ещё остаются ссылки на этот объект, он продолжает существовать в памяти, если же ссылок больше нет, то он удаляется.

Нужно ещё отметить, что после отказа работать с объектом, нужно очищать указатель на него, чтобы больше в ту область памяти не обращаться. Потому что к тому времени там может быть уже совершенно другой объект или абстрактный набор байт.
Для этого есть специальный макрос: "zRELEASE(obj)". Он как раз освобождает одну ссылку на объект и обнуляет текущий указатель на него.
Пример работы системы создания и удаления объекта:
C++:
void Game_Loop()
{
    // объявляем пустой указатель на объект
    // (можно вынести из функции, но уже без приставки статик)
    static zCWorld* pWorld = NULL;

    // При нажатии клавиши "NUMPAD4"
    if (zKeyToggled(KEY_NUMPAD4))
    {
        cmd << "NUM 4" << endl;
        // если указатель пуст
        if (!pWorld)
        {
            // создаём новый мир
            pWorld = new zCWorld();

            // если создан
            if (pWorld)
            {
                // выводим сообщение
                cmd << "Create new zCWorld object (ptr = " << (int)pWorld << ", refs = " << pWorld->refCtr << ")" << endl;

                // задаём название миру
                pWorld->m_strlevelName = "NEW_WORLD_01";
            }
        }
    }

    // При нажатии клавиши "NUMPAD5"
    if (zKeyToggled(KEY_NUMPAD5))
    {
        cmd << "NUM 5" << endl;

        // если мир создан
        if (pWorld)
        {
            // выводим сообщение
            cmd << "Call Release()" << endl;

            // освобождаем его
            pWorld->Release();

            // и обнуляем наш указатель
            pWorld = NULL;

            // или всё тоже самое делает:
            // zRELEASE(pWorld);
        }
    }
}

// Перехват деструктора объекта класса zCWorld.
// 0x006200F0 protected: virtual __thiscall zCWorld::~zCWorld(void)
static void __fastcall zCWorld_Destructor(zCWorld* _this);
static CInvoke <void(__thiscall*)(zCWorld*)> pzCWorld_Destructor(0x006200F0, zCWorld_Destructor, IVK_AUTO);
static void __fastcall zCWorld_Destructor(zCWorld* _this)
{
   // сообщение для отладки
    cmd << "~zCWorld (LevelName = '" << _this->m_strlevelName << "', ptr = " << (int)_this << ")" << endl;
    pzCWorld_Destructor(_this);
}

Create_zCWorld_Test.jpg


Как видим, при нажатии NumPad4, создаётся новый объект, который имеет свой указатель ptr и число ссылок = 1.
Потом, при нажатии NumPad5, вызывается Release(), она уменьшает число ссылок до 0, затем объект удаляется.
Но перед удалением, вызывается функция деструктора объекта: ~zCWorld(). В ней мы и видим наш недавно созданный объект, судя по названию и указателю на него.
Всё. Дальше можешь сам поэкспериментировать с AddRef() и Release().

Не совсем понял, для чего служит распаковка, ране я прописал только через player->inventory2.GetItem(inumItem).
Механизм упаковки вещей в текстовую строку, используется, например, во время сохранения инвентарей.
Если система активна, то содержимое инвентаря сохраняется в виде зашифрованной строки, в которой, для каждого предмета, пишется название инстанции и кол-во предметов данной инстанции. После загрузки сейва, некоторые вещи так и остаются в запакованном виде.
Если система неактивна, то вещи в сейв пишутся как объекты.
По умолчанию, эта система активна, и за это отвечает свойство объектов: BOOL oCNpcInventory:: PackAbility.
При первой же необходимости, вещи из строки обратно распаковываются в объекты класса oCItem.

Возможно этот механизм был введён для экономии расхода памяти, чтобы не держать постоянно в памяти все свойства этих вещей.

Некоторые функции движка, учитывают не только содержимое инвентаря, но и вещи в зашифрованной строке.
Но если делать перебор вещей вручную, то можно не досчитаться некоторых экземпляров.

Можешь посмотреть на механизм в действии с помощью этого тестового кода:
C++:
void GetNumItems(oCNpc* pNpc, int& num, zSTRING& packItems)
{
    num = 0;
    packItems = "";

    if (!pNpc)
        return;
 
    zCListSort<oCItem>* node = pNpc->inventory2.contents->GetNextInList();
    while (node)
    {
        oCItem* item = node->GetData();
        node = node->GetNextInList();
        if (item)
        {
            packItems += (item->GetInstanceName() + Z ":" + Z item->amount + " ");
            num++;
        }
    }
}

void Game_Loop()
{
    if (player && screen)
    {
        int x = 100;
        int y = 100;
        int ystep = 200;
        int numItems;
        zSTRING packStr;

        // число вещей, упакованных в строку
        int numPacked = player->inventory2.GetNumItemsInPackString();

        // способность упаковки вещей (1 - вкл, 0 - выкл)
        screen->Print(x, y, Z "player:packAbility = " + Z player->inventory2.packAbility);
        y += ystep;

        // упакованные вещи игрока (название инстанций и кол-во предметов)
        screen->Print(x, y, Z "player:packString( " + Z numPacked + Z " ) = " + Z player->inventory2.packString);
        y += ystep;

        GetNumItems(player, numItems, packStr);
        screen->Print(x, y, Z "player:numItems(" + Z numItems + Z ") = " + packStr);
        y += ystep;



        // указатель на НПС в фокусе ГГ
        oCNpc* pFocusNpc = player->GetFocusNpc();
        if (pFocusNpc)
        {
            zSTRING npcName = pFocusNpc->GetName(0);
            screen->Print(x, y, npcName + Z ":packAbility = " + Z pFocusNpc->inventory2.packAbility);
            y += ystep;

            screen->Print(x, y, npcName + Z ":packString( " + Z numPacked + Z " ) = " + Z pFocusNpc->inventory2.packString);
            y += ystep;

            GetNumItems(pFocusNpc, numItems, packStr);
            screen->Print(x, y, npcName + Z ":numItems(" + Z numItems + Z ") = " + packStr);
            // y += ystep;


            // упаковать все вещи НПС в фокусе
            if (zKeyToggled(KEY_NUMPAD2))
            {
                cmd << npcName << " -> Pack all items" << endl;
                pFocusNpc->inventory2.packAbility = TRUE;
                BOOL packEquipped = FALSE;
                pFocusNpc->inventory2.PackAllItems(packEquipped);
            }
            // распаковать все вещи НПС в фокусе
            if (zKeyToggled(KEY_NUMPAD3))
            {
                cmd << npcName << " -> Unpack all items" << endl;
                pFocusNpc->inventory2.UnpackAllItems();
            }
        }
   
        // упаковать все вещи ГГ
        if (zKeyToggled(KEY_NUMPAD0))
        {
            cmd << "Player -> Pack all items" << endl;
            player->inventory2.packAbility = TRUE;
            BOOL packEquipped = FALSE;
            player->inventory2.PackAllItems(packEquipped);
        }
        // распаковать все вещи ГГ
        if (zKeyToggled(KEY_NUMPAD1))
        {
            cmd << "Player -> Unpack all items" << endl;
            player->inventory2.UnpackAllItems();
        }
    }
}

Например, загружаю ранее сделанный сейв, и вижу, что часть предметов находится в инвентаре (9), а часть осталась в зашифрованном виде(21):
TestPack1.jpg


При открытии инвентаря вижу следующее:
TestPack2.jpg

При этом, все вещи из запакованной строки, превратились в реальные предметы, и у ГГ стало 30 вещей.
Как видим, всё происходит по первому же требованию.

У НПС после загрузки тоже такое бывает.
Например, здесь мы видим, что у него 2 вещи (топор и фартук кузнеца) находятся в инвентаре, а остальные вещи только в виде строки:
TestPack3.jpg


здесь мы видим, что вещей в строке нет, т.к. они уже отправились в инвентарь:
TestPack4.jpg


открываем инвентарь, видим шесть вещей и один фартук (не отображается, т.к. экипирован):
TestPack5.jpg


ну или вот, 4 вещи и фартук:
TestPack6.jpg


В общем, можешь поэкспериментировать с помощью нажатия различных NumPad клавиш, перед этим заглядывая в код, чтобы понимать что должно произойти.

Ну и в целом, не рекомендую создавать постоянно новый мир и удалять его после рендера каждого предмета. Это ненужная операция. Лучше его создать перед рендером всех предметов, а по окончанию работы функции - удалить. Либо создать единожды, на всю игровую сессию, как глобальный объект, и никогда не удалять.

И ещё одна заметка. Нужно понимать, что экран можно закрасить всего лишь одной вьюшкой, постоянно меняя её координаты и вызывая функцию рендера Blit().
Да, и последнее, что хотел сказать. Если твоя система будет нормально работать, выйдя за рамки теста, то подумай над альтернативным управлением, кроме мышки.
Чтобы люди, использующие джойстик, не были в пролёте.
 
Последнее редактирование:

DAMROCK

Участник форума
Регистрация
23 Янв 2017
Сообщения
30
Благодарности
7
Баллы
160
Возник вопрос по поводу коллизии, необходимо отследить столкновение zCVob с моделью мира. Потом, в месте столкновения проследить какой материал с текстурой наложен на полигон.
 

DAMROCK

Участник форума
Регистрация
23 Янв 2017
Сообщения
30
Благодарности
7
Баллы
160
С учётом воды или без?
И в целом, что хочешь сделать?
Желательно с учётом воды. На данный момент работаю с камерой, надо просчитать столкновение, чтобы она не вылазила в пустое пространство. Наверное, это как-то делается с помощью AI методов камеры, но там я ничего не понимаю. А про материал- это на будущее, так как рассматриваю возможность сделать повреждение состояний оружия в зависимости от того о какой материал будет совершён удар. Поэтому, тут одно к другому, всё-равно придётся искать доступ к конкретным полигонам level-mesh.
 

Jr13San


Модостроитель
Регистрация
1 Апр 2010
Сообщения
473
Благодарности
307
Баллы
230
На данный момент работаю с камерой, надо просчитать столкновение, чтобы она не вылазила в пустое пространство.
А что, с ней есть какие-то постоянные проблемы, которые можно решить на раз-два? Есть какая-нибудь запись или скриншот проблемы?

рассматриваю возможность сделать повреждение состояний оружия
А где ты собираешься сохранять это значение? Нужно учитывать, что оружие может храниться в виде пачки, а пачка - это один экземпляр oCItem, с определённым кол-вом штук. И как понять какая из штук на сколько изношена?
Также нужно учесть, что вещи могут путешествовать по слотам, инвентарям, сундукам и по миру.

А так, проверка на пересечение оружия с поверхностью делается с помощью трассировки лучей.
Т.е. вычисляется вектор длины, исходящий от рукояти и затем этот вектор трассируется в момент проигрывания анимации удара.
У меня какие-то наработки по этой теме есть, но там решение не в две строчки кода.
Желательно с учётом воды.
А вода то тебе зачем?
 

DAMROCK

Участник форума
Регистрация
23 Янв 2017
Сообщения
30
Благодарности
7
Баллы
160
А что, с ней есть какие-то постоянные проблемы, которые можно решить на раз-два? Есть какая-нибудь запись или скриншот проблемы?
По сути это даже не проблема, просто я перепрограммирую камеру для свободного вращения в режиме интерактивной мыши в новом инвентаре. Так как приходится слипать камеру для работы с ней, айшник камеры вырубается, мне необходимо менять положение камеры через slerp + изменение ориентации так-же. Без учета столкновений, камера игнорирует level-mesh, и проходит сквозь стены.
изображение_2025-01-06_171727787.png
Не знаю понятно будет или нет, но вот здесь камера в стену залезла, как пример.
А где ты собираешься сохранять это значение? Нужно учитывать, что оружие может храниться в виде пачки, а пачка - это один экземпляр oCItem, с определённым кол-вом штук. И как понять какая из штук на сколько изношена?
Также нужно учесть, что вещи могут путешествовать по слотам, инвентарям, сундукам и по миру.
Я это делал через обновление класса oCItem в oCItemEx, При инициализации определённых предметов я задаю личный id каждого предмета, который отсутствует в оригинале, после первого сохранения игры все эти предметы жёстко идентифицированы. Пачки проработаю в механике контейнера, думаю, с этим серьёзных проблем возникнуть не должно, ну там будет видно. Чтобы ответить на этот вопрос конкретно, надо за это взяться, а сейчас только на бумаге.
А так, проверка на пересечение оружия с поверхностью делается с помощью трассировки лучей.
Т.е. вычисляется вектор длины, исходящий от рукояти и затем этот вектор трассируется в момент проигрывания анимации удара.
У меня какие-то наработки по этой теме есть, но там решение не в две строчки кода.
Да мне хотя-бы в общих чертах, что уж-там, главное понять что/от куда вызывать и как оно называется.
А вода то тебе зачем?
честно, не знаю :oops:пока-что...) Но в целом, это может быть важно при работе с коллизией, пока-что не тестировал этот момент.
 
Последнее редактирование:

Slavemaster


Модостроитель
Регистрация
10 Июн 2019
Сообщения
1.096
Благодарности
1.928
Баллы
320
я перепрограммирую камеру для свободного вращения в режиме интерактивной мыши в новом инвентаре
Хукни и перепиши zCAICamera::CheckKeys
Меняй в нём поля bestRotX/Y/Z и вызывай moveTracker->SetHintMouseUsed

Удары оружием об меш локации детектируются в oCAniCtrl_Human::CheckMeleeWeaponHitsLevel и звук воспроизводится в зависимости от материала.
 

DAMROCK

Участник форума
Регистрация
23 Янв 2017
Сообщения
30
Благодарности
7
Баллы
160
Меняй в нём поля bestRotX/Y/Z и вызывай moveTracker->SetHintMouseUsed
не совсем понятно что именно делают эти переменные и метод.
Удары оружием об меш локации детектируются в oCAniCtrl_Human::CheckMeleeWeaponHitsLevel и звук воспроизводится в зависимости от материала.
так-же, что возвращает CheckMeleeWeaponHitsLevel? как вообще в таких методах понять что они возвращают?
 

Slavemaster


Модостроитель
Регистрация
10 Июн 2019
Сообщения
1.096
Благодарности
1.928
Баллы
320
так-же, что возвращает CheckMeleeWeaponHitsLevel?
Возвращает true, если был зафиксирован удар об меш

как вообще в таких методах понять что они возвращают?
Подучить английский и/или стащить где-нибудь исходники готы:confused:
 

Slavemaster


Модостроитель
Регистрация
10 Июн 2019
Сообщения
1.096
Благодарности
1.928
Баллы
320
не совсем понятно что именно делают эти переменные и метод.
Простейший пример:

C++:
        if (GetMode() != "CAMMODINVENTORY")
            return THISCALL(Ivk_zCAICamera_CheckKeys)();

        const float mouseMultiplier = moveTracker->frameTime * 5.0f;

        float mx, my, mz;
        zinput->GetMousePos(mx, my, mz);

        bestRotX += my * mouseMultiplier;
        bestRotY += mx * mouseMultiplier;

        moveTracker->bMouseUsed = mx != 0.0f || my != 0.0f;
 

DAMROCK

Участник форума
Регистрация
23 Янв 2017
Сообщения
30
Благодарности
7
Баллы
160
В целом, нашёл то что искал, плюс-минус. Несмотря на то, что указан флаг до первого столкновения, почему-то, иногда, всё-равно указывает на другой полигон, лежащий дальше тестируемого. Случайно, наткнулся на код Gratt`a по zSoftskinCollisions, на его основе сделал это.
C++:
    /* Проверяет на столкновение*/
    bool  ICamCon::CollisionTo_WorldPolyMesh()
    {
        //std::cout << "CollisionTo_WorldPolyMesh\n";

        oCWorld* world = ogame->GetGameWorld();     // Получаем мир
      
        zVEC3 vAt  = player->GetAtVectorWorld();    // Получаем направление указателя
        zVEC3 vPos = player->GetPositionWorld();    // Получаем начало луча
        zVEC3 vTo  = vAt * 300.0f;                    // Получаем направление

      
        int ok = world->TraceRayNearestHit(vPos, vTo, player, zTRACERAY_FIRSTHIT); // Трассировка до первого столкновения
        if (ok)
        {
            zlineCache->Line3D(world->traceRayReport.foundPoly->vertex[0]->position, world->traceRayReport.foundPoly->vertex[1]->position, GFX_GREEN, 0);    /* Выделяем полигон столкновения */
            zlineCache->Line3D(world->traceRayReport.foundPoly->vertex[1]->position, world->traceRayReport.foundPoly->vertex[2]->position, GFX_GREEN, 0);    /* Выделяем полигон столкновения */
            zlineCache->Line3D(world->traceRayReport.foundPoly->vertex[2]->position, world->traceRayReport.foundPoly->vertex[0]->position, GFX_GREEN, 0);    /* Выделяем полигон столкновения */
            zlineCache->Line3D(player->GetPositionWorld(), vTo + vPos, GFX_RED, 0);        // Направление трассировки
            return TRUE;
        }

        zlineCache->Line3D(player->GetPositionWorld(), vTo + vPos, GFX_RED, 0);        // Направление трассировки
        return  FALSE;
    }
TR1.png
TR2.png
 
Последнее редактирование:

MEG@VOLT

★★★★★★★★★
ТехАдмин
Регистрация
24 Мар 2006
Сообщения
9.963
Благодарности
6.813
Баллы
1.775
извини за вмешательство в твой соурс код,
Все норм. Спасибо! Мои коды вечно нуждаются в правке))
Тогда такой вопрос:
Чем это:
C++:
oCItem* item;
item = (oCItem*)par->GetInstance();
Отличается от этого:
C++:
oCItem* item = dynamic_cast<oCItem*>((zCVob*)par->GetInstance());
?
По сути ничем?
Твой код не проверял, но с моим кодом, когда он выглядел вот так:
C++:
    int EquipMeleeWeapon() {
        zCParser* par;
        par = zCParser::GetParser();
        oCItem* item;
        item = (oCItem*)par->GetInstance();
        oCNpc* npc;
        npc = (oCNpc*)par->GetInstance();
        oCItem* emw;
        emw = npc->GetEquippedMeleeWeapon();
        if (emw) {
            npc->UnequipItem(emw);
        }
        if (item->mainflag != ITM_CAT_NF) { return false; }
        npc->EquipWeapon(item);
        return true;
    }
Вроде все хорошо...
И когда я вызывал эту функцию из скриптов все было нормально до одного момента:
Когда в инвентаре было восемь и больше одинаковых мечей, при вызове этой функции вылетала игра. Методом тыка обнаружил что вылет происходит в момент item = (oCItem*)par->GetInstance();. И дальше функция не выполнялась.
По этому на каждое новое объявление я налепил проверку))
 

Jr13San


Модостроитель
Регистрация
1 Апр 2010
Сообщения
473
Благодарности
307
Баллы
230
Мега, если в нескольких предложениях, то "dynamic_cast" в данном случае, выступает в качестве защитника от невнимательных действий со стороны разработчика или каких-нибудь "ломателей скриптов". Ну, или же используется, когда результат заранее неизвестен. Например, когда в функцию передаётся объект в фокусе НПС. А что это за объект? Вот, как раз и может определить динамичное преобразование (но с некоторыми оговорками, потому что мы работаем в пространстве плагина, а не в самом движке). В общем, дайнамик изо всех сил пытается преобразовать в рамках структуры классов, а не на прямую(число -> в число).
Также есть ещё один, но слегка усовершенствованный вариант для преобразования (см. шаблон "zDYNAMIC_CAST"). Но его нужно тестить, потому что иногда он чудит.

Давай протестируем касты на таком примере
C++:
// func void DynCastTestFunc(C_ITEM itm, C_NPC npc)
int DynCastTestFunc()
{
    zCParser* par = zCParser::GetParser();

    // принимаем параметры функции (только в обратном порядке)
    void* ptr_Npc = par->GetInstance();
    void* ptr_Item = par->GetInstance();
    // выводим в консоль указатели входящих параметров
    cmd << string::Combine("ptr_Npc(in) = %u, ptr_Item(in) = %u", (uint32)ptr_Npc << (uint32)ptr_Item) << endl << endl;

    // пытаемся преобразовать указатели через "напрямую" (или статик каст)
    oCNpc* pNpc = (oCNpc*)ptr_Npc;
    oCItem* pItem = (oCItem*)ptr_Item;
    // выводим в консоль результат преобразования
    cmd << string::Combine("static_cast:    pNpc = %u,    pItem = %u", (uint32)pNpc, (uint32)pItem) << endl;

    // также пытаемся преобразовать указатели через "дайнамик каст"
    pNpc = dynamic_cast<oCNpc*>((zCVob*)ptr_Npc);
    pItem = dynamic_cast<oCItem*>((zCVob*)ptr_Item);
    // выводим результат
    cmd << string::Combine("dynamic_cast:    pNpc = %u,    pItem = %u", (uint32)pNpc, (uint32)pItem) << endl << endl << endl;

    return 0;
}

void Game_DefineExternals()
{
    // регистрируем функцию в основных скриптах
    parser->DefineExternal("DynCastTestFunc", DynCastTestFunc, zPAR_TYPE_VOID, zPAR_TYPE_INSTANCE, zPAR_TYPE_INSTANCE, 0);
}

В скриптах:
Daedalus:
func event GameLoop()
{
    // при нажатии на клавишу 'Z'
    if (Hlp_KeyToggled(KEY_Z))
    {
        var C_NPC npc;
        var C_ITEM itm;
       
        // получаем ссылку на ГГ
        npc = Hlp_GetNpc(PC_Hero);
       
        // и его экипированное оружие
        itm = Npc_GetEquippedMeleeWeapon(npc);
       
        // выполняем тесты:
        // DynCastTestFunc(C_ITEM itm, C_NPC npc)
        Hlp_PrintConsole("Test1(normal):");
        DynCastTestFunc(itm, npc);
       
        Hlp_PrintConsole("Test2(item both):");
        DynCastTestFunc(itm, itm);
       
        Hlp_PrintConsole("Test3(npc both):");
        DynCastTestFunc(npc, npc);
       
        Hlp_PrintConsole("Test4(reverse):");
        DynCastTestFunc(npc, itm);
    };
};

где:
normal - идеальные условия передачи параметров,
item both - передача в оба параметра ссылок на предмет
npc both - передача в оба параметра ссылок на НПС
reverse - перепутывание местами двух аргументов

ptr_Npc(in)/ptr_Item(in) - входящие параметры в cpp-функцию.

Результат с экипированным оружием:
WeaponIsEquipped.jpg

Что видим. Указатели на входе есть.
1. В идеальных условиях (normal) - все указатели успешно преобразуются в обоих кастах.
2. Если пришло две ссылки на предмет, то у статик каста: и предмет и НПС - это одно и то же (pNpc = pItem).
Дайнамик же, сначала подумал, и получил ссылку только на предмет, НПС - оставил пустым.
3. Если пришло две ссылки на НПС, то у статика, как обычно: обе преобразованные ссылки одновременно являются и НПС и предметом.
Дайнамик же, решил преобразовать только ссылку на НПС, предмет - оставил пустым.
4. Если произошёл реверс при передаче параметров. Т.е. вместо НПС - подставили ссылку на предмет, а вместо предмета - ссылку на НПС.
У статик всё хорошо - просто преобразовал как есть (число -> в число) и не важно кто есть что, а что есть кто.
Дайнамик сказал - ну его нафиг и обнулил оба указателя, что в принципе правильно.


Без оружия на поясе:
WeaponIsNotEquipped.jpg

Ничего интересного.
Но опять же, если результаты твоих тестов тебя полностью устраивают, и есть уверенность, что в функцию никогда не попадёт ничего лишнего, можешь оставить как есть.
Но всякое бывает... ;)

Твой код не проверял, но с моим кодом, когда он выглядел вот так
Проследи за моментом передачи в функцию предмета, не имеющего флаг "ITM_CAT_NF". В твоём варианте произойдёт просто деэкипировка оружия, с последующим выходом. В мной-редактированном случае, выход из функции произойдёт раньше и "холостой деэкипировки" не будет.
А так, решай сам что лучше.
 

MEG@VOLT

★★★★★★★★★
ТехАдмин
Регистрация
24 Мар 2006
Сообщения
9.963
Благодарности
6.813
Баллы
1.775
Лекция прослушана, усвоена. Спасибо!
Кефебрейк*coffee* ))
 
Сверху Снизу