Автор: Кирилл Коваленко, Андрей Коваленко
Разместил: Rhainer
Дата размещения: 08.02.2008
Источник: http://flashzone.ru/s/flash-training/delaem_igru/
В ролике, который мы предлагаем вашему вниманию, практически вся анимация будет реализована с помощью Action Script. Начиная с версии 4, во Flash включен достаточно мощный язык сценариев (скриптов), по синтаксису напоминающий C. Если вы знакомы с этим замечательным языком — это здорово поможет вам в освоении Flash Action Script.
Мы будем делать ролик-игру. Причем игру для двоих человек. Все очень просто — внизу ездит пушка и стреляет, скажем, по летающей тарелке, а тарелка, как вы уже догадались, стреляет по пушке. Такой, знаете ли, симулятор звездных войн.
Мы не будем подробно описывать создание рисунков (символов), а постараемся больше внимания уделить Action Script’ам (языку сценариев).
Итак, Flash 5 (mx) загрузилась на вашем компьютере. Создаем новый проект (File > New), сохраняем его с каким-нибудь именем (например, «StarWars») и настраиваем параметры ролика (Modify > Movie). Здесь надо только изменить количество кадров в секунду — сделаем его равным 60. Затем настраиваем параметры публикации (File > Publish Settings) — убираем галочку с HTML и ставим галочку на Windows Projector. Это значит, что наша программа будет публиковаться отображаться) не в HTML, а в виде исполняемого exe-файла. Это сделано для временного удобства — разумеется, истинная красота и мощь Flash (ну вот, обещал ведь без рекламы ) проявляется именно в возможности встраивания роликов в
web-странички. В конце статьи мы расскажем, как это сделать.
Теперь создаем символы, которые нам пригодятся в работе.
Создаем новый символ (Insert > New Symbol), называем его plate («тарелка»), кликаем OK и рисуем тарелку (летающую, с зелеными кровожадными человечками. Тарелка (как и все остальные символы) должна находиться приблизительно по центру окна (центр символа совпадает с крестиком). Особо не утруждайтесь, простой овал подойдет для начала, а наводить красоту можно и после написания самой программы. Чтобы увидеть все символы, находящиеся в ролике, откройте библиотеку символов (Window > Library).
Таким же образом создаем пушку (вытянутый по горизонтали прямоугольник), называем ее cannon. Дуло для пушки (назовите символ cannon_barrel) и для тарелки (plate_barrel) изобразите в виде вертикальных прямоугольников. Не забывайте: для каждого из этих игровых объектов мы создаем новый символ (Insert > New Symbol). Затем создайте снаряды, которыми будет стрелять тарелка и пушка (это тоже отдельные символы), и назовите их соответственно plate_shell и cannon_shell.
Нарисуйте их в виде маленьких прямоугольников или кругов. Для начала этого хватит.
После этого в основном ролике создайте 7 слоев — по одному для каждого символа и один для сценариев (Insert > Layer) — и переместите по одному символу в каждый слой (простым перетаскиванием символов из окна библиотеки), оставив верхний слой пустым. Сделайте так, чтобы слои с тарелкой и пушкой находились выше слоев со стволами, а те, в свою очередь, поверх снарядов (простым перетаскиванием в окне со слоями). Это нужно для того, чтобы дуло орудия было скрыто корпусом машины, а не наоборот.
Символы, находящиеся в основном ролике, называются экземплярами (instance). В одном ролике может быть несколько экземпляров одного и того же символа. Этим экземплярам нужно дать имена (поле Name на закладке Instance на вспомогательной панели; по умолчанию эта закладка находится слева на нижней панели). Для удобства имена экземплярам дайте те же, что и самим символам.
Теперь перейдем собственно к Action Script’ам. Они бывают двух типов — сценарии символа и сценарии кадра. Сценарии символа ориентированы на внешние команды, поступающие от пользователя (нажатие клавиши, перемещение мышки и т.д.), сценарии кадра могут исполняться и сами по себе. В нашей программе будут использоваться и те, и другие. Сама программа будет построена с помощью трехкадрового цикла — одного из самых распространенных приемов написания сценариев во Flash. В первом кадре устанавливаются начальные значения переменных, второй кадр — тело цикла, основная часть ролика. В третьем кадре происходит переход на второй кадр с помощью функции gotoAndPlay() или gotoAndStop().
Итак, начнем. В каждом слое должно быть по три кадра, причем только первый из них — ключевой (keyframe), остальные два обычные. Чтобы создать новый ключевой кадр (или превратить простой кадр в ключевой), надо щелкнуть на нем правой кнопкой и из выпадающего меню выбрать команду Insert Keyframe. Однако в слое со сценариями все три кадра должны быть ключевыми, так как писать сценарии можно только в ключевых кадрах. Кстати, чтобы не путаться, слоям тоже можно дать имена, соответствующие содержащимся в них экземплярам символа (слою со сценариями можно дать имя action). Для начала напишем сценарий, благодаря которому вся наша техника будет устанавливаться в начальное положение — например, по центру экрана (тарелка — вверху, пушка — внизу). Открываем окно сценариев (Window > Actions) и щелкаем на первом кадре. Обратите внимание — сверху появившегося окна написано Frame Actions, то есть сейчас
мы пишем сценарий кадра. Для удобства перейдите в режим эксперта (Expert Mode), нажав Ctrl+E или с помощью стрелки в верхнем правом углу.
Итак,пишем:
plate._x=275; plate_barrel._x=275; plate_shell._x=600; cannon._x=275; cannon_barrel._x=275; cannon_shell._x=600;
Затем щелкаем на третьем кадре и в сценарии третьего кадра пишем:gotoAndPlay(2); Давайте разберемся. _x — это свойство экземпляра, указывающее на горизонтальную координату экземпляра символа. По умолчанию ширина ролика равна 550, и поэтому все экземпляры (кроме снарядов) будут находиться как раз посередине. Нам пока не надо, чтобы снаряды были видны. При выполнении функции gotoAndPlay(2) происходит переход на второй кадр без остановки (то есть воспроизведение продолжается). Теперь сохраняем, нажимаем F12 (клавиша публикации), наслаждаемся, закрываем окно и думаем дальше. Придется подобрать вертикальные координаты экземпляров. Можно, конечно, сразу при расстановке
экземпляров расставить их вручную, но это не наш метод! Итак, записываем
y-координаты:
color=#0000ff> plate._y=60; plate_barrel._y=70; cannon._y=350; cannon_barrel._y=330;
У вас они могут немного отличаться (в зависимости от размеров символов). Только оставьте вверху и внизу немного места, оно понадобится нам в дальнейшем (там мы будем отображать военные успехи наших механических бойцов). А как, собственно говоря, будет происходить движение? А очень просто! Мы будем всего лишь менять x-координату (а для снарядов еще и y-координату) экземпляров во втором кадре. Но перед этим надо написать сценарий на событие нажатия клавиш и . Это мы будем делать уже в сценарии символа. Итак, нажмем на экземпляр тарелки, а потом снова на окно Action. Обратите внимание, теперь в заголовке окна должно быть написано Object Action, то есть мы пишем сценарий символа. Перейдите в режим эксперта (Ctrl+E или с помощью стрелки в верхнем правом углу окна) и напишите следующий код:
onClipEvent (keyDown) { if (key.isdown(key.right)){ plate_flag=true; } if (key.isdown(key.left)){ plate_flag=false; } }
onClipEvent — ключевое слово, означающее событие. isDown — это метод встроенного объекта key, который возвращает значение true (истина), если нажата указанная клавиша. Теперь при нажатии любой клавиши будет выполняться код, написанный во внешних скобках, в котором проверяется, нажата ли клавиша или , а переменной plate_flag присваивается соответствующее значение. Обратите внимание, что переменная plate_flag нигде до этого не определялась. Это одно из отличий Flash Action Script от языка C. Еще один важный момент: если вы объявляете некую переменную (свойство) в сценарии объекта (Object Action), то при обращении к этому свойству из сценария кадра (Frame Action) необходимо перед именем переменной (перед свойством) указать имя объекта с точкой. В нашем случае, объявив свойство plate_flag в сценарии объекта plate, мы будем обращаться к нему из сценария кадра вот так: plate.plate_flag (по аналогии со структурами в C).
Почти такой же код будет и для экземпляра пушки:
onClipEvent (keyDown){ if (key.isDown(ord(“D”))){ cannon_flag = true; } if (key.isDown(ord(“A”))){ cannon_flag = false; } }
Для пушки клавиша A будет означать движение влево, а клавиша D — вправо. Так что при переходе на нашу гениальную игру с Counter-Strike долго переучиваться не придется . Теперь во втором кадре перейдем к сценарию кадра. Напишем следующий текст:
if (plate.plate_flag == true) { plate._x = plate._x+3; } else { plate._x = plate._x–3; }
Этот код проверяет переменную plate_flag и взависимости от ее значения передвигает тарелку на 3 пикселя влево или вправо.
Опять сохраняемся, жмем F12, нажимаем по очереди клавиши и , а затем возвращаемся к раздумьям. Тарелка-то двигается, но вот ее дуло остается на месте. Однако это запросто исправляется. Во втором кадре пишем:plate_barrel._x = plate._x;Заодно напишем код и для пушки:if (cannon.cannon_flag == true) { cannon._x = cannon._x+3;} else { cannon._x = cannon._x–3;}cannon_barrel._x = cannon._x; Опять смотрим. Все хорошо, вот только тарелка с пушкой плавно уплывают куда-то вбок и исчезают из поля нашего тактического зрения. Надо, чтобы они появлялись с другой
стороны. Сказано — сделано. Пишем следующий код (в сценарии второго кадра):
if (plate._x<10) plate._x=560; if (plate._x>560) plate._x=–10; if (cannon._x<–10) cannon._x=560; if (cannon._x>560) cannon._x= — 10;
То есть если тарелка «почти зашла» за левый край, то она появляется на правом крае, и наоборот. То же самое происходит и с пушкой. Кстати, советую код для тарелки и для пушки отделять комментарием (любая строка после //, как в C++). И вообще, пишите больше комментариев, потом это поможет не запутаться.
Теперь разберемся с орудийными стволами нашей техники. Пусть для тарелки клавишами поворота ствола орудия будут и , а для пушки — W и S.
Итак, код для экземпляра ствола тарелки:
onClipEvent (keyDown) { if (key.isDown(key.UP)) { _rotation = _rotation+3; } if (key.isDown(key.Down)) { _rotation = _rotation–3; } if (_rotation<–90) _rotation=–90; if (_rotation>90) _rotation=90;}
_rotation — свойство экземпляра, определяющее угол поворота (в градусах). Сначала мы изменяем значение _rotation в зависимости от того, какая клавиша нажата, а потом проверяем, не повернулось ли дуло на угол, больший 90 градусов (ведь оно не может вращаться на все 360 градусов).
Почти такой же код будет и для дула пушки:
onClipEvent (keyDown) { if (key.isDown(ord(“W”))) { _rotation = _rotation+3; } if (key.isDown(ord(“S”))) { _rotation = _rotation–3; } if (_rotation<–90) _rotation = –90; if (_rotation>90) _rotation = 90;}
В сценарии второго кадра напишем следующий код:
plate_shell._rotation=plate_barrel._rotation; cannon_shell._rotation=cannon_barrel._rotation;
Теперь угол поворота снарядов будет равен углу поворота стволов.
Перейдем теперь к выстрелам. Все будет происходить так: при нажатии клавиши «огонь» координаты снаряда становятся равными координатам тарелки или пушки, а затем начинают двигаться с учетом угла поворота ствола.
Итак, код для снаряда тарелки (пишем в сценарии экземпляра снаряда тарелки plate_shell):
onClipEvent (keyDown) { if (key.isDown(key.DELETEKEY)) { flag = 1; } }
Переменная plate_shell.flag будет принимать три значения: 0 — выстрела нет (снаряд не летит и на экране невидим; реализуется это установлением координат снаряда числом, большим размеров видимого окна), 1 — был произведен выстрел (приравниваются координаты снаряда и ствола), 2 — снаряд в полете.
В сценарии первого кадра обнулите значение переменной plate_shell.flag:
plate_shell.flag =0;
Далее, в сценарии второго кадра допишите :
if (plate_shell.flag == 1) { plate_shell._x = plate._x; plate_shell._y = plate._y; // Координаты снаряда = координатам орудия plate_shell.flag =2; }
То есть, если была нажата клавиша Delete, то координаты снаряда становятся равными координатам тарелки, и переменной plate_shell.flag присваивается значение 2 — снаряд полетел громить вражеское орудие.
А сейчас нам понадобятся знания, почерпнутые из школьного курса геометрии. Итак, нам известен угол поворота снаряда и его скорость (допустим, 40 пикселей за кадр). Требуется найти смещение по x- и y-координатам.
Напоминаем: отношение –x/s равно sin(A) (отношение противолежащего катета к гипотенузе), отсюда x=–s*sin(A). Отношение y/s равно cos(A), отсюда y=s*cos(A).
В языке сценариев Flash 5 существует объект Math, предназначенный для математических вычислений. Нам потребуются два метода этого объекта: метод Math.cos() и Math.sin(), вычисляющие косинус и синус указанного в скобках параметра. Параметр должен быть указан в радианах. Но мы работали только с углами, заданными в градусах.
Специально чтобы обойти это препятствие хитрые математики придумали формулу, позволяющую переводить градусы в радианы: B = *A/180, где B — угол в радианах, а A — угол в градусах. Если взять скорость снаряда, равную 40, то x = –40*sin(*A/180), y = 40*cos(*A/180). Константа задается свойством pi объекта Math: Math.pi.
Чтобы не писать очень длинные строки, будем записывать значения синусов и косинусов в отдельные переменные, и уже затем писать код смещения. Пусть значение выражения sin(*A/180) хранится в переменной PS, а cos(*A/180) — в переменной PC.
Итак, в сценарии второго кадра пишем:
if (plate_shell.flag == 2) { PS = Math.sin(Math.pi*Plate_shell._rotation/180); PC = Math.cos(Math.pi*Plate_shell._rotation/180); Plate_shell._x = Plate_shell._x–40*PS; Plate_shell._y = Plate_shell._y+40*PC; }
То есть, если plate_shell.flag равен 2, мы вычисляем синус и косинус угла поворота, а затем происходит смещение.
Настало время посмотреть, что у нас получилось.
Нажмите F12 и оцените творение своих рук. Все замечательно, но обращают на себя внимание две досадные мелочи: во-первых, снаряд, вылетев за левый край, так и улетает в неизвестность, а во-вторых, если быстро нажимать на клавишу выстрела, снаряд возвращается на исходную для стрельбы позицию и опять начинает полет. С первой трудностью справиться легко — как и в случае с тарелкой, напишем в сценарии второго кадра следующие строки:
if (plate_shell._x < –10) { plate_shell._x = 560; } if (plate_shell._x > 560) { plate_shell._x = –10; }
А в отношении второй поступим так: когда произойдет выстрел, включится какой-то счетчик, и пока он не будет больше какого-то числа (например, 60), выстрелить опять будет нельзя (пока зарядятся импульсные энерго-генераторы, пока то-се …) Когда значение счетчика будет больше 60, движение снаряда прекратится, и он займет исходную позицию за границей видимого окна (т.е. «спрячется» — до следующего выстрела).
Итак, замените блок, который начинается со строки
if (plate_shell.flag == 2)…
на следующий:
if ((plate_shell.flag == 2) || (plate_shell.flag == 1)) { plate_shell_counter = plate_shell_counter+1; if (plate_shell_counter<60) { PS = Math.sin(Math.pi*Plate_shell._rotation/180); PC = Math.cos(Math.pi*Plate_shell._rotation/180); Plate_shell._x = Plate_shell._x–40*PS; Plate_shell._y = Plate_shell._y+40*PC; } else { plate_shell.flag = 0; plate_shell_counter = 0; plate_shell._x = 600; } }
То есть теперь в этом блоке еще увеличивается счетчик, а затем, в зависимости от его значения, либо происходит смещение, либо обнуляются счетчик и флаг снаряда, а сам снаряд сдвигается за область экрана. Кстати, в сценарии первого кадра обнулите счетчик
color=#0000ff> plate_shell_counter:plate_shell_counter = 0;
Кроме того, расширилось условие — флаг plate_shell.flag может быть равен как двойке, так и единице (двойным символом | обозначается булево (логическое) ИЛИ). Дело в том, что если мы нажмем клавишу выстрела, а счетчик еще не зашкалит за 100, то значение plate_shell.flag станет равным единице и без дополнительного условия смещение снаряда выполняться не будет.
Далее, строку
if (plate_shell.flag == 1) {
замените на строку
if ((plate_shell.flag == 1) & (plate_shell_counter == 0)) {
Здесь мы, наоборот, сужаем условие. Ведь выстрелить заново можно, только если движения снаряда уже нет (счетчик plate_shell_counter обнулен).
Далее, если во время полета снаряда изменять угол наклона пушки, то можно заметить, что траектория полета снаряда тоже будет меняться. Конечно, пришельцы в техническом плане очень продвинуты, но будем считать, что управлять каким-нибудь там плазменным снарядом им не под силу. Поэтому угол наклона траектории снаряда будет меняться только тогда, когда снаряд не движется (счетчик plate_shell_counter обнулен).Дляэтого строку
plate_shell._rotation = plate_barrel._rotation;
замените на следующий блок:
if (plate_shell_counter == 0) { plate_shell._rotation = plate_barrel._rotation; }
Теперь вроде бы все. Почти все. Осталось написать то же самое для пушки и придумать, каким образом будет происходить попадание.
Для пушки код сценария будет почти таким же — только немного изменятся формулы, по которым будет вычисляться смещение снаряда.
Итак, в сценарии снаряда для пушки (cannon_shell) пишем:
onClipEvent (keyDown) { if (key.isDown(ord(“Q”))) { flag = 1; } }
Тут все ясно — если нажата клавиша Q, то переменной cannon_shell.flag присваивается значение 1.
Далее в первом кадре обнуляем флаг:
cannon_shell.flag =0;
Объявляем и обнуляем счетчик для снаряда пушки (тоже в первом кадре):
cannon_shell_counter = 0;
Теперь во втором кадре заменяем строку
cannon_shell._rotation = cannon_barrel._rotation;
на блок
if (cannon_shell_counter == 0) { cannon_shell._rotation = cannon_barrel._rotation; }
Далее пишем во втором кадре:
if ((cannon_shell.flag == 1) & (cannon_shell_counter == 0)) { cannon_shell._x = cannon._x; cannon_shell._y = cannon._y; cannon_shell.flag = 2; } if ((cannon_shell.flag == 2) || (cannon_shell.flag == 1)) { cannon_shell_counter = cannon_shell_counter+1; if (cannon_shell_counter<60) { CS = Math.sin(Math.pi*cannon_shell._rotation/180); CC = Math.cos(Math.pi*cannon_shell._rotation/180); cannon_shell._x = cannon_shell._x+40*CS; cannon_shell._y = cannon_shell._y–40*CC; } else { cannon_shell.flag = 0; cannon_shell_counter = 0; cannon_shell._x = 600; } } if (cannon_shell._x<–10) { cannon_shell._x = 560; } if (cannon_shell._x>560) { cannon_shell._x = –10; }
То есть все делаем так же, как для случая с тарелкой, но только заменяем слово «plate» на слово «cannon» и немного изменяем формулы (переменная CS вместо PS, СС вместо PC, также меняем местами знаки + и –, т.к. пушка стреляет вверх, а тарелка — вниз).
Вот теперь можно и протестировать. Нажимаем F12 и проверяем, все ли работает так, как мы задумали. Вроде, все нормально. Тогда переходим к самой главной (но не самой сложной!) части.
Как же будут происходить попадания? И как будет реагировать на это наша техника? А примерно так: во время полета снаряда будет проверяться расстояние от центра вражеской машины. При достижении определенного значения будет считаться, что снаряд попал. Играть до первого попадания неинтересно, поэтому будем играть до пяти. Количество оставшихся жизней будем отображать в виде вытянутого по горизонтали красного прямоугольника, который при каждом попадании будет уменьшаться.
Итак, создайте новый символ с именем hits в виде прямоугольника красного цвета. Его размеры сейчас не важны. Придется создать еще два слоя. Назовите их plate_hit и cannon_hit. Поместите в каждый из этих слоев по одному экземпляру символа hits и в поле Name закладки Instance дайте им имена: plate_hit и cannon_hit. Разместите их так, чтобы экземпляр plate_hit находился выше тарелки, а экземпляр cannon_hit — ниже пушки. Теперь подберем размеры этих экземпляров.
Во-первых, они должны находится по центру. Их длина должна быть равна длине ролика, т.е. 550 пикселей. Ширину можно взять 10 пикселей. Длина и ширина символа во Flash задается свойствами _height и _width.
Итак, в сценарии первого кадра пишем:
color=#0000ff> plate_hit._width=550; plate_hit._height=10; plate_hit._x=275; plate_hit._y=10; color=#0000ff> cannon_hit._width=550; cannon_hit._height=10; cannon_hit._x=275; cannon_hit._y=390;
Вертикальные координаты указаны приблизительно, у вас они могут быть немного другими. Теперь о попаданиях. Снаряд попал, если он находится в прямоугольнике, центр которого совпадает с центром тарелки (пушки). То есть, если размеры тарелки 40 пикселей в высоту и 70 — в длину, то условие попадания будет выглядеть так:
if ((math.abs(cannon_shell._y–plate._y)<20) & (math.abs(cannon_shell._x–plate._x)<35)) { plate_hit._width = plate_hit._width–110; plate_hit._x = plate_hit._x–55; cannon_shell._y = –100; }
То есть если снаряд попал, то «линию жизни» надо уменьшить на 110 пикселей, а затем сдвинуть ее на 55 пикселей влево (чтобы полоска оставалась у левого края).
Если размеры тарелки и пушки совпадают, то для снаряда тарелки код будет почти таким же (если нет, то сделайте их размеры одинаковыми):
if ((math.abs(plate_shell._y–cannon._y)<20) & (math.abs(plate_shell._x–cannon._x)<35)) { cannon_hit._width = cannon_hit._width–110; cannon_hit._x = cannon_hit._x–55; plate_shell._y = 500; }
Теперь, в третьем кадре после строки gotoAndPlay(2); пишем:
if (plate_hit._width == 0) { gotoAndPlay (10); } if (cannon_hit._width == 0) { gotoAndPlay (11); }
То есть, если длина одной из линий будет равна нулю, то произойдет переход на нужный кадр. Теперь создаем еще один слой. В этом слое десятый и одиннадцатый кадр делаем ключевыми (keyframe). В десятом кадре создаем надпись Humans win!. На вспомогательной панели Character увеличиваем размер шрифта. В сценарии этого кадра пишем stop(), чтобы остановить ролик. В одиннадцатом кадре пишем Aliens win!, увеличиваем размер шрифта, в сценарии, как и для десятого кадра, пишем stop(), сохраняем, запускаем. Вот в общем-то и все. Осталась работа по улучшению геймплея. Можно увеличить количество кадров в секунду, индивидуализировать скорости для тарелки и пушки, изменить скорости для
снарядов, изменить «живучесть», нарисовать красивый задний фон, скажем, красные плоскогорья Марса (придется создать еще один слой). Только внимательно следите за тем, чтобы снаряд не проскочил «сквозь» цель, то есть смещение снаряда не должно превышать размеров цели. Играйте и выигрывайте! Теперь о публикации. Как уже говорилось, основное назначение Flash — возможность размещения качественных анимированных роликов на веб-страницах. Так что зайдите снова в меню File > Publish Settings и поставьте галочку в checkbox’e HTML. После публикации кроме файла StarWars.swf появится еще и автоматически созданная HTML-страница StarWars.html, которая содержит ссылку на созданный нами Flash-ролик. Открываем этот файл, копируем код, заключенный между тэгами <OBJECT> и </OBJECT>, в код своей страницы, и вот — на ней уже появилась наша игра. Обратите внимание: по умолчанию предполагается, что ролик (swf-файл) «лежит» в той же папке, что и html-страница. Это же правило сохраняется для интернет-адресов. Если, допустим, ваша страница имеет адрес http://server.com/my/page.html, то swf-файл в данном случае будет иметь адрес http://server.com/my/StarWars.swf.