Помощь - Поиск - Пользователи - Календарь
Полная версия: Азы скриптинга NWN/NWN2
Город Мастеров > РЕДАКТОРЫ > Neverwinter Nights 2 Obsidian Toolset
Страницы: 1, 2
Aiwan
В данной теме разрешается писать только обучающимся. Если кто-то найдет в моих материалах грубую ошибку, то может сообщить мне или Лексу и мы поправим.

Во первых, забудьте что скрипты это трудно непонятно и сложно. Забудьте все то что бедный ваш мозг пытается обработать безуспешно и вы не можете понять простые вещи. Это нормально. Я буду говорить не по книгам а просто, описывая ситуацию или знания необходимые вам максимально понятно и по возможности проектировать это на бытовом уровне, оперируя образными картинками. Так как именно такие схемы будут работать в вашем случае. Ведь люди с математическим умом все схватывают сами и на лету. Итак начнем.

Скрипт сам по себе это маленький файлик. Он работает как любой файл в программе. То что вы пишете в редакторе это NSS просто информация для компилятора, в игре то что написано вами не работает, а работает скомпилированный файл NSC. Поэтому написанное в редакторе и не компельнутое не работает.

Как отличить код хорошего кодера от разгильдяя? По аккуратности и описаниям-комментариям. У тех, кто понимает скрипт все правильно, кто не понимает все кидает в кучу. Поэтому сразу учитесь писать красиво и понятно. Сделайте себе шапку для скриптов и пишите в ней, давая описания. В папке скриптов через пол года вы сможете понять кто и что сделал в скрипте и элементарно для чего вам он нужен. Это важно. Аккуратность.

Пример шапки скрипта:
Neverwinter Script Source
//:://////////////////////////////////////////////
//:: Name:
//:: Copyright © 2006 WRG! Team
//:://////////////////////////////////////////////
/*
    Описание
*/

//:://////////////////////////////////////////////
//:: Created By: Aiwan
//:: Created On: 06.10.2007
//:://////////////////////////////////////////////

void main()
{ }


Дальше.
Aiwan
Все функции в движке, это какие-то действия и примерно такие же куски кода как пишете вы, но они вставлены в движок и работают когда вы ими пользуетесь: создаете предмет, дестроите его, раздаете команды и т.д. Соотвественно, движок это не идеал, а обыкновенная программа и она имеет много неучтенных, ненайденных или незапротоколированных особенностей в исполнении функций. Поэтому вы должны сразу понять, что ТО ЧТО КОМПЕЛИТСЯ НЕ ВСЕГДА РАБОТАЕТ.

В скриптинге есть определенные типы данных с которыми мы будем работать.
Object - объект. Например, НПС, локация и т.д.
Стринг String - то что пишем буквами и движок понимает это как надписи. Например в фразы, имена и т.д.
Интеджер integer - это целое значение. может быть 1, 10, 25 и т.д.
Флоат Float - НЕцелое значение может быть 1.0, 23.04 и т.д.
Локация Location - это позиция, местоположение определенная точка.
Вектор vector - вектор.
Талант talent - я сам не знаю что это prankster2.gif
Эффект effect - кто много курил и ел всякую гадость в виде наркоты в ПЛ тот знает что это.

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

Для начала мы разберем самое простое - присвоене локальной переменной.
Neverwinter Script Source
// Set oObject's local integer variable sVarName to nValue
void SetLocalInt(object oObject, string sVarName, int nValue)


Итак, что это такое? Образно, локальная переменная это строка которая вписывается в служебное место любого объекта и состоит из буквенной строки и числового значения. Видите эта функция состоит из трех частей: Объект, Стринг, Число. Попросту вы метите объект и пишете на нем слово вашей переменной писваивая к нему знацение циферкой. Это вроде как повесить на спину Лекса бумажку и написать на ней ВАНЮЧКА, 1. А у Нео повесить бумажку ВАНЮЧКА, 2 и потом по этим данным работать с этимим объектами. Нужен вам ванючка со значением 2, смотрите на спинах у кого бумажка и делаете что вам надо. Так вот. Вы часто сталкивались с такими словами как TRUE и FALSE. Так вот, что бы вам было понятно вопрос КТО ИЗ АДМИНОВ ВАНЮЧКА будет выглядеть так: Лекс - TRUE, Нео TRUE, Айван - FALSE. Но. Движок НВН считает, что если на Айване нет бумажки с надписью ВАНЮЧКА или она есть, но выглядит так "ВАНЮЧКА, 0" то это FALSE. Попросту, если стереть на Лексе 1 и поставить там 0 то результат будет FALSE хоть и табличка будет висеть. Это надо учитывать в некоторых случаях. В виде кода это будет выглядеть так:

Присвоим:
Neverwinter Script Source
void main()
{
  object oAiwan = GetObjectByTag("Aiwan ROOT Boss");
  object oLex = GetObjectByTag("LEX Administrator");
  object oNeo = GetObjectByTag("NEO Ррразгильдяй");

  SetLocalInt(oLex, "VANUCHKA", 1);
  SetLocalInt(oNeo, "VANUCHKA", 2);
}


Вот так в скрипте диалога проверить, то что я сказал.

Neverwinter Script Source
int StartingConditional()
{
  object oLex = GetObjectByTag("LEX Administrator");
  if (GetLocalInt(oLex, "VANUCHKA") == 1)
  {
      return TRUE;
  }
  return FALSE;
}


Объект может содержать только одно значение переменной. Либо не иметь ее вообще. Скриптом можно менять значение и результаты будут разные. К примеру, в модулях часто одному квесту присваивают одну локальную переменную. А вот от ее значения зависит разные концовки.

Что непонятно спрашивайте.
Ёжик
Кто такие Интеджер и Флоат? Почему oNeo, oLex, oAiwan? Что значит object oAiwan = GetObjectByTag("Aiwan ROOT Boss")? Что такое int StartingConditional()? Зачем скобки? void main - это что?
Aiwan
Ёжик, я могу дать все эти понятия, но то что ты поймешь что это я не уверен. Пока, мне нужно общее ваше представление. Как только ты поймешь что все то что я написал означает свою задачу я сочту выполненой. Но раз такой расклад то могу все объяснить:

Цитата(Ёжик @ Oct 7 2007, 16:03) *
void main - это что?

NWScript использует 'функции', которые сообщают программе, что нужно делать. Некоторые функции уже написаны и нам остается только поставить их в скрипт. Тем не менее, когда мы пишем скрипт, мы, фактически, пишем свои собственные функции. Эта строка - своего рода 'установка' функции.
void - Это слово сообщает скрипту, какого рода 'ответ' выдаст функция.
Слово main сообщает, что это основная часть скрипта. Мы можем написать множество функций в нашем скрипте, но 'main' - эта та часть скрипта, которая будет непосредственно выполняться. Между '(' и ')' как правило пишут входящие данные, необходимые скрипту для корректного исполнения. Наш скрипт не требует никаких входных данных, и это значит, что между скобками ничего не пишется. Всегда необходимо ставить '(' и ')', даже если нет никаких входящих данных.

Фигурные скобки необходимы, чтобы отделять блоки скрипта один от другого. В нашем случае это обозначает, что все то, что мы написали, является частью void main(). Каждая строка заканчивается знаком ';'.

Про интеджер и флоат мне опять писать или что? Что непонятно? Тебе непоянтно что такое целое число а что нет?

Так же, в скрипте все типы данных и объекты обычно представляются в начале скрипта например так:

Neverwinter Script Source
//:://////////////////////////////////////////////
//:: OnEnter [CUT_FIN_1] am_cut_final_1
//:: Copyright © 2006 WRG!
//:://////////////////////////////////////////////
/*
         ФИНАЛЬНЫЙ РОЛИК ИСТОРИИ
*/

//:://////////////////////////////////////////////
//:: Created By: Aiwan
//:: Created On: 23.03.2006
//:://////////////////////////////////////////////
#include "am_inc_cutscene"
#include "am_inc_common"

void main()
{
//------------------------------- СОЗДАНИЯ -------------------------------------
   object oPC = GetFirstPC(); // Игрок
   object oBoss = GetObjectByTag("AM_CUT1_BOSS"); // Главный
   object oNpc1 = GetObjectByTag("AM_CUT1_NPC_01"); // НПС 1
   object oNpc2 = GetObjectByTag("AM_CUT1_NPC_02"); // НПС 2
   object oDog = GetObjectByTag("AM_CUT1_DOG"); // Собака
   object oDeer = GetObjectByTag("AM_CUT1_DEER"); // Олень
   object oOx = GetObjectByTag("AM_CUT1_OX_01"); // Бык (живой)
   object oOxD = GetObjectByTag("AM_CUT1_OX_02"); // Бык (мертвый)
   object oArea = GetArea(oBoss);
//-------------------------------- ОБЪЕКТЫ -------------------------------------
   object oSound = GetObjectByTag("AM_CUT1_SOUND_WOLF"); // Звук воя волков
   object oSound1 = GetObjectByTag("AM_CUT1_SOUND_WOLF1"); // Звук воя волков
   object oFlame = GetObjectByTag("AM_CUT1_FLAME"); // Костер (потухший)
   object oGate = GetObjectByTag("AM_D_EXT_HUNT_GATE"); // Ворота во двор
   object oDoor = GetObjectByTag("AM_D_EXT_HUNT_H_IN"); // Дверь домика
//------------------------ ТОЧКИ ДВИЖЕНИЯ И КАМЕРЫ -----------------------------
   object oWpCamPC1 = GetWaypointByTag("AM_WP_CUT1_CAM_PC_01"); // Точки Игрока
   object oWpCamPC2 = GetWaypointByTag("AM_WP_CUT1_CAM_PC_02");
   object oWpExit = GetWaypointByTag("AM_WP_CUT1_EXIT"); // Для ухода с карты
   object oWpIn = GetWaypointByTag("AM_WP_CUT1_IN"); // Эмитация входа в дом
   object oWpP1 = GetWaypointByTag("AM_WP_CUT1_POST_01"); // Точка НПС
//-------------------------- ВИЗУАЛЬНЫЕ ЭФФЕКТЫ --------------------------------
   effect eCutInv = EffectVisualEffect(VFX_DUR_CUTSCENE_INVISIBILITY);
   effect eCutPar = EffectCutsceneParalyze();
   effect eGhost = EffectVisualEffect(VFX_DUR_GHOSTLY_VISAGE);
   effect eParalyz = EffectVisualEffect(VFX_DUR_FREEZE_ANIMATION); // Заставим застынуть
   effect eFreeze = EffectVisualEffect(VFX_DUR_ICESKIN); // Заморозим
//-------------------------- Поврежедния ---------------------------------------------
   int iMax = GetMaxHitPoints(oPC);
   int iD = iMax-5;


В таком виде мы обычно представляем все объекты и данные с которыми будем работать в скрипте. В скрипте мы обращаемся к объекту, но мы должны четко дать понять какой именно объект нам нужен. Самый простой способ это указать OBJECT_SELF. Это значит тот кто содержит или исполняет скрипт написаный вами будет взят в качестве объекта. К примеру, скрипт стоящий на НПС может взять object oNpc = OBJECT_SELF; Теперь вы заметили, что мы как то структурировали наши сокращения в описании объектов. Например, зачем нам постоянно писать одно и то же десятки раз типа: GetObjectByTag("AM_CUT1_BOSS") - тут мы говорим скрипту взять объект с тегом AM_CUT1_BOSS. Скрипт ищет в модуле у кого такой тег и берет его. Каждый раз это долго и неправильно. Мы попросту в начале найдем этого НПС и объявим его сократив до минимума oBoss. Вся строка будет выглядеть так:

Neverwinter Script Source
object oBoss = GetObjectByTag("AM_CUT1_BOSS"); // Главный


в ней мы говорим, что наш object oBoss ЭТО ОБЪЕКТ это не число и не эффект. Соотвественно для удобства мы сократили и поставили впереди буковку "o" что бы мы знали, что это ОБЪЕКТ. У интеджер мы обычно ставим буковку "i" и встретив в большом скрипте такие данные oPC, iMax, fDel мы уже понимаем что первый это ОБЪЕКТ так как у него буковка о, второй это число так как у него буковка i, третий это флоат нецелое число так как перед ним буковка f. Но все это ДЛЯ УДОБСТВА и никак не влияеет на работу скриптов. Это ПРАВИЛЬНЫЙ И ХОРОШИЙ ТОН.

Внизу простой пример как все правильно описав и сократив мы накладываем на OBJECT_SELF еффект.

Neverwinter Script Source
void main()
{
   object oSelf = OBJECT_SELF;
   int iMax = GetMaxHitPoints(oSelf);
   int iD = iMax-5;
   effect eDamage = EffectDamage(iD);

   AssignCommand(oSelf, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,
Self));
}

Neverwinter Script Source
void main()
{
   int iMax = GetMaxHitPoints(OBJECT_SELF);
   int iD = iMax-5;
   effect eDamage = EffectDamage(iD);
   AssignCommand(OBJECT_SELF, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,OBJECT_SELF));
}

Neverwinter Script Source
void main()
{
   int iD = GetMaxHitPoints(OBJECT_SELF)-5;
   effect eDamage = EffectDamage(iD);
   AssignCommand(OBJECT_SELF, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,OBJECT_SELF));
}

Neverwinter Script Source
void main()
{
   effect eDamage = EffectDamage(GetMaxHitPoints(OBJECT_SELF)-5);
   AssignCommand(OBJECT_SELF, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,OBJECT_SELF));
}

Neverwinter Script Source
void main()
{
   AssignCommand(OBJECT_SELF, ApplyEffectToObject(DURATION_TYPE_INSTANT,EffectDa
age(GetMaxHitPoints(OBJECT_SELF)-5,OBJECT_SELF)));
}

Сравните первый код и этот последний. Они идентичны, но в каждом последующем я убрал описание по одному объекту или эффекту. В результате кусок кода разобрать на порядок труднее. И это ОДНА СТРОКА! А что будет если у вас тысяча строк?

Добавлено через 2 минуты 49 секунд

Я не прошу понимать что делает этот скрипт мне важно что бы вы на каждом следующем примере видели какое изменение произошло.

Добавлено через 11 минуту 31 секунду

п.с. При переносе форум съедает одну букву в строке NSS кода. Не пинать мну, не моя вина. Нео обещал найти причину.
Ёжик
Цитата(Aiwan @ Oct 7 2007, 19:44) *
я могу дать все эти понятия, но то что ты поймешь что это я не уверен

ну тогда вычеркни меня и всё
Aiwan
Цитата(Ёжик @ Oct 8 2007, 08:31) *
ну тогда вычеркни меня и всё

ВЫЧЕРКИВАЮ.

Добавлено через 9 минут 34 секунды

п.с. основные понятия по объектам и прочему даны в переводах Лексикона. Кому что то неясно почитайте еще там.
Перевод Лексикона

Добавлено через 2056 секунд

Я ХОЧУ ЧТО БЫ ВЫ ПОНЯЛИ. Мне не надо что бы вы поняли что сделал тот или иной скрипт. В тот самый момент, когда вы сможете ответить на это и на вопросы которые описала Ёжик в первом посте МОЖНО СЧИТАТЬ ЧТО ВЫ УЖЕ НАУЧИЛИСЬ ПОНИМАТЬ СКРИПТЫ. Я не могу за вас думать, вы должны напряч свой мозг и отогнать свою лень. Без этого не получится диалога здесь. Я ПРОШУ ПОНИМАТЬ ТО ЧТО ДАЮ Я. Если я дал описание чего то то и понимать осознавать надо ЭТО а не все, что содержит мой пример кода.
Griffon
Цитата(Aiwan @ Oct 7 2007, 20:44) *
Между '(' и ')' как правило пишут входящие данные, необходимые скрипту для корректного исполнения.


1. Это ты про () после void main? Спрашиваю поскольку на примере твоего скрипта вот это же и есть данные
Neverwinter Script Source
object oSelf = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oSelf);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);


2. Зачем так сильно сокращать? Ведь в таком виде - понятно какие данные используются и как они прописаны в команде, разве нет?
Neverwinter Script Source
void main()
{
  object oSelf = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oSelf);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);

AssignCommand(OBJECT_SELF, ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectDamage-5,OBJECT_SELF));
}


3. DURATION_TYPE_INSTANT - вроде же есть цифровое значение соответствующее данной фразе? И если да, то в данных можно указать его через int, а в команде (AssignCommand) уже прописать саму цифру, или не так?

Добавлено через 26 минут 36 секунд

Цитата(Aiwan @ Oct 8 2007, 11:21) *
п.с. основные понятия по объектам и прочему даны в переводах Лексикона. Кому что то неясно почитайте еще там.
Перевод Лексикона
Ссылка битая

Добавлено через 6 минут 49 секунд

P.S. А может сайт опять полег?
Aiwan
Griffon, Я ПРОСТО ПОКАЗАЛ ЧТО ПОЛУЧИТСЯ если не описывать и не сокращать. и не забегай вперед. Все разберем acute.gif Я про константу.

Добавлено через 1 минуту 23 секунды

Мне важно что бы вы УВИДЕЛИ что изменилось от шага к шагу.

Добавлено через 1 час 44 секунды

Уточнение. Что бы ыбло понятно.
Представление объекта, можно сравнить с прозвищем. Мы же не говорим постоянно Иванов Иван Иванович, мы говорим просто Вано. Коротко и понятно. Вот это выглядит так:

Neverwinter Script Source
object oВано = GetObjectByTag("Иванов Иван Иванович");
Neo
Цитата(Griffon @ Oct 8 2007, 12:11) *
DURATION_TYPE_INSTANT

есть константа, что означает - постоянно принятая величина для данного скрипта, обозначенная идентификатором
в данном случае идентификатор - DURATION_TYPE_INSTANT, а величина описана в nwmain.nss
костанты могут быть числовыми и текстовыми, да любого типа
мастера, не пинайти сильна, объяснил как асилил

Добавлено через 1 минуту 33 секунды

Цитата(Ёжик @ Oct 8 2007, 06:31) *
ну тогда вычеркни меня и всё

Цитата(Aiwan @ Oct 8 2007, 11:21) *
ВЫЧЕРКИВАЮ.

ыххыыыыхыыыыыыы, ну вы ребята даете smile.gif а я не умею скриптить как лекс, вычеркивайте меня из БАНДЫ, что за детство, ей богу
Aiwan
Не надо сложных понятий. Не надо пытаться понять что-то что я не прошу понять. Поймите то что даю. Пусть пока вам не понятны какие то слова в скрипте. Дойдем до этого.

Добавлено через 107 секунд

Так первое домашнее задание. Откройте тулсет и из этого скрипта сделайте вот что:
Neverwinter Script Source
void main()
{
  object oSelf = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oSelf);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);

  AssignCommand(oSelf, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,
Self));
}

Каждый в скрипте имя oSelf сменит на свой ник, например oAiwan и напишет в этой теме СВОЙ КОД. ПИШИТЕ В РЕДАКТОРЕ И КОМПИЛИТЕ. ЗАТЕМ ПЕРЕНОСИТЕ СЮДА.

п.с. ЕЖИК СНОВА В ТЕМЕ. Понедельник трудный день. Извини.

Добавлено через 2 часа 45 минут 52 секунды

ЕЩЕ РАЗ ПОВОРЯЮ ДЛЯ ВСЕХ. В ЭТОЙ ТЕМЕ ТОЛЬКО МОИ ПОСТЫ И ПОСТЫ ОБУЧАЮЩИХСЯ.
Solo
ну если я правильно понял задание, то справился с ним yahoo.gif
вот что получилось:
Neverwinter Script Source
void main()
{
  object oSolo = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oSolo);
  int iD = iMax - 5;
  effect eDamage = EffectDamage(iD);

  AssignCommand( oSolo, ApplyEffectToObject( DURATION_TYPE_INSTANT, eDamage, oSolo ) );
}


вот что написал компилятор:
08.10.2007 20:40:08: 0 Ошибок. 'script1' составление успешно
Aiwan
Да ,все правильно. Но в следующий раз юзай тег NSS на форуме. yes3.gif
Vhall
Neverwinter Script Source
void main()
{
  object oVhall = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oVhall);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);

  AssignCommand(oVhall, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,
oVhall));
}

В итоге результатом стало понимание работы бажного nss.
Artlira
Neverwinter Script Source
void main()
{
  object oLira = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oLira);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);

AssignCommand(oLira, ApplyEffectToObject(DURATION_TYPE_INSTANT, eDamage,oLira));

}



Мне кажется, единственная ошибка, которая могла быть в этом скрипте, это если в какой-нибудь строке не поменять oSelf на другое определение объекта?
Aiwan
Цитата(Artlira @ Oct 9 2007, 05:22) *
Мне кажется, единственная ошибка, которая могла быть в этом скрипте, это если в какой-нибудь строке не поменять oSelf на другое определение объекта?

Совершенно верно. Мне важно, что бы вы поняли что oSelf это просто имя описываемого объекта. И оно используется во многих местах.
Griffon
Neverwinter Script Source
void main()
{
    object oGriffon = OBJECT_SELF;
    int iMax = GetMaxHitPoints(oGriffon);
    int iD = iMax-5;
    effect eDamage = EffectDamage(iD);
   
    AssignCommand(oGriffon, ApplyEffectToObject(DURATION_TYPE_INSTANT, eDamage,oGriffon));
}


Результат - 0.85.nss(1): ERROR: ELLIPSIS IN IDENTIFIER

Мы в каком пишем НВНе?

Aiwan
Вы все поняли, что объект можно представить в любом обозначении. Вы верно проследили, что надо во всем скрипте изменить порядок. Что бы лучше понять что происходит, я вам опишу словами. Мы попросту скрипту представляем, что у нас есть объект и мы будем с ним работать дальше. Что бы каждый раз не икать его мы сразу, четко его определяем.

Neverwinter Script Source
object oSelf = OBJECT_SELF;


Дананя строка сосотит из двух важных частей они заключены между знаком равно. В этой строке ДВА ДЕЙСТВИЯ. Первым, мы объявили скрипту, что у нас есть объект object oSelf. Второым мы ПРИРОВНЯЛИ этот наш объявлленный объект к OBJECT_SELF. Тем самый четко дали понять скрипту кто есть ХУ. Верхний код можно написать и таким образом:

Neverwinter Script Source
void main()
{
   object oSelf; // Объявляем скрипту, что у нас используется ОБЪЕКТ oSelf
   oSelf = OBJECT_SELF; // Определяем его прировняв к OBJECT_SELF
}


То же самое мы делаем и с двумя цифровыми данными в скрипте:

Neverwinter Script Source
int iMax = GetMaxHitPoints(oSelf);
    int iD = iMax-5;


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

Neverwinter Script Source
void main()
{
    object oSelf; // Объявляем скрипту, что у нас используется ОБЪЕКТ
    oSelf = OBJECT_SELF; // Определяем его прировняв

    int iMax; // Объявляем, что у нас будет использоваться интеджер с именем iMax
    iMax  = GetMaxHitPoints(oSelf); // Он будет равен >>
    // Get Max Hit Points ВЗЯТЬ МАКСИМАЛЬНЫЕ ХИТ ПОИНТЫ объекта oSelf

    int iD;  // объявляем, что у нас будет еще оно число
    iD = iMax-5; // Оно будет равно от максимального числа хитпоинов отнять 5
}


В общем суть на пальцах такая. Мы говорим: "В нашей команде есть ИГРОКИ (object, int, float). У них есть своя специализация и ФАМИЛИЯ." Когда мы объявляем ИГРОК то это представление объекта. Когда говорим ИГРОК ПЕТРОВ, ВРАТАРЬ СИДОРОВ, ЗАЩИТНИК ИВАНОВ, то это представление и определение кто именно. Игроки могут меняться и наше определение объекта может меняться в скриптах тоже. ЗАЩИТНИКИ, НАПАДАЮЩИЕ могут смениться. Не важно кто защищает ворота и кто в нападении. Мы можем их сменить. Но прежде чем ввести их в игру судья объявляет что игроки поменялись! Так же можно менять в скрипте.

Neverwinter Script Source
void main()
{
    object oSelf; // Объявляем скрипту, что у нас используется ОБЪЕКТ
    oSelf = OBJECT_SELF; // Определяем его прировняв

    int iMax; // Объявляем, что у нас будет использоваться интеджер с именем iMax
    iMax  = GetMaxHitPoints(oSelf); // Он будет равен >>
    // Get Max Hit Points ВЗЯТЬ МАКСИМАЛЬНЫЕ ХИТ ПОИНТЫ объекта oSelf

    int iD;  // объявляем, что у нас будет еще оно число
    iD = iMax-5; // Оно будет равно от максимального числа хитпоинов отнять 5

    effect eDamage; // объявляем, что у нас используется эффект
    eDamage = EffectDamage(iD); // Эффект Effect Damage поврежедения равен числу iD

    if(GetIsPC(oSelf)==TRUE) // Если oSelf это РС
    {
      // Если это игрок то мы сменим определение еффекта и вылечим РС
      eDamage = EffectHeal(iMax); // на число iMax его максимальные хитпоинты
    }
    AssignCommand(oSelf, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,
Self));
}


Если есть вопросы спрашивайте.

Добавлено через 106 секунд

Мне важно, что бы вы поняли описание объектов и их определение можно менять в скрипте.

Добавлено через 54 секунды

Скажите что изменилось в скрипте?
Neverwinter Script Source
void main()
{
  object oSelf = OBJECT_SELF;
  int iMax = GetMaxHitPoints(oSelf);
  int iD = iMax-5;
  effect eDamage = EffectDamage(iD);

  if(GetIsPC(oSelf)==TRUE)
  {
      eDamage = EffectHeal(iMax);
  }
  AssignCommand(oSelf, ApplyEffectToObject(DURATION_TYPE_INSTANT,eDamage,
Self));
}
Artlira
Изменилась сама программа скрипта. Теперь он работает выборочно, то есть одному из объектов – в данном случае игроку будет явная выгода (лечение), а у всех прочих по-прежнему изымет по 5 хитов?
Aiwan
Самое главное что мы сделали? Мы ПОМЕНЯЛИ ефекты.
Griffon
Цитата(Artlira @ Oct 9 2007, 14:18) *
а у всех прочих по-прежнему изымет по 5 хитов

Эффекты конечно же сменились, но чуть поправлю предыдущее высказывание - оно будет верным, если скрипт не запускается единожды. В противном случае или пан или пропал. Причем оба и игрок и его соперни к на шарде (если для него) или нпс (он ведь тоже попадет под раздачу, если станет выпонять определенные действия.
Artlira
Если я правильно понимаю, то смысл в том, что скриптом можно определить любой, необходимый в конкретном случае объект (непись, игрок, плейс). И заставить этот объект выполнить действие (серию действий) или (в этом примере) наложить на него эффект, который нужен в определенной ситуации?
Смысл не в конкретном скрипте, а в том, как и что можно заставить работать?
Griffon
Все верно, основной смысл кода в том - кто и что будет делать.
Aiwan
Griffon, не гони лошадей многим это будет не понять. declare.gif Сейчас важно видеть ПРОСТОЕ то что я говорю.
Griffon
Все, передаю слово учителю. Хотя учащимся стоит понять, что такое код и что он может делать, ну конечно, при ограничениях по имеющимся функциям.
Vhall
QUOTE(Aiwan @ Oct 9 2007, 13:48) *
Скажите что изменилось в скрипте?

Объекты, целые и эффекты задаются по-разному?
Griffon
Цитата(Griffon @ Oct 9 2007, 12:24) *
Neverwinter Script Source
void main()
{
object oGriffon = OBJECT_SELF;
int iMax = GetMaxHitPoints(oGriffon);
int iD = iMax-5;
effect eDamage = EffectDamage(iD);

AssignCommand(oGriffon, ApplyEffectToObject(DURATION_TYPE_INSTANT, eDamage,oGriffon));
}


Результат - 0.85.nss(1): ERROR: ELLIPSIS IN IDENTIFIER


Почему же пишет что ошибка? И что это за ошибка?
Aiwan
Цитата(Griffon @ Oct 10 2007, 18:35) *
Почему же пишет что ошибка? И что это за ошибка?

Может из за НВН2 функций. Я пока компилю на первом НВН так как второй еще не ставил, карту недавно купил и винду переставил. Попробуй в первом редакторе компильнуть. Не переживайте логика и все что в первом НВн то и во втором. Но в редакторе первого НВН легче и понятней. Так что рекомендую юзать пока его.
mamuc
Цитата
Результат - 0.85.nss(1): ERROR: ELLIPSIS IN IDENTIFIER

Griffon у тебя имя файла "0.85" ?
Griffon
Цитата(mamuc @ Oct 11 2007, 11:43) *
Griffon у тебя имя файла "0.85" ?


Какого? Что за файл? Я не сохранял скрипт, написал и просто нажал скомпилировать. Может после перехода на след строку не нужно табуляцией сдвигать ее в право, он сам сдвинет?
mamuc
в тулсете при компиляции обязательно создаеться файл rolleyes.gif

Цитата
Результат - 0.85.nss(1): ERROR: ELLIPSIS IN IDENTIFIER


сообщение о ошибке следует читать так
"0.85" имя файла меня здесь делает осторожным присутствие точки ( есть древнее правило неупотреблять в именах некоторые специальные симболы, в частности точка в имени файла должна отделять имя и тип файла и не использоваться никак иначе)
"nss" - это и есть тот тип файла про который я упоминал выше
"(1)" - номер строки в котором произошла ошибка
"ERROR: ...." что именно компилятору не нравиться
попробуй написать скрипт и сохранить его под каким то другим именем и потом прокопилировать
Griffon
Уфффф..... Пока разобрался crazy.gif

2mamuc, спасибо за подсказку, помогло найти что вызывало сбой.

По порядку... та ошибка появлялась когда просто создавал новый скрипт - файл -> создать -> скрипт. Оболочка редактора у меня почему-то на русском, видно из-за региональных настроек, надо будет проверить перед очередным запуском.

Затем создал локацию -> затем новый скрипт -> написал, причем именно так без отступлений
Neverwinter Script Source
void main()
{
object oChest = OBJECT_SELF;
object oItem = GetFirstItemInInventory(oChest);
while (GetIsObjectValid(oItem))
{
DestroyObject(oItem);
oItem = GetNextItemInInventory(oChest);
}


Получил ошибку - None.nss(1): ERROR: INVALID DECLARATION TYPE □

После этого переименовал локацию с зона1 на zone1 и Скрипт1 на script1 и еще раз -> компилировать. И, о чудо! выдал ответ, что компиляция успешна, файл находится там-то и там-то. Спасибо.



Добавлено через 23 минуты 13 секунды

P.S. Сменил язык на английский и компилит без проблем file -> new -> script biggrin.gif
Lex
Цитата(Griffon @ Oct 11 2007, 14:26) *
Скрипт1 на script1

самое важное. На русском названия скриптов низя писать. Поэтому от русификаторов в работе с тулсетом лучше держаться подальше - русские дефолтовые названия порождают проблемы.
mamuc
to Griffon только добавь в тобой приведенном примере еще одну закрывающую скобку плз biggrin.gif
Aiwan
Такс... Продолжаем разговор (с) Карлсон который живет на крыше.

Мы с вами уже изучили то, что объекты, еффекты и прочие данные можно и НУЖНО описывать заранее. Теперь мы немного разберем то, как определить объект? Мы взяли очень простую схему OBJECT_SELF. попросту говорим что oSelf это ты сам. Но не всегда так просто можно найти нужный объект. В тулсете полно функций, которые определяют объект, например по тегу.

Neverwinter Script Source
object oNpc = GetObjectByTag("NPC_AIWAN");


Функция Get Objec tBy Tag говорит Взять Объект С Тегом. Но вы должны учитывать, что такой объект должен быть с уникальным тегом, что бы ваш скрипт работал нормально. Либо вы должны будете описать еще условия которые позволят найти вам ваш нужный объект. К примеру, если у вас несколько одинаковых НПС и вам нужен один из них то омжно воспользоваться функцией Get Nearest Object By Tag которая говорит Взять Ближайший Объект с Тегом. Эта функция имеет несколько параметров, в ней можно задать условие ближайший объект от кого? Ведь надо точно указать точку отсчета ближайшего объекта от, например двери. По умолчанию у скрипт будет брать ближайший объект от того кто запустил скрипт. Если скрипт стоял на триггере то возьмет ближайший от него. Дети ктороые играют в салочки кричат: "Кто последний тот тухлый банан! Кто первый тот чемпион!" и бегут в дверь. Так вот, чемпион будет именно тот, кто первым добежит и он скорее всего самый ближайший объект у двери. А зачастую он и кричит такую фразу. Всем задание. Создайте в редакторе модуль. Поставьте одну внутреннюю локацию. В ней три нпс с тегами NPC_ВАШ_НИК, поставьте дверь с тегом DOOR_ВАШ_НИК. Начертите подальше от двери триггер и на сло OnEnter поставьте скрипт и настройте его в соответсвии с вашими тегами.

Neverwinter Script Source
// Поиск чемпиона канады
void main()
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag("DOOR_AIWAN_01"); // Наша дверь
  object oNpc = GetNearestObjectByTag("NPC_AIWAN", oDoor);
  object oPC = GetEnteringObject(); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if(GetIsPC(oPC)) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand(oNpc, ClearAllActions()); // Чистим стек команд
    AssignCommand(oNpc, SpeakString(sSpeak)); // Говорит фразу
    AssignCommand(oNpc, ActionForceMoveToObject(oDoor, TRUE)); // Бежит к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}


В этом скрипте мы находим объекты как то: по тегу (Двери), по действию (Тот кто наступил на триггер), и берем ближайший с тегом. Мне важно что бы вы хотя бы попробовали и поняли как это все работает.

Для того что бы понять что я дал поставьте НПС в линейку между триггером и дверью. Потом посомтрев что происходит вы удалите в строке:
Neverwinter Script Source
object oNpc = GetNearestObjectByTag("NPC_AIWAN", oDoor); // Измените на
  object oNpc = GetNearestObjectByTag("NPC_AIWAN"); // ЭТО


Мы поменяли точку отсчета и теперь к двери побежит ближайший к ТРИГГЕРУ объект. У кого все получилось говорите. У кого что не получилось тоже. Жду...
Artlira
[ OFFTOP ]
В первом варианте к определение объекта (oNpc) было указано, как объект с заданным тагом, ближайший к двери (oDoor). Тот из трех неписей, который был ближе всех к двери к ней и побежал и произнес фразу.
Neverwinter Script Source
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "NPC_LIRA", oDoor ); //Ближайший к двери объект
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if( GetIsPC( oPC )) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak )); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE )); // Бежит к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}

Во втором варианте изменилось определение объекта, то есть не было указано, ближайший к чему. Наверно, поэтому, к двери побежал тот непись, который был ближе к триггеру. Сменился приоритет объекта – по отношению к которому выбирался ближайший. Еще я убрала TRUE – непись действительно не побежал, а пошел!
Neverwinter Script Source
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "NPC_LIRA" ); //Ближайший к триггеру объект
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if( GetIsPC( oPC )) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak )); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor )); // Идет к двери
  }
}

Вроде все получилось.
Lex
советую так же, как и на внеклассных занятиях, прятать скрипты, чтобы другие участники случайно не увидели. smile.gif
Vhall
О! Побежал. -_-
[ OFFTOP ]
Neverwinter Script Source
// Поиск чемпиона канады
void main()
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag("PLC_DC_Slum01"); // Наша дверь
  object oNpc = GetNearestObjectByTag("n_watchman", oDoor);
  object oPC = GetEnteringObject(); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if(GetIsPC(oPC)) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand(oNpc, ClearAllActions()); // Чистим стек команд
    AssignCommand(oNpc, SpeakString(sSpeak)); // Говорит фразу
    AssignCommand(oNpc, ActionForceMoveToObject(oDoor, TRUE)); // Бежит к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}


Neverwinter Script Source
// Поиск чемпиона канады
void main()
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag("PLC_DC_Slum01"); // Наша дверь
  object oNpc = GetNearestObjectByTag("n_watchman", oDoor);
  object oPC = GetEnteringObject(); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if(GetIsPC(oPC)) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand(oNpc, ClearAllActions()); // Чистим стек команд
    AssignCommand(oNpc, SpeakString(sSpeak)); // Говорит фразу
    AssignCommand(oNpc, ActionForceMoveToObject(oDoor)); // Идет к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}
shadowdweller
Бежит!
[ OFFTOP ]
Neverwinter Script Source
// Поиск чемпиона канады
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_KOSHIKO" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "KOSHIKO", oDoor );
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последним прибежит, тот прибежит последним!!"; // Фраза ( (С ) Тигра )

  if( GetIsPC( oPC ) ) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}

Neverwinter Script Source
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_KOSHIKO" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "NPC_KOSHIKO" ); //Ближайший к триггеру объект
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak = "Кто последним прибежит, тот прибежит последним!!"; // Фраза  ( (С ) Тигра )

  if( GetIsPC( oPC ) ) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor ) ); // Идет к двери
  }
}

Aiwan
Продолжаем.
Итак, мы уже говорили о том, что объект мало описать и представить, его надо точно ОПРЕДЕЛИТЬ. В нашем скрипте про трех НПС мы находили игрока взяв любого, кто наступил на триггер, потом мы поставили условие, что только тот если он представляет из себя РС выполнить следующее.

Neverwinter Script Source

  if( GetIsPC( oPC ) ) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
  {
      // ТУТ ЛЮБОЕ ДЕЙСТВИЕ
  }


Этот кусок кода содержит функцию, которая вовращает результат в виде TRUE или FALSE. Вы напряглись и начали дико соображать. О как все сложно! Никапельки. Объясняю. Все функции в тулсете выдают какой то результат, производят действия и т.д. Результат может быть в виде любый знакомых нам данных (integer, object, string, vector и т.д.) либо ИСТИНА/ЛОЖЬ. Мы в нашей скрипте проверяли истино ли то, что тот кто наступил на триггер игрок? Но я написал эту строку сокращенно, по всем правилам она будет выглядеть так:

Neverwinter Script Source

   if( GetIsPC( oPC )==TRUE ) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ
   {
      // ТУТ ЛЮБОЕ ДЕЙСТВИЕ ЕСЛИ НАСТУПИВШИЙ == ИСТИНА
   }


Вот так правильно описать, но в написании скриптов есть свои сокращений и вам их надо будет знать. Функция GetIsPC проверяет объект на то, игрок он или нет (может он ДМ или НПС). И выдает результат в виде TRUE (ИСТИНА) FALSE (ЛОЖЬ). Если не ставить знак проверки результата что делают многие в описании своих скриптов, то результат ИСТИНА будет внутри скобок а ЛОЖЬ дальше. Теперь смотрите как выглядит сама функция:

Neverwinter Script Source

// * Returns TRUE if oCreature is a Player Controlled character.
int GetIsPC( object oCreature )


Видите, она в самом начале показывает, что результатом будет ЧИСЛОВОЕ значение в нашем случае FALSE это НОЛЬ 0 либо TRUE ЭТО ЕДИНИЦА 1. В компьютере все данные представлены именно в таком виде либо да либо нет smile.gif Если разобрать подробно, то функция будет выглядеть так:

Neverwinter Script Source

int GetIsPC( object oCreature )
// Результат либо FALSE либо TRUE ИГРОК_ЛИ( Объект, который мы проверяем )


Еще я употребил ДВА занка равенства вместе. Это проверка равенства. Вот если я употребляю один знак равно, как в описании объектов, то я ПРИРАВНИВАЮ а не проверяю. Понимаете разницу? Написав "ЛЕКС = ВАНЮЧКА" будет звучать как Лекс ЭТО ВАНЮЧКА! А "ЛЕКС == ВАНЮЧКА" будет означать "Лекс ВАНЮЧКА или нет?" Утверждение "ЛЕКС != ВАНЮЧКА" будет звучать как "Лекс НЕ ВАНЮЧКА!"

Код
= =    Тест равенства ("равняется").
!=    Тест не равенства ("не равняется").
<    Меньше чем, тест равенства
>    Больше чем, тест равенства
<=    Меньше или равно, тест равенства
>=    Больше или равно, тест равенства
&&    Логическое И.
&    Битовое И.
||    Логическое ИЛИ.
|    Битовое ИЛИ.
!    Логическое НЕ.


Вот все логические операторы. Можно не пугаться нам пока нужны основные и все примеры мы разберем подробно. Вам не надо забивать голову всеми правилами сокращений, главное, что бы вы понимали суть скрипта, а уж пишите как вам удобно и главное ПОНЯТНО. Когда вы научитесь хорошо понимать код вы сами все свои написания приведете в должный порядок.

Я сейчас покажу как можно записать один и тот же результат разными способами:

Neverwinter Script Source
   if( GetIsPC( oPC ) ) // ЕСЛИ тот кто наступил ИГРОК ТО ВЫПОЛНЯЕМ

   if( GetIsPC( oPC )==TRUE ) // ЕСЛИ тот кто наступил ИГРОК равено TRUE

   if( GetIsPC( oPC )==1 ) // ЕСЛИ тот кто наступил ИГРОК равно 1

   if( GetIsPC( oPC )!=FALSE ) // ЕСЛИ тот кто наступил ИГРОК НЕ равно ЛОЖЬ

   if( GetIsPC( oPC )!=0 ) // ЕСЛИ тот кто наступил ИГРОК НЕ равно 0


Так вот, вернемся в начало. Мы в скрипте брали того, кто наступит на триггер и проверяли это ли игрок? Это очень важно правильно задать вопрос, что бы получить на него правильны и нужный нам ответ для скрипта. В тулсете много функций которые помогают найти объект по действиям последнего: кто наступил, кто сошел, кто ближе, кто получил урон, кто умер и т.д. Естественно на слоте триггера OnEnter трудно получить результат GetLastOpenedBy() (кто последний открыл). Это одна из первых основных ошибок. Учитывайте где утсановлен скрипт и если вы взяли кусок кода из скрипта на сундуке то он может не подойти в описании РС для триггера.

Теперь задание. Ваш рабочий скрипт про трех НПС поместите в слот OnExit триггера и опишите игрока правильно при этом условие if(GetIsPC(oPC)) должно быть любым другим способом описанным выше мной. Результаты публиковать, но главное что бы был ответ и понимание того что вы сделали. Отвечать в тему. Жду.
Artlira
Вот что получилось:
[ OFFTOP ]
В первом варианте скрипт стоял на OnEnter, поэтому игрок определялся как объект, наступивший на триггер.
Если скрипт поставить на OnExit, то надо будет поменять определение объекта, с ”того кто наступил”(GetEnteringObject()), на ”того кто сошел с триггера”(GetExitingObject())?
То есть object oPC = GetExitingObject()
И сам скрипт в таком случае будет выглядеть так:
Neverwinter Script Source
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "NPC_LIRA", oDoor ); //Ближайший к двери объект
  object oPC = GetExitingObject( ); // Тот кто сошел с триггера
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

  if( GetIsPC( oPC )==TRUE ) // ЕСЛИ тот кто наступил ИГРОК равно Истина( TRUE )
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери
  }
}

И сработать скрипт будет должен после того, как игрок сойдет с триггера, а не наступит на него.
Aiwan
Господа и дамы. Так мы все позабываем...Давайте хотя бы перерывы большие не делать. Процесс обучения может быть только обоюдный я говорю, вы делаете простые задания. ТОЛЬКО я писать не буду, толку от этого без практики мало.
Vhall
[ OFFTOP ]
Neverwinter Script Source
// Поиск чемпиона канады
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "PLC_DC_Slum01" ); // Наша дверь
  object oNpc = GetNearestObjectByTag( "n_watchman", oDoor );
  object oPC = GetExitingObject( ); // Тот кто СОШЕЛ с триггера
  string sSpeak = "Кто последний тот тухлый банан!"; // Фраза

if( GetIsPC( oPC )==1 ) // ЕСЛИ тот кто наступил ИГРОК равно 1
  {
    AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
    AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
    AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери
    // Если убрать TRUE то НПС просто пойдет пешком...
  }
}

Edited
Aiwan
Vhall, ты ошибся. В какой слот ты ставил скрипт? А каким образом определял игрока? Подумай внимательно.

Добавлено через 54 минуты 24 секунды

п.с. Теперь верно. Понял что важно при условиях срабатывания скрипта, так же правильно указывать условия поиска объекта?
Vhall
Ну как тебе сказать... -_-
Вроде бы.
Aiwan
Так. Продолжаем.

Я уже разбирал в начале, что такое переменная. Теперь самое время применить знания на практике. Но мыдолжны сперва с вами изучить структуру постороения скрипта. Так вот, внутри скрипта все делится на блоки со своими условиями. Естественно можно обойтись простым действием без блоков, но мы разберем сложный скрипт в котором четыре блока. Вот они:

Neverwinter Script Source
void main( )
{
      if ( TRUE ) // это ПЕРВЫЙ БЛОК
      {
            if ( TRUE ) // это ВНУТРИПЕРВЫЙ БЛОК относится к первому блоку он ВЛОЖЕН в него
            {
                // выполняем какое то действие для ВНУТРИПЕРВОГО БЛОКА
            }

            // тут выполняется то что мы хотим для ПЕРВОГО БЛОКА
      }

      if ( TRUE ) // это ВТОРОЙ БЛОК
      {
            // тут выполняется то что мы хотим для ВТОРОГО БЛОКА
      }

      if ( TRUE ) // это ТРЕТИЙ БЛОК
      {
            // тут выполняется то что мы хотим для ТРЕТЬЕГО БЛОКА
      }
}


Теперь мы создадим ситуацию, когда в нашем уже готовом скрипте будет выполнено действие при помощи присвоения переменных внутри блоков. Попросту, мы выполняем проверку. Если условия верны выполняем блок и присваиваем переменную:

Neverwinter Script Source

void main( )
{
   object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
   object oDoor = GetObjectByTag( "DOOR_AIWAN_01" ); // Наша дверь
   object oNpc; // Объявим НПС но не укажем, кто он именно
   object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
   string sSpeak = "Кто последний тот тухлый банан!";

   if( GetIsPC( oPC ) == TRUE ) // Если это верно ТО выполняется то что между скобок
   { // ПЕРВАЯ открытая скобка

     /* ВТОРОЙ ВНУТРЕННИЙ БЛОК */

     if ( GetLocalInt( oTrigger, "AIWAN_PC_DO_IT" )== 1 ) // Если локалка РАВНА 2 ТО:
     { // ВТОРАЯ открытая скобка

        oNpc = GetNearestObjectByTag( "NPC_AIWAN" );
        SetLocalInt( oTrigger, "AIWAN_PC_DO_IT", 2 );
        SendMessageToPC( oPC, "Сработал второй блок" );

     } // ВТОРАЯ закрытая скбка

     /* ТРЕТИЙ ВНУТРЕННИЙ БЛОК */

     if ( GetLocalInt( oTrigger, "AIWAN_PC_DO_IT" ) == 0 )
     { // ТРЕТЬЯ открытая

        oNpc = GetNearestObjectByTag( "NPC_AIWAN" );
        SetLocalInt( oTrigger, "AIWAN_PC_DO_IT", 1 );
        SendMessageToPC( oPC, "Сработал третий блок" );
     }  // ТРЕТЬЯ закрытая

     /* ТУТ НУЖНО ДОПИСАТЬ ЕЩЕ ОДИН БЛОК*/

      SendMessageToPC( oPC, "Сработал ОСНОВНОЙ блок" );
      AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
      AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
      AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери

   } // ПЕВАЯ закрытая скобка

}


В этом скрипте, каждый раз наступая на триггер один НПС бежит к двери. Что бы вам было понятно какой кусок кода работает я работу скрипта пояснил дебаг сообщениями. Этот скрипт сработает ДВА раза. Ваша задача, сделать так, что бы все ТРИ нпс убежали, а попросту, сдобавить один блок внизу, что бы скрипт срабатывал постоянно. Название переменной содержит ВАШ ник. Для самых сообразительных задача посложнее. Сделать так, что бы каждый раз НПС говорили разные фразы.

Что неясно спрашивайте.
Artlira
Что-то у меня не все получается…

Сделать так, что бы неписи говорили разные фразы у меня получилось. Просто скопировала вариант с определением object oNpc;
То есть string sSpeak; указала, что фраза будет, но не писала какая именно.
А потом в каждом блоке добавила строчку:
sSpeak = ”фраза”; (фраза_1, фраза_2)

А вот с четвертым блоком у меня почему-то не получается. Вроде, если рассуждать логически, то в 4 блоке необходимо сделать проверку на переменную LIRA_PC_DO_IT, 2 и дальше просто ”обнулить” ее. В этом случае скрипт должен будет постоянно работать, присваивая переменные триггеру, затем, обнуляя последнюю, возвращаться снова к 0, то есть к отсутствию переменной.
Записала так:
[ OFFTOP ]

Neverwinter Script Source
void main( )
{
   object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
   object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
   object oNpc;
   object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
   string sSpeak;

   if( GetIsPC( oPC ) == TRUE )
   {

  /* ВТОРОЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" )== 1 )
     {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Тухлый банан будет следующий!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 2 );
        SendMessageToPC( oPC, "Сработал второй блок" );

     }

  /* ТРЕТИЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 0 )
     {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Кто последний тот тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 1 );
        SendMessageToPC( oPC, "Сработал третий блок" );
     }

     /* ЧЕТВЕРТЫЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 2 )
     {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Сам ты тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 0 );
        SendMessageToPC( oPC, "Сработал четвертый блок" );
     }

      SendMessageToPC( oPC, "Сработал ОСНОВНОЙ блок" );
      AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
      AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
      AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери

   }

}



Скрипт действительно срабатывает постоянно, но сам 4 блок не работает… Нет фразы из этого блока и неписи бегут только по условиям блока 2 и 3…
Когда присваивается переменная LIRA_PC_DO_IT, 2, скрипт продолжая считывать условие, автоматически ее проверяет и тут же обнуляет в 4 блоке. Поэтому в следующий раз, когда игрок наступает на триггер, у триггера уже нет переменной LIRA_PC_DO_IT, 2 и он срабатывает при условии, что LIRA_PC_DO_IT ==0…
В модуле это хорошо видно по сообщениям:
”Сработал второй блок”
”Сработал четвертый блок”
”Сработал ОСНОВНОЙ блок”
Получается, что задание вроде бы выполнено, все три непися уходят, скрипт работает постоянно, за счет переменных на триггере, а вот с фразами не могу разобраться.
Не могу понять, как заставить работать и 4 блок?
Aiwan
Ты написала все верно, молодец. Я не просил, что бы в блоке что то делали. Мне важно было что бы вы поняли как изменить переменную. smile.gif Молодец good.gif Что бы все блоки работали как ты говоришь, надо просто сделать задержку DelayCommand типа:

Neverwinter Script Source

    if ( GetLocalInt( oTrigger, "AIWAN_PC_DO_IT" )== 1 ) // Если локалка РАВНА 2 ТО:
    { // ВТОРАЯ открытая скобка

        oNpc = GetNearestObjectByTag( "NPC_AIWAN" );
        DelayCommand( 0.3, SetLocalInt( oTrigger, "AIWAN_PC_DO_IT", 2 ));
        SendMessageToPC( oPC, "Сработал второй блок" );

    } // ВТОРАЯ закрытая скбка


Что бы переменная присваивалась уже тогда, когда скрипт сработал. 0.3 секунды обчно хватает. В этом скрипте и 0.1 секунды хватит, он короткий. А у тебя блоки срабатывают, просто два блока вместе. Все верно. wink3.gif И фраза берется из того блока, который сработал последним.

Добавлено через 2 минуты 1 секунду

п.с. А скажи, как еще можно изменить скрипт и сделать все три блока рабочими? Даю подсказку. Ввести новую переменную...
Artlira
Цитата
Что бы все блоки работали как ты говоришь, надо просто сделать задержку DelayCommand типа:

Понятно. Сделала через DelayCommand() - все заработало! yahoo.gif

Цитата
п.с. А скажи, как еще можно изменить скрипт и сделать все три блока рабочими? Даю подсказку. Ввести новую переменную...


Через другую переменную заставить просто работать 4 блок у меня получилось. Перенесла 4 блок наверх, что бы он не сработал сразу после присвоения переменной, и во втором блоке добавила еще одну переменную.

[ OFFTOP ]
Neverwinter Script Source
void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
  object oNpc;
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak;

  if( GetIsPC( oPC ) == TRUE )
  {
/* ЧЕТВЕРТЫЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IS" ) == 1 ) // Проверяю вторую переменную
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Сам ты тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IS", 2 );
        SendMessageToPC( oPC, "Сработал четвертый блок" );
    }
  /* ВТОРОЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" )== 1 )
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Тухлый банан будет следующий!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IS", 1 );  //Другая переменная
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 2 );
        SendMessageToPC( oPC, "Сработал второй блок" );

    }

  /* ТРЕТИЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 0 )
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Кто последний тот тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 1 );
        SendMessageToPC( oPC, "Сработал третий блок" );
    }

      SendMessageToPC( oPC, "Сработал ОСНОВНОЙ блок" );
      AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
      AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
      AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери

  }

}


Срабатывают все три блока по очереди, но по одному разу. Все попытки сделать так, что бы скрипт еще и работал постоянно на одних переменных не получается.
Aiwan
Neverwinter Script Source

void main( )
{
  object oTrigger = OBJECT_SELF; // То на чем стоит скрипт
  object oDoor = GetObjectByTag( "DOOR_LIRA_01" ); // Наша дверь
  object oNpc;
  object oPC = GetEnteringObject( ); // Тот кто НАСТУПИЛ на триггер
  string sSpeak;

  if( GetIsPC( oPC ) == TRUE )
  {

  /* ВТОРОЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" )== 1 && GetLocalInt( oTrigger, "WAIT" )==FALSE )
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Тухлый банан будет следующий!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 2 );
        SendMessageToPC( oPC, "Сработал второй блок" );
        SetLocalInt( oTrigger, "WAIT", TRUE );
    }

  /* ТРЕТИЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 0  && GetLocalInt( oTrigger, "WAIT" )==0 )
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Кто последний тот тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 1 );
        SendMessageToPC( oPC, "Сработал третий блок" );
        SetLocalInt( oTrigger, "WAIT", 1 );
    }

    /* ЧЕТВЕРТЫЙ ВНУТРЕННИЙ БЛОК */

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 2  && !GetLocalInt( oTrigger, "WAIT" ))
    {

        oNpc = GetNearestObjectByTag( "NPC_LIRA" );
        sSpeak = "Сам ты тухлый банан!";
        SetLocalInt( oTrigger, "LIRA_PC_DO_IT", 0 );
        SendMessageToPC( oPC, "Сработал четвертый блок" );
        SetLocalInt( oTrigger, "WAIT", TRUE );
    }

      SetLocalInt( oTrigger, "WAIT", FALSE ); // SetLocalInt( oTrigger, "WAIT", 0 );
      SendMessageToPC( oPC, "Сработал ОСНОВНОЙ блок" );
      AssignCommand( oNpc, ClearAllActions( ) ); // Чистим стек команд
      AssignCommand( oNpc, SpeakString( sSpeak ) ); // Говорит фразу
      AssignCommand( oNpc, ActionForceMoveToObject( oDoor, TRUE ) ); // Бежит к двери

  }

}


В каждом блоке надо сделать проверку двух переменных сразу вот так как на скрипте:

Neverwinter Script Source

if ( GetLocalInt( oTrigger, "LIRA_PC_DO_IT" ) == 0  && GetLocalInt( oTrigger, "WAIT" )==0 )


Если есть одна переменная и (&&) нет воторй ТО. Сразу же в сработавшем блоке ставим еще ту переменную WAIT какую мы проверяем для одного раза. Теперь любой другой блок не сработает так как мы уже пометили что блок выполнен первый. Дальше скрипт идет вниз и после всех выполнений он присваивает локалку 0. Что бы наступив в следующи раз выполнился блок с переменной 1.

Neverwinter Script Source
SetLocalInt( oTrigger, "WAIT", FALSE ); // SetLocalInt( oTrigger, "WAIT", 0 );


Понятно? К тому же, я специально написал тебе три разных варианта проверки переменных. Спрашивай smile.gif
Artlira
Цитата
Понятно? К тому же, я специально написал тебе три разных варианта проверки переменных. Спрашивай


Ой… Понятно, не догадалась проверить сразу две переменные… Надо было не над скриптом издеваться, а прочесть еще раз тему о переменных…

Двумя переменными сразу действительно легко создать такое условие, что бы оно не сработало ни на одном из блоков. Переменная WAIT при работе скрипта всегда становится равной 1? Такого условия нет ни для одного блока, и они не срабатывают, а “обнуляется” она уже тогда, когда все блоки пройдены и скрипт об нее уже не спотыкается? А при следующей работе – она уже равна 0 и не мешает работать тому из блоков, который должен сработать по условию переменной LIRA_PC_DO_IT?
Это текстовая версия — только основной контент. Для просмотра полной версии этой страницы, пожалуйста, нажмите сюда.
Invision Power Board © 2001-2018 Invision Power Services, Inc.