|
|
4 months ago | |
|---|---|---|
| .. | ||
| README.md | 4 months ago | |
| assignment.md | 4 months ago | |
README.md
ساخت یک بازی فضایی بخش ۳: افزودن حرکت
journey
title سفر انیمیشن بازی شما
section اصول حرکت
درک اصول حرکت: 3: Student
یادگیری بهروزرسانی مختصات: 4: Student
پیادهسازی حرکت پایه: 4: Student
section کنترلهای بازیکن
مدیریت رویدادهای صفحهکلید: 4: Student
جلوگیری از رفتارهای پیشفرض: 5: Student
ایجاد کنترلهای پاسخگو: 5: Student
section سیستمهای بازی
ساخت حلقه بازی: 5: Student
مدیریت چرخه عمر اشیاء: 5: Student
پیادهسازی الگوی انتشار/اشتراک: 5: Student
به بازیهای مورد علاقهتان فکر کنید – چیزی که آنها را جذاب میکند فقط گرافیکهای زیبا نیست، بلکه نحوه حرکت و واکنش همه چیز به عملهای شماست. همین حالا بازی فضایی شما مانند یک نقاشی زیباست، اما ما قصد داریم حرکتی به آن اضافه کنیم که آن را زنده کند.
وقتی مهندسان ناسا کامپیوتر راهنمای مأموریتهای آپولو را برنامهنویسی کردند، با چالشی مشابه روبرو بودند: چگونه میتوان یک سفینه فضایی را طوری پاسخگو کرد که به ورودی خلبان واکنش نشان دهد و در عین حال تصحیح خودکار مسیر را حفظ کند؟ اصولی که امروز یاد میگیریم همان مفاهیمی هستند که مدیریت حرکت تحت کنترل بازیکن در کنار رفتارهای خودکار سیستم را پوشش میدهند.
در این درس، یاد میگیرید چگونه سفینههای فضایی را در صفحه نمایش سر بخورانید، به فرمانهای بازیکن پاسخ دهید و الگوهای حرکت روان ایجاد کنید. همه چیز را به مفاهیم قابل مدیریت تقسیم میکنیم که به صورت طبیعی بر یکدیگر بنا میشوند.
در پایان، بازیکنان کشتی قهرمان خود را در صفحه پرواز خواهند داد در حالی که کشتیهای دشمن بالای سر گشتزنی میکنند. مهمتر از همه، اصول بنیادینی را درک خواهید کرد که سیستمهای حرکت بازی را قدرت میبخشند.
mindmap
root((انیمیشن بازی))
Movement Types
کنترل شده توسط بازیکن
حرکت خودکار
مبتنی بر فیزیک
مسیرهای اسکریپتی
Event Handling
ورودی کیبورد
رویدادهای موس
کنترلهای لمسی
جلوگیری پیشفرض
Game Loop
بهروزرسانی منطق
رندر فریم
پاک کردن بوم
کنترل نرخ فریم
Object Management
بهروزرسانی موقعیت
تشخیص برخورد
مدیریت چرخه عمر
پیگیری وضعیت
Communication
الگوی انتشار/اشتراک
تولیدکنندههای رویداد
ارسال پیام
اتصال سبک
آزمون پیش از درس
درک حرکت بازی
بازیها زمانی زنده میشوند که چیزها شروع به حرکت کنند، و اساساً دو روش برای این اتفاق وجود دارد:
- حرکت تحت کنترل بازیکن: وقتی یک کلید فشار میدهید یا روی ماوس کلیک میکنید، چیزی حرکت میکند. این ارتباط مستقیم شما با دنیای بازی است.
- حرکت خودکار: وقتی خود بازی تصمیم میگیرد چیزی را حرکت دهد – مانند کشتیهای دشمن که باید در صفحه گشت بزنند چه شما کاری انجام دهید و چه ندهید.
حرکت دادن اشیاء روی صفحه نمایش کامپیوتر سادهتر از چیزی است که فکر میکنید. آن مختصات x و y را که در کلاس ریاضی یاد گرفتید، به خاطر دارید؟ دقیقاً همان چیزی است که اینجا با آن کار میکنیم. وقتی گالیله در سال 1610 قمرهای مشتری را ردیابی میکرد، اساساً همین کار را میکرد – موقعیتها را در طول زمان ترسیم میکرد تا الگوهای حرکت را بفهمد.
حرکت دادن اشیاء روی صفحه نمایش شبیه ساخت یک انیمیشن کتابچهای است – باید این سه مرحله ساده را دنبال کنید:
flowchart LR
A["فریم N"] --> B["بهروزرسانی موقعیتها"]
B --> C["پاک کردن بوم"]
C --> D["کشیدن اشیاء"]
D --> E["فریم N+1"]
E --> F{ادامه؟}
F -->|بله| B
F -->|خیر| G["پایان بازی"]
subgraph "چرخه انیمیشن"
H["1. محاسبه موقعیتهای جدید"]
I["2. پاک کردن فریم قبلی"]
J["3. رسم فریم جدید"]
end
style B fill:#e1f5fe
style C fill:#ffebee
style D fill:#e8f5e8
- بروزرسانی موقعیت – تغییر جایی که شیء باید باشد (مثلاً ۵ پیکسل به سمت راست حرکت کند)
- پاک کردن فریم قدیمی – صفحه را پاک کنید تا ردهای ارواح را نبینید
- رسم فریم جدید – شیء خود را در جای جدید بگذارید
این کار را به اندازه کافی سریع انجام دهید، و بوم! حرکت روانی دارید که برای بازیکنان طبیعی به نظر میرسد.
این چیزی است که میتواند در کد به این صورت به نظر برسد:
// موقعیت قهرمان را تنظیم کنید
hero.x += 5;
// مستطیلی که قهرمان در آن قرار دارد را پاک کنید
ctx.clearRect(0, 0, canvas.width, canvas.height);
// پسزمینه بازی و قهرمان را دوباره رسم کنید
ctx.fillRect(0, 0, canvas.width, canvas.height);
ctx.fillStyle = "black";
ctx.drawImage(heroImg, hero.x, hero.y);
این کد چه کاری انجام میدهد:
- مختصات x قهرمان را ۵ پیکسل افزایش میدهد تا به صورت افقی حرکت کند
- کل بوم را پاک میکند تا فریم قبلی حذف شود
- بوم را با رنگ پسزمینه سیاه پر میکند
- تصویر قهرمان را در موقعیت جدیدش دوباره رسم میکند
✅ میتوانید دلیل احتمالیای را که چرا رسم مجدد قهرمان در فریمهای زیاد ممکن است باعث افزایش هزینه عملکرد شود، بیان کنید؟ درباره گزینههای جایگزین این الگو بخوانید.
مدیریت رویدادهای صفحه کلید
اینجا جایی است که ورودی بازیکن به عملکرد بازی متصل میشود. وقتی کسی کلید فاصله را برای شلیک لیزر فشار میدهد یا کلید جهت را برای جاخالی دادن از یک سیارک میزند، بازی شما باید آن ورودی را شناسایی و واکنش نشان دهد.
رویدادهای صفحه کلید در سطح پنجره رخ میدهند، یعنی کل پنجره مرورگر شما منتظر آن فشار کلیدها است. در مقابل، کلیکهای ماوس میتوانند به عناصر خاصی (مثلاً کلیک روی یک دکمه) گره خورده باشند. برای بازی فضایی ما، تمرکزمان روی کنترلهای صفحه کلید است زیرا این همان چیزی است که حس کلاسیک آرکید را میدهد.
این موضوع به من یادآوری میکند که چگونه اپراتورهای تلگراف در قرن ۱۸ مجبور بودند ورودی کد مورس را به پیامهای معنادار تبدیل کنند – ما نیز کاری مشابه انجام میدهیم، ترجمه فشار کلیدها به فرمانهای بازی.
برای مدیریت یک رویداد باید از متد addEventListener() پنجره استفاده کنید و دو پارامتر ورودی به آن بدهید. پارامتر اول نام رویداد است، مثلاً keyup. پارامتر دوم تابعی است که باید پس از رخ دادن رویداد فراخوانی شود.
مثالی در زیر آمده است:
window.addEventListener('keyup', (evt) => {
// evt.key = نمایش رشتهای کلید
if (evt.key === 'ArrowUp') {
// انجام کاری
}
});
شرح اتفاقاتی که اینجا میافتد:
- گوش میدهد به رویدادهای صفحه کلید در کل پنجره
- رویداد را میگیرد که اطلاعاتی درباره کلیدی که فشار داده شده در خود دارد
- بررسی میکند آیا کلید فشار داده شده کلید خاصی است (در این حالت، کلید پیکان بالا)
- کدی را اجرا میکند وقتی شرط برآورده شود
برای رویدادهای کلید دو خصوصیت روی رویداد وجود دارد که میتوانید ببینید کدام کلید فشار داده شده:
key- این یک رشته است که کلید فشار داده شده را نشان میدهد، مثلاً'ArrowUp'keyCode- این یک عدد است، مثلاً37که متناظر باArrowLeftاست
✅ تغییر رویدادهای کلیدی کاربردهایی فراتر از توسعه بازی دارد. شما به چه کاربردهایی برای این تکنیک فکر میکنید؟
sequenceDiagram
participant User
participant Browser
participant EventSystem
participant GameLogic
participant Hero
User->>Browser: کلید پیکان بالا را فشار میدهد
Browser->>EventSystem: رویداد keydown
EventSystem->>EventSystem: preventDefault()
EventSystem->>GameLogic: emit('KEY_EVENT_UP')
GameLogic->>Hero: hero.y -= ۵
Hero->>Hero: بهروزرسانی موقعیت
Note over Browser,GameLogic: جریان رویداد از رفتار پیشفرض مرورگر جلوگیری میکند
Note over GameLogic,Hero: الگوی pub/sub ارتباط تمیز را ممکن میسازد
کلیدهای خاص: نکتهای مهم!
برخی کلیدها رفتارهای پیشفرض مرورگر دارند که ممکن است با بازی شما تداخل پیدا کند. کلیدهای پیکان صفحه را پیمایش میکنند و کلید فاصله صفحه را به پایین میبرد – رفتارهایی که هنگام هدایت سفینه فضایی نمیخواهید.
ما میتوانیم این رفتارهای پیشفرض را جلوگیری کنیم و اجازه دهیم بازی ما خودش ورودی را مدیریت کند. این شبیه به کاری است که برنامهنویسان اولیه کامپیوتر میکردند تا وقفههای سیستم را برای ایجاد رفتارهای سفارشی نادیده بگیرند – ما این کار را در سطح مرورگر انجام میدهیم. این نحوه انجام آن است:
const onKeyDown = function (e) {
console.log(e.keyCode);
switch (e.keyCode) {
case 37:
case 39:
case 38:
case 40: // کلیدهای جهتنما
case 32:
e.preventDefault();
break; // فاصله
default:
break; // سایر کلیدها را مسدود نکنید
}
};
window.addEventListener('keydown', onKeyDown);
درک این کد جلوگیری:
- بررسی کلیدهای مشخصی که ممکن است باعث رفتار ناخواسته مرورگر شوند
- جلوگیری از عملکرد پیشفرض مرورگر برای کلیدهای پیکان و فاصله
- اجازه دادن به دیگر کلیدها برای عملکرد عادی
- استفاده از
e.preventDefault()برای متوقف کردن رفتار تعبیه شده مرورگر
🔄 بررسی آموزشی
درک مدیریت رویداد: پیش از رفتن به حرکت خودکار، اطمینان حاصل کنید که میتوانید:
- ✅ تفاوت رویدادهای
keydownوkeyupرا توضیح دهید - ✅ دلیل جلوگیری از رفتارهای پیشفرض مرورگر را بفهمید
- ✅ توضیح دهید که چگونه گوشکنندههای رویداد ورودی کاربر را به منطق بازی متصل میکنند
- ✅ تشخیص دهید کدام کلیدها میتوانند با کنترلهای بازی تداخل داشته باشند
آزمون سریع خودکار: اگر رفتار پیشفرض برای کلیدهای پیکان را متوقف نکنید، چه اتفاقی میافتد؟ پاسخ: مرورگر صفحه را پیمایش میکند و با حرکت بازی تداخل میکند
معماری سیستم رویداد: اکنون درک میکنید:
- گوش دادن در سطح پنجره: گرفتن رویدادها در سطح مرورگر
- خصوصیات شی رویداد: رشتههای
keyدر مقابل شمارههایkeyCode - جلوگیری از پیشفرض: متوقف کردن رفتارهای ناخواسته مرورگر
- منطق شرطی: واکنش به ترکیب کلیدهای خاص
حرکت القاشده توسط بازی
حال بیایید درباره اشیایی صحبت کنیم که بدون ورودی بازیکن حرکت میکنند. فکر کنید به کشتیهای دشمن که در صفحه گشت میزنند، گلولههایی که در خطوط مستقیم پرواز میکنند، یا ابرهایی که در پسزمینه شناورند. این حرکت خودکار باعث میشود جهان بازی حتی وقتی کسی کنترل را لمس نمیکند، زنده به نظر برسد.
ما از تایمرهای داخلی جاوااسکریپت استفاده میکنیم تا موقعیتها را در فواصل منظم بهروزرسانی کنیم. این مفهوم شبیه به عملکرد ساعتهای پاندولدار است – مکانیزمی منظم که اقدامات زمانبندی شده ثابت را فعال میکند. اینقدر ساده است:
const id = setInterval(() => {
// دشمن را در محور y جابجا کنید
enemy.y += 10;
}, 100);
این کد حرکت چه میکند:
- ایجاد تایمری که هر ۱۰۰ میلیثانیه اجرا میشود
- بروزرسانی مختصات y دشمن هر بار ۱۰ پیکسل
- ذخیره شناسه بازه زمانی تا در صورت نیاز بتوانیم آن را متوقف کنیم
- حرکت دادن دشمن به صورت خودکار به سمت پایین صفحه
حلقه بازی
اینجا مفهومی است که همه چیز را به هم پیوند میدهد – حلقه بازی. اگر بازی شما یک فیلم بود، حلقه بازی همان پروژکتور فیلم بود که فریم به فریم را آنقدر سریع نشان میدهد که همه چیز به صورت روان به نظر برسد.
هر بازی یکی از این حلقهها را پشت صحنه اجرا میکند. این یک تابع است که همه اشیای بازی را بهروزرسانی میکند، صفحه را دوباره رسم میکند و این فرایند را به طور مداوم تکرار میکند. این تابع وضعیت قهرمان شما، همه دشمنها، هر لیزری را که پرواز میکند – کل وضعیت بازی را مدیریت میکند.
این مفهوم من را یاد انیماتورهای اولیه فیلم مانند والت دیزنی میاندازد که مجبور بودند شخصیتها را فریم به فریم بازنقش کنند تا توهم حرکت را ایجاد کنند. ما همین کار را انجام میدهیم، فقط با کد به جای مداد.
یک حلقه بازی معمولی به این شکل میتواند باشد، به صورت کد بیان شده:
flowchart TD
A["شروع حلقه بازی"] --> B["پاک کردن بوم"]
B --> C["پر کردن پسزمینه"]
C --> D["بهروزرسانی اشیاء بازی"]
D --> E["رسم قهرمان"]
E --> F["رسم دشمنان"]
F --> G["رسم عناصر رابط کاربری"]
G --> H["انتظار برای فریم بعدی"]
H --> I{بازی در حال اجرا هست؟}
I -->|بله| B
I -->|خیر| J["پایان بازی"]
subgraph "کنترل نرخ فریم"
K["۶۰ FPS = ۱۶.۶۷ میلیثانیه"]
L["۳۰ FPS = ۳۳.۳۳ میلیثانیه"]
M["۱۰ FPS = ۱۰۰ میلیثانیه"]
end
style B fill:#ffebee
style D fill:#e1f5fe
style E fill:#e8f5e8
style F fill:#e8f5e8
const gameLoopId = setInterval(() => {
function gameLoop() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.fillStyle = "black";
ctx.fillRect(0, 0, canvas.width, canvas.height);
drawHero();
drawEnemies();
drawStaticObjects();
}
gameLoop();
}, 200);
درک ساختار حلقه بازی:
- پاک کردن کل بوم برای حذف فریم قبلی
- پرکردن پسزمینه با یک رنگ ثابت
- رسم کردن همه اشیای بازی در موقعیتهای فعلیشان
- تکرار این فرایند هر ۲۰۰ میلیثانیه برای ایجاد انیمیشن روان
- مدیریت نرخ فریم با کنترل زمانبندی بازه
ادامه بازی فضایی
اکنون حرکت را به صحنه ساکن که قبلاً ساختهاید اضافه میکنیم. قرار است آن را از یک عکس ثابت به تجربهای تعاملی تبدیل کنیم. گام به گام پیش میرویم تا هر قسمت به صورت پیوسته بر قسمت قبل بنا شود.
کد را از جایی که در درس قبلی متوقف شده بودید بردارید (یا اگر نیاز به شروع تازه دارید، با کد داخل پوشه Part II- starter شروع کنید).
امروز این موارد را میسازیم:
- کنترلهای قهرمان: کلیدهای پیکان کشتی فضایی شما را در صفحه هدایت میکنند
- حرکت دشمن: آن کشتیهای بیگانه شروع به پیشروی میکنند
بیایید شروع به پیادهسازی این ویژگیها کنیم.
مراحل پیشنهادی
فایلهایی که برای شما در زیرپوشه your-work ایجاد شدهاند را بیابید. باید شامل موارد زیر باشند:
-| assets
-| enemyShip.png
-| player.png
-| index.html
-| app.js
-| package.json
شما پروژهتان را در پوشه your-work با تایپ دستور زیر شروع میکنید:
cd your-work
npm start
این دستور چه کاری انجام میدهد:
- رفتن به دایرکتوری پروژه شما
- راهاندازی یک سرور HTTP در آدرس
http://localhost:5000 - خدمترسانی فایلهای بازی شما تا بتوانید آنها را در مرورگر تست کنید
طبق بالا یک سرور HTTP در آدرس http://localhost:5000 راهاندازی خواهد شد. یک مرورگر باز کنید و آن آدرس را وارد کنید، وضعیت فعلی باید قهرمان و همه دشمنان را نشان دهد؛ اما هنوز هیچ حرکتی ندارید!
افزودن کد
-
اضافه کردن اشیای اختصاصی برای
heroوenemyوgame objectکه باید ویژگیهایxوyداشته باشند. (بخشی که درباره ارثبری یا ترکیب بود را به خاطر داشته باشید).راهنمایی
game objectباید همان کسی باشد کهxوyو توانایی رسم خود روی بوم را دارد.نکته: با افزودن کلاس جدید
GameObjectبا سازنده تعریف شده به شکل زیر شروع کنید و سپس آن را روی بوم رسم کنید:class GameObject { constructor(x, y) { this.x = x; this.y = y; this.dead = false; this.type = ""; this.width = 0; this.height = 0; this.img = undefined; } draw(ctx) { ctx.drawImage(this.img, this.x, this.y, this.width, this.height); } }درک این کلاس پایه:
- تعریف ویژگیهای مشترکی که همه اشیای بازی دارند (مکان، اندازه، تصویر)
- شامل یک پرچم
deadبرای پیگیری اینکه آیا شیء باید حذف شود - ارائه متد
draw()که شیء را روی بوم رسم میکند - تنظیم مقادیر پیشفرض برای همه ویژگیها که کلاسهای فرزند میتوانند بازنویسی کنند
classDiagram
class GameObject {
+x: number
+y: number
+dead: boolean
+type: string
+width: number
+height: number
+img: Image
+draw(ctx)
}
class Hero {
+speed: number
+type: "قهرمان"
+width: ۹۸
+height: ۷۵
}
class Enemy {
+type: "دشمن"
+width: ۹۸
+height: ۵۰
+setInterval()
}
GameObject <|-- Hero
GameObject <|-- Enemy
class EventEmitter {
+listeners: object
+on(message, listener)
+emit(message, payload)
}
حالا این `GameObject` را گسترش دهید تا `Hero` و `Enemy` را ایجاد کنید:
```javascript
class Hero extends GameObject {
constructor(x, y) {
super(x, y);
this.width = 98;
this.height = 75;
this.type = "Hero";
this.speed = 5;
}
}
```
```javascript
class Enemy extends GameObject {
constructor(x, y) {
super(x, y);
this.width = 98;
this.height = 50;
this.type = "Enemy";
const id = setInterval(() => {
if (this.y < canvas.height - this.height) {
this.y += 5;
} else {
console.log('Stopped at', this.y);
clearInterval(id);
}
}, 300);
}
}
```
**مفاهیم کلیدی در این کلاسها:**
- **ارثبری** از `GameObject` با استفاده از کلمه کلیدی `extends`
- **فراخوانی** سازنده والد با `super(x, y)`
- **تنظیم** ابعاد و ویژگیهای خاص برای هر نوع شیء
- **پیادهسازی** حرکت خودکار برای دشمنها با استفاده از `setInterval()`
-
افزودن هندلرهای رویداد کلید برای مدیریت ناوبری کلید (حرکت قهرمان به بالا/پایین/چپ/راست)
یادتان باشد این یک سیستم کارتزین است، بالا-چپ مختصات
0,0است. همچنین یادتان باشد که کد جلوگیری از رفتار پیشفرض را اضافه کنید.نکته: تابع
onKeyDownخود را بسازید و به پنجره متصل کنید:const onKeyDown = function (e) { console.log(e.keyCode); // کد درس بالا را برای جلوگیری از رفتار پیشفرض اضافه کنید switch (e.keyCode) { case 37: case 39: case 38: case 40: // کلیدهای جهت case 32: e.preventDefault(); break; // فاصله default: break; // کلیدهای دیگر را مسدود نکنید } }; window.addEventListener("keydown", onKeyDown);این هندلر رویداد چه کار میکند:
- گوش دادن به رویدادهای keydown در کل پنجره
- ثبت کد کلید برای کمک به دیباگ اینکه کدام کلیدها فشرده شدهاند
- جلوگیری از رفتار پیشفرض مرورگر برای کلیدهای پیکان و فاصله
- اجازه دادن به کلیدهای دیگر برای عملکرد نرمال
کنسول مرورگرتان را در این مرحله بررسی کنید و کلیدهای زده شده را ببینید که ثبت میشوند.
-
اجرای الگوی Pub Sub (الگو را اینجا ببینید)، این کار کد شما را مرتب نگه میدارد همانطور که باقی بخشها را دنبال میکنید.
الگوی انتشار-اشتراک به شما کمک میکند کدتان را با جدا کردن تشخیص رویداد از مدیریت رویداد سازماندهی کنید. این کد شما را مدولارتر و نگهداری آن را آسانتر میکند.
برای انجام این بخش آخر، میتوانید:
-
افزودن یک گوشکننده رویداد به پنجره:
window.addEventListener("keyup", (evt) => { if (evt.key === "ArrowUp") { eventEmitter.emit(Messages.KEY_EVENT_UP); } else if (evt.key === "ArrowDown") { eventEmitter.emit(Messages.KEY_EVENT_DOWN); } else if (evt.key === "ArrowLeft") { eventEmitter.emit(Messages.KEY_EVENT_LEFT); } else if (evt.key === "ArrowRight") { eventEmitter.emit(Messages.KEY_EVENT_RIGHT); } });
این سیستم رویداد چه کاری انجام میدهد:
- شناسایی ورودی صفحه کلید و تبدیل آن به رویدادهای سفارشی بازی
- جدا کردن تشخیص ورودی از منطق بازی
- سهولت در تغییر کنترلها بعداً بدون تاثیر روی کد بازی
- اجازه دادن به سیستمهای متعدد برای پاسخ به همان ورودی
-
flowchart TD
A["ورودی صفحه کلید"] --> B["گوش دهنده رویداد پنجره"]
B --> C["انتشاردهنده رویداد"]
C --> D["رویداد کلید به بالا"]
C --> E["رویداد کلید به پایین"]
C --> F["رویداد کلید به چپ"]
C --> G["رویداد کلید به راست"]
D --> H["حرکت قهرمان"]
D --> I["سیستم صدا"]
D --> J["افکتهای بصری"]
E --> H
F --> H
G --> H
style A fill:#e1f5fe
style C fill:#e8f5e8
style H fill:#fff3e0
-
ساخت یک کلاس EventEmitter برای انتشار و اشتراک پیامها:
class EventEmitter { constructor() { this.listeners = {}; } on(message, listener) { if (!this.listeners[message]) { this.listeners[message] = []; } this.listeners[message].push(listener); } -
افزودن ثابتها و راهاندازی EventEmitter:
const Messages = { KEY_EVENT_UP: "KEY_EVENT_UP", KEY_EVENT_DOWN: "KEY_EVENT_DOWN", KEY_EVENT_LEFT: "KEY_EVENT_LEFT", KEY_EVENT_RIGHT: "KEY_EVENT_RIGHT", }; let heroImg, enemyImg, laserImg, canvas, ctx, gameObjects = [], hero, eventEmitter = new EventEmitter();
درک تنظیمات:
- تعریف ثابتهای پیام برای جلوگیری از اشتباهات تایپی و سهولت اصلاح کد
- اعلام متغیرها برای تصاویر، کانتکست بوم و وضعیت بازی
- ایجاد یک انتشاردهنده رویداد جهانی برای سیستم انتشار-اشتراک
- آرایهای را مقداردهی اولیه میکند تا همه اشیاء بازی را در خود نگه دارد
-
بازی را مقداردهی اولیه کنید
function initGame() { gameObjects = []; createEnemies(); createHero(); eventEmitter.on(Messages.KEY_EVENT_UP, () => { hero.y -= 5; }); eventEmitter.on(Messages.KEY_EVENT_DOWN, () => { hero.y += 5; }); eventEmitter.on(Messages.KEY_EVENT_LEFT, () => { hero.x -= 5; }); -
راهاندازی حلقه بازی
تابع
window.onloadرا بازنویسی کنید تا بازی را مقداردهی اولیه کند و یک حلقه بازی با فاصله زمانی مناسب راهاندازی کند. همچنین یک پرتو لیزر اضافه خواهید کرد:window.onload = async () => { canvas = document.getElementById("canvas"); ctx = canvas.getContext("2d"); heroImg = await loadTexture("assets/player.png"); enemyImg = await loadTexture("assets/enemyShip.png"); laserImg = await loadTexture("assets/laserRed.png"); initGame(); const gameLoopId = setInterval(() => { ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.fillStyle = "black"; ctx.fillRect(0, 0, canvas.width, canvas.height); drawGameObjects(ctx); }, 100); };درک راهاندازی بازی:
- منتظر میماند تا صفحه کاملاً بارگذاری شود قبل از شروع
- عنصر canvas و زمینه رندر دوبعدی آن را دریافت میکند
- تمام داراییهای تصویری را به صورت ناهمزمان و با استفاده از
awaitبارگذاری میکند - حلقه بازی را با فواصل 100 میلیثانیه (10 فریم بر ثانیه) آغاز میکند
- هر فریم کل صفحه را پاک کرده و دوباره رسم میکند
-
کدی اضافه کنید که دشمنان را در یک فاصله زمانی معین حرکت دهد
تابع
createEnemies()را بازنویسی کنید تا دشمنان را ایجاد کرده و آنها را به کلاس gameObjects جدید اضافه کند:function createEnemies() { const MONSTER_TOTAL = 5; const MONSTER_WIDTH = MONSTER_TOTAL * 98; const START_X = (canvas.width - MONSTER_WIDTH) / 2; const STOP_X = START_X + MONSTER_WIDTH; for (let x = START_X; x < STOP_X; x += 98) { for (let y = 0; y < 50 * 5; y += 50) { const enemy = new Enemy(x, y); enemy.img = enemyImg; gameObjects.push(enemy); } } }کارکرد ایجاد دشمنان:
- موقعیتها را محاسبه میکند تا دشمنان در مرکز صفحه قرار گیرند
- یک شبکه از دشمنان با استفاده از حلقههای تو در تو ایجاد میکند
- تصویر دشمن را به هر شیء دشمن اختصاص میدهد
- هر دشمن را به آرایه جهانی اشیاء بازی اضافه میکند
و تابع
createHero()را اضافه کنید تا فرآیندی مشابه برای قهرمان انجام دهد.function createHero() { hero = new Hero( canvas.width / 2 - 45, canvas.height - canvas.height / 4 ); hero.img = heroImg; gameObjects.push(hero); }کارکرد ایجاد قهرمان:
- قهرمان را در وسط پایین صفحه قرار میدهد
- تصویر قهرمان را به شیء قهرمان اختصاص میدهد
- قهرمان را برای رندر شدن به آرایه اشیاء بازی اضافه میکند
و در نهایت، تابع
drawGameObjects()را اضافه کنید تا رسم شروع شود:function drawGameObjects(ctx) { gameObjects.forEach(go => go.draw(ctx)); }درک تابع رسم:
- از روی تمام اشیاء بازی در آرایه عبور میکند
- متد
draw()را روی هر شی فراخوانی میکند - زمینه canvas را منتقل میکند تا اشیاء بتوانند خود را رندر کنند
🔄 بررسی آموزشی
درک کامل سیستم بازی: تسلط خود را بر کل معماری بررسی کنید:
- ✅ ارثبری چگونه به Hero و Enemy اجازه میدهد خصوصیات مشترک GameObject را بهاشتراک بگذارند؟
- ✅ چرا الگوی pub/sub نگهداری کد شما را آسانتر میکند؟
- ✅ حلقه بازی چه نقشی در ایجاد انیمیشن روان دارد؟
- ✅ شنوندههای رویداد چگونه ورودی کاربر را به رفتار اشیاء بازی متصل میکنند؟
ادغام سیستم: بازی شما اکنون نشان میدهد:
- طراحی شیءگرا: کلاسهای پایه با ارثبری تخصصی شده
- معماری رویدادمحور: الگوی pub/sub برای اتصال کم
- چارچوب انیمیشن: حلقه بازی با بهروزرسانی فریمهای پیدرپی
- مدیریت ورودی: رویدادهای صفحهکلید با جلوگیری از رفتار پیشفرض
- مدیریت دارایی: بارگذاری تصویر و رندر اسپریت
الگوهای حرفهای: شما پیادهسازی کردهاید:
- جداسازی نگرانیها: جدا بودن ورودی، منطق و رندر
- چندریختی: همه اشیاء بازی رابط رسم مشترک دارند
- ارسال پیامها: ارتباط پاک بین اجزاء
- مدیریت منابع: مدیریت کارآمد اسپریتها و انیمیشنها
دشمنانتان باید شروع به پیشروی روی سفینه قهرمان شما کنند! } }
and add a `createHero()` function to do a similar process for the hero. ```javascript function createHero() { hero = new Hero( canvas.width / 2 - 45, canvas.height - canvas.height / 4 ); hero.img = heroImg; gameObjects.push(hero); }و در نهایت، تابع
drawGameObjects()را اضافه کنید تا رسم شروع شود:function drawGameObjects(ctx) { gameObjects.forEach(go => go.draw(ctx)); }دشمنانتان باید شروع به پیشروی روی سفینه قهرمان شما کنند!
چالش GitHub Copilot Agent 🚀
اینجا چالشی است که کیفیت بازی شما را بهبود میبخشد: اضافه کردن مرزها و کنترلهای روان. در حال حاضر، قهرمان شما میتواند از صفحه خارج شود و حرکت ممکن است تکهتکه به نظر برسد.
ماموریت شما: سفینه فضایی خود را واقعیتر کنید با پیادهسازی مرزهای صفحه و حرکت نرم. این شبیه سیستمهای کنترل پرواز ناسا است که از عبور سفینهها از پارامترهای عملیاتی ایمن جلوگیری میکنند.
چه چیزی بسازید: سیستمی بسازید که سفینه قهرمان شما را در صفحه نگه دارد و کنترلها را روان کند. وقتی بازیکنان کلید پیکان را نگه میدارند، سفینه باید به طور مداوم لغزندگی کند به جای حرکتهای گسسته. فکر کنید که وقتی سفینه به مرزهای صفحه میرسد، بازخورد بصری اضافه کنید – شاید یک اثر ظریف برای نشان دادن لبه منطقه بازی.
برای کسب اطلاعات بیشتر درباره حالت agent اینجا را ببینید.
🚀 چالش
سازماندهی کد بهمرور زمان اهمیت بیشتری پیدا میکند، مخصوصاً وقتی پروژهها بزرگتر میشوند. ممکن است متوجه شده باشید فایل شما شلوغ از توابع، متغیرها و کلاسهای مخلوط است. این شما را یاد مهندسانی میاندازد که کد ماموریت آپولو را سازماندهی کردند و سیستمهایی واضح و قابل نگهداری ایجاد کردند تا چند تیم بتوانند همزمان روی آنها کار کنند.
ماموریت شما: مثل یک معمار نرمافزار فکر کنید. چگونه کد خود را سازماندهی میکنید که شش ماه دیگر شما یا همکارتان بتوانید آنچه اتفاق میافتد را بفهمید؟ حتی اگر همه چیز در یک فایل بماند، میتوانید سازماندهی بهتری داشته باشید:
- گروهبندی توابع مرتبط با سربرگهای کامنت واضح
- جداسازی نگرانیها - منطق بازی را از رندر جدا نگه دارید
- استفاده از نامگذاری سازگار برای متغیرها و توابع
- ایجاد ماژولها یا فضای نام برای سازماندهی بخشهای مختلف بازی
- افزودن مستندات که هدف هر بخش اصلی را توضیح دهد
سؤالات بازتابی:
- کدام بخشهای کد شما در بازگشت سختتر فهمیده میشود؟
- چگونه میتوانید کد خود را سازماندهی کنید تا برای دیگران آسانتر باشد مشارکت کنند؟
- اگر بخواهید ویژگیهای جدیدی مانند قدرتها یا نوعهای مختلف دشمن اضافه کنید چه اتفاقی میافتد؟
آزمون پس از کلاس
مروری و مطالعه خودخوان
ما همه چیز را از صفر ساختیم که عالی است برای یادگیری، اما این یک راز کوچک است – چندین فریمورک جاوااسکریپت عالی وجود دارد که میتوانند بخش زیادی از وظایف سنگین را برای شما انجام دهند. پس از اینکه با اصول پایه که پوشش دادیم راحت شدید، ارزش دارد که بررسی کنید چه چیزی در دسترس است.
فریمورکها مانند داشتن یک جعبهابزار پر شده از ابزار هستند به جای ساختن تکتک ابزارها با دست. آنها میتوانند بسیاری از چالشهای سازماندهی کد که صحبت کردیم را حل کنند و امکاناتی ارائه دهند که ساختن دستیشان هفتهها طول میکشد.
مواردی که ارزش بررسی دارند:
- چگونه موتورهای بازی کد را سازماندهی میکنند – الگوهای هوشمندانهای که استفاده میکنند شما را شگفتزده میکند
- ترفندهای عملکرد برای اجرای روان بازیهای canvas
- ویژگیهای مدرن جاوااسکریپت که کد شما را تمیزتر و قابل نگهداریتر میکند
- رویکردهای مختلف برای مدیریت اشیاء بازی و روابطشان
🎯 جدول زمانی تسلط بر انیمیشن بازی شما
timeline
title پیشرفت یادگیری انیمیشن و تعامل بازی
section اصول حرکت (۲۰ دقیقه)
اصول انیمیشن: انیمیشن مبتنی بر فریم
: بهروزرسانی موقعیت
: سیستمهای مختصات
: حرکت نرم
section سیستمهای رویداد (۲۵ دقیقه)
ورودی کاربر: مدیریت رویدادهای کیبورد
: جلوگیری از رفتار پیشفرض
: ویژگیهای شیء رویداد
: شنود سطح پنجره
section معماری بازی (۳۰ دقیقه)
طراحی شیء: الگوهای ارثبری
: ایجاد کلاس پایه
: رفتارهای تخصصی
: رابطهای چندریختی
section الگوهای ارتباطی (۳۵ دقیقه)
پیادهسازی انتشاردریافت: انتشاردهنده رویدادها
: ثابتهای پیام
: اتصال شل
: ادغام سیستم
section تسلط بر حلقه بازی (۴۰ دقیقه)
سیستمهای بلادرنگ: کنترل نرخ فریم
: چرخه بهروزرسانی/رندر
: مدیریت وضعیت
: بهینهسازی عملکرد
section تکنیکهای پیشرفته (۴۵ دقیقه)
ویژگیهای حرفهای: تشخیص برخورد
: شبیهسازی فیزیک
: ماشینهای حالت
: سیستمهای مؤلفه
section مفاهیم موتور بازی (۱ هفته)
درک چارچوب: سیستمهای موجودیت-مؤلفه
: نمودارهای صحنه
: خطوط تولید دارایی
: پروفایل عملکرد
section مهارتهای تولید (۱ ماه)
توسعه حرفهای: سازماندهی کد
: همکاری تیمی
: استراتژیهای تست
: بهینهسازی استقرار
🛠️ خلاصه ابزارهای توسعه بازی شما
پس از اتمام این درس، حالا در موارد زیر تسلط دارید:
- اصول انیمیشن: حرکت فریممحور و انتقالات نرم
- برنامهنویسی رویدادمحور: مدیریت ورودی صفحهکلید با مدیریت رویداد مناسب
- طراحی شیءگرا: سلسلهمراتب ارث و رابطهای چندریختی
- الگوهای ارتباطی: معماری pub/sub برای کد قابل نگهداری
- معماری حلقه بازی: چرخههای بروز رسانی و رندر در زمان واقعی
- سیستمهای ورودی: نگاشت کنترل کاربر با جلوگیری از رفتار پیشفرض
- مدیریت دارایی: بارگذاری اسپریت و تکنیکهای رندر بهینه
⚡ آنچه در ۵ دقیقه آینده میتوانید انجام دهید
- کنسول مرورگر را باز کنید و
addEventListener('keydown', console.log)را امتحان کنید تا رویدادهای صفحهکلید را ببینید - یک عنصر div ساده بسازید و با کلیدهای پیکان آن را حرکت دهید
- با
setIntervalآزمایش کنید برای ایجاد حرکت مداوم - تلاش کنید با
event.preventDefault()از رفتار پیشفرض جلوگیری کنید
🎯 آنچه در این ساعت میتوانید به انجام برسانید
- آزمون پس از درس را کامل کنید و برنامهنویسی رویدادمحور را بفهمید
- سفینه قهرمان متحرک با کنترل کامل صفحهکلید بسازید
- الگوهای حرکتی روان دشمنان را پیادهسازی کنید
- مرزهایی اضافه کنید تا اشیاء بازی از صفحه خارج نشوند
- تشخیص برخورد پایه بین اشیاء بازی را ایجاد کنید
📅 سفر انیمیشن یک هفتهای شما
- بازی فضایی کامل با حرکت و تعاملات صیقل داده شده را تکمیل کنید
- الگوهای حرکت پیشرفته مثل منحنیها، شتاب و فیزیک را اضافه کنید
- انتقالهای نرم و توابع تعدیلی را پیادهسازی کنید
- افکتهای ذرات و سیستمهای بازخورد بصری بسازید
- عملکرد بازی را برای اجرای روان ۶۰ فریم بر ثانیه بهینه کنید
- کنترلهای لمسی موبایل و طراحی واکنشگرا اضافه کنید
🌟 توسعه تعاملی یک ماهه شما
- برنامههای تعاملی پیچیده با سیستمهای انیمیشن پیشرفته بسازید
- کتابخانههای انیمیشن مثل GSAP را یاد بگیرید یا موتور انیمیشن خود را بسازید
- در پروژههای بازی و انیمیشن متنباز مشارکت کنید
- بهینهسازی عملکرد برای برنامههای گرافیکی سنگین را تسلط یابید
- محتوای آموزشی درباره توسعه بازی و انیمیشن ایجاد کنید
- پرتفوی خود را با مهارتهای پیشرفته برنامهنویسی تعاملی بسازید
کاربردهای واقعی: مهارتهای انیمیشن بازی شما به طور مستقیم به این حوزهها مرتبط است:
- برنامههای تعاملی وب: داشبوردهای پویا و رابطهای زمان واقعی
- بصریسازی دادهها: نمودارهای متحرک و گرافیکهای تعاملی
- نرمافزار آموزشی: شبیهسازیهای تعاملی و ابزارهای یادگیری
- توسعه موبایل: بازیهای لمسی و مدیریت ژستها
- برنامههای دسکتاپ: برنامههای Electron با انیمیشنهای روان
- انیمیشنهای وب: کتابخانههای انیمیشن CSS و جاوااسکریپت
مهارتهای حرفهای کسب شده: شما اکنون میتوانید:
- معماری سیستمهای رویدادمحور که با پیچیدگی مقیاسپذیرند طراحی کنید
- انیمیشنهای نرم را با استفاده از اصول ریاضی پیادهسازی کنید
- اشکالزدایی سیستمهای تعاملی پیچیده با ابزارهای توسعه مرورگر
- بهینهسازی عملکرد بازی برای دستگاهها و مرورگرهای مختلف
- طراحی ساختارهای کد قابل نگهداری با الگوهای اثبات شده
مفاهیم توسعه بازی مسلط شده:
- مدیریت نرخ فریم: درک FPS و کنترل زمانبندی
- مدیریت ورودی: سیستمهای صفحهکلید و رویداد متقابل پلتفرم
- چرخه عمر شیء: الگوهای ایجاد، بهروزرسانی و نابودسازی
- همگامسازی وضعیت: حفظ وضعیت بازی در فریمهای متعدد
- معماری رویداد: ارتباط بدون وابستگی بین سیستمهای بازی
مرحله بعد: آمادهاید که تشخیص برخورد، سیستمهای امتیازدهی، جلوههای صوتی اضافه کنید یا فریمورکهای بازی مدرن مثل Phaser یا Three.js را کاوش کنید!
🌟 دستاورد قفلگشاییشده: یک سیستم بازی تعاملی کامل با الگوهای معماری حرفهای ساختهاید!
تکلیف
توضیح مهم:
این سند با استفاده از سرویس ترجمه هوش مصنوعی Co-op Translator ترجمه شده است. در حالی که ما تلاش میکنیم دقت را حفظ کنیم، لطفاً توجه داشته باشید که ترجمههای خودکار ممکن است شامل خطا یا نواقصی باشند. سند اصلی به زبان مادری آن باید به عنوان منبع معتبر شناخته شود. برای اطلاعات حیاتی، ترجمه حرفهای انسانی توصیه میشود. ما مسئول هیچ گونه سوء تفاهم یا تفسیر نادرستی که ناشی از استفاده از این ترجمه باشد، نیستیم.