【ChatGPT】組隊ChatGPT製作SEGA Mega Drive(MD)遊戲-Google Chrome小恐龍遊戲

更新於 發佈於 閱讀時間約 40 分鐘
Photo by Joshua Doherty on Unsplash

開通ChatGPT PLUS

話說ChatGPT+SGDK能不能蹦出MD(SEGA Mega Drive)新遊戲呢?這幾天開通了ChatGPT PLUS想說來試試使用ChatGPT作款能在MD主機上玩的小遊戲,順便記錄一下製作這款遊戲的過程,至於要作哪款遊戲呢?上篇文章惡搞了Google Chrome小恐龍,那就選Google Chrome小恐龍好了,不囉嗦說作就作。。。

簡單企劃文件與環境安裝

在開始動手寫Google Chrome小恐龍執行企劃的時候,忽然想到可以動動手請ChatGPT寫,二話不說開啟ChatGPT直接輸入"請幫我整理Google Chrome恐龍遊戲簡單製作企畫說明文件",ChatGPT也立馬回應:
唉呦!不錯喔,有模有樣的,雖然沒比我寫的好(心虛),最少寫的比我快,那我們就勉強著參考這份文件開始製作遊戲了,首先來建構開發環境,請連至以下網址下載工具一共有三個檔案:
  • SGDK 1.80 (sgdk180.7z)
  • KMod v0.7.3模擬器 (檔案名稱:Gens_KMod_v0.7.3.7z)
  • Visual Studio Code (檔案名稱:VSCodeUserSetup-x64–1.78.0-insider.exe)
下載完後建立目錄MegaDrive將三個檔案複製進去,先將sgdk180.7z與Gens_KMod_v0.7.3.7z解壓縮然後安裝VSCodeUserSetup-x64–1.78.0-insider.exe,待Visual Studio Code安裝完成請將之開啟,參考下圖找到Genesis Code外掛並安裝:
接下來設定Genesis Code,開啟Settings選項
依照下圖進行設定
設定完畢將Visual Studio Code關閉,接下來進入sgdk180目錄,執行build_adv.bat與build_lib.bat
完成後重新開啟Visual Studio Code,按下熱鍵Ctrl+Shift+P,依照下圖步驟建立一個測試專案:
建立test目錄並選擇此目錄
開啟Terminal
依照下圖步驟執行設定與編譯程式
編譯後如果看到以下視窗跳出恭喜!我們開發環境就設定完成了

開始與ChatGPT溝通製作遊戲

再來就要請出我們月薪六百台幣的ChatGPT PLUS,開門見山就跟他說"使用請用SGDK 1.70的寫一個類似Google Chrome恐龍的遊戲",因為太緊張還有結巴,原本想說因該要跑個一小時吧,正準備轉身去茶水間泡個咖啡偷懶一下時,ChatGPT已經啪啦啪啦將程式碼寫一半就停了,只好叫他繼續我看著呢!
前後大概不到二分鐘的時間就完成了,好吧!這個產出速度個人是自認沒法比了,只能安慰自己寫的快內容不見的是對的,依照ChatGPT說的我們先建立一個sgdk_dino專案(作法前面有教跟建立test專案的步驟一樣),建立完成後開啟main.c並將裡面資料全部清除,並將ChatGPT寫好的程式碼全部複製貼上,然後開始理解程式碼,發現只有兩個錯誤#include <resources.h>用不到先刪除,加上沒有驅動搖桿的程式碼,幫忙加上:
u8 value;
value = JOY_getPortType(PORT_1);
switch (value)
{
case PORT_TYPE_MENACER:
JOY_setSupport(PORT_1, JOY_SUPPORT_MENACER);
break;
case PORT_TYPE_JUSTIFIER:
JOY_setSupport(PORT_1, JOY_SUPPORT_JUSTIFIER_BOTH);
break;
case PORT_TYPE_MOUSE:
JOY_setSupport(PORT_1, JOY_SUPPORT_MOUSE);
break;
case PORT_TYPE_TEAMPLAYER:
JOY_setSupport(PORT_1, JOY_SUPPORT_TEAMPLAYER);
break;
}
其他程式碼看上去沒問題,整合進我們程式內並編譯執行,按下手把A可以控制恐龍方塊跳起,按下手把Start重新開始遊戲,以下是執行影片:
老實說當時我害怕極了,真的可以玩,想當初學SGDK搞了一個禮拜才看到Hello SEGA!!我們繼續,想說每次都要自己玩有點累,加上也想知道恐龍方塊在不死的狀況下持續往右跑會不會出現其他問題,所以接下來我們繼續對ChatGPT下指令"請修改讓恐龍遇到障礙物自動跳起"
看了一下程式碼,沒大問題直接整合到我們程式內,編譯執行後也確實能正常執行:
在遊戲自動玩了一小段時間後果然發現恐龍方塊在往上跳下降後會掉落到地板下,繼續指揮ChatGPT說"主角在往上跳落下後會掉落地板下,請修正這個問題":
看程式碼,也沒大問題直接整合進去,編譯執行後也都正常運作:
高興地玩了一陣子後發現障礙物都只有一格,只好繼續指揮ChatGPT讓障礙物每次出現有不同高度,請輸入"請修改obstacle變數,讓障礙物每次出現時都有不同高度":
繼續理解程式碼發現多處有問題,所以就不叫ChatGPT修改了,我們自行修正,首先frameCounter變數用不到,所以直接刪除,將障礙物初始高度設定為1
obstacle.height = OBSTACLE_HEIGHT
修改
obstacle.height = 1
繪製障礙物迴圈內繪製障礙方塊的參數有誤請作以下修改
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL2, FALSE, FALSE, FALSE, 1), 
obstacle.x / 8, (obstacle.y - i * 8) / 8, obstacle.width, 2);
修改
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL2, FALSE, FALSE, FALSE, 1),
obstacle.x / 8, (obstacle.y - i * 8) / 8, 1, 2);
修正完畢後繼續編譯執行:
影片中可以看出障礙物已經會出現不同高度,可是恐龍方塊碰到障礙物不會結束遊戲,繼續對ChatGPT說道"經過以上修改恐龍碰到障礙物不會遊戲結束了,請修正這個問題"
看了一下程式碼除了障礙物高度錯誤外其他沒大問題,先整合進去,然後障礙物高度部分修正如下:
if (dino.x + 16 < obstacle.x || dino.x > obstacle.x + obstacle.width * 8) {
修改
if (dino.x + 16 < obstacle.x || dino.x > obstacle.x + obstacle.width * 2) {
對了,因該要加個Game Over顯示,讓恐龍方塊碰到障礙物後可以顯示,這個也先不麻煩ChatGPT,自己動手改一下,將主迴圈修改如下:
while(1)
{
SYS_doVBlankProcess();

// 更新恐龍與障礙物.
updateDino();
updateObstacle();

// 修改.
if (checkCollision()) {
// 清除障礙物.
VDP_setTileMapDataRect ( BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), 0, 0, 40, 25, 1, DMA);

// 當遊戲結束時,顯示文字提示
VDP_drawTextBG( BG_A, "Game Over!", 15, 12);

while (1) {
SYS_doVBlankProcess();
u16 keys = JOY_readJoypad(JOY_1);
if (keys & BUTTON_A){
// 清除畫面.
VDP_clearPlane (BG_A, TRUE);
// 初始遊戲.
createDino();
createObstacle();
break;
}
VDP_waitVSync();
}
}
VDP_waitVSync();
}
修改完畢直接編譯執行:
開始有點像樣了,繼續來加點分數顯示,直接跟ChatGPT說"請加分數功能,分數計算方法為每往右移動一格(obstacle.x — )分數就加1"
看了一下程式碼沒大問題,將之整合進去,不過整合的時候忽然想到如果每往右移動1Pixel分數就加1這樣分數會過大,所以順手改成每移動一個方塊(8Pixel)分數就加1這樣會比較好一點,然後也加了分數越高移動速度越快的設定:
void updateObstacle() {

// ... 保持代碼不變

// CHANNEL修改.
// 更新障礙物位置,依照分數設定障礙物一動速度.
if(score>1200){
obstacle.x-=8;
}else if(score>500){
obstacle.x-=4;
}else if(score>50){
obstacle.x-=2;
}else{
obstacle.x--;
}

// 讓恐龍往右每走一格(8像素)就加1分.
if(obstacle.x%8 == 0){
score++;
}

// ... 保持代碼不變,直到主迴圈

}
完成後編譯執行
開始有模有樣了,我們在來加個最高分數紀錄,直接跟ChatGPT說"請加入每局結束後判斷最高分數功能,並顯示在畫面中間上方位置"
看了下程式碼也沒什麼大問題將之整合進去,感覺畫面有點不明顯,所以對畫面區塊顏色作了一些調整:
int main() {
// ... 保持代碼不變

// 設定背景顏色.
PAL_setColor (0, RGB24_TO_VDPCOLOR(0xa1adff));
// 設定障礙物顏色.
PAL_setColor (1, RGB24_TO_VDPCOLOR(0xde5718));
// 設定恐龍顏色.
PAL_setColor (2, RGB24_TO_VDPCOLOR(0x0600ff));
// 設定寶物顏色.
PAL_setColor (3, RGB24_TO_VDPCOLOR(0xff0000));
// 設定雲顏色.
PAL_setColor (4, RGB24_TO_VDPCOLOR(0xc9c9c9));

// ... 保持代碼不變
}
為了讓遊戲更有趣,來加個由右往左移動的方塊,恐龍方塊可以跳起來吃掉並增加分數,一樣對著ChatGPT說"請增加判斷score變數每增加100分就從右邊生出一個方塊往左移動,移動到左邊盡頭就消失,並判斷如果恐龍碰到此方塊就將score加20同時關閉方塊"
看了一下程式碼發現由右往左移動的方塊那部分沒將移動過的方塊清除,所以會有殘留方塊,繼續跟ChatGPT說"方塊往左移動後會殘留舊方塊請修改將之清除"
看完程式碼沒問題,將之整合進去,並修正了一些邏輯問題,最後編譯執行
遊戲大致完成,玩了幾十分鐘,感覺可以在加兩朵雲來修飾一下畫面,所以跟ChatGPT說"請在上方製作2朵雲使其從最右往左慢慢移動,移動到最左邊後使其在回到右邊在繼續往左移動,反覆以上動作,雲移動過後要清除舊的雲不要留下殘影"
程式碼沒大問題將之整合進去,編譯執行
影片中為了測試方便將恐龍方塊改成無敵了。
完成,整理程式碼
好的!到此Google Chrome小恐龍遊戲已經完成,將程式碼重頭到尾整理一遍加入註解如下:
#include <genesis.h>
#include "pal.h"

// 常數設定.
#define GROUND_Y 200
#define DINO_Y 180
#define DINO_WIDTH 16
#define DINO_HEIGHT 16
#define OBSTACLE_WIDTH 8
#define OBSTACLE_HEIGHT 16
#define JUMP_HEIGHT 40

#define BONUS_START_X 320
#define BONUS_Y 120
#define BONUS_SPEED 4

#define CLOUD1_START_X 320
#define CLOUD1_Y 16
#define CLOUD2_START_X 400
#define CLOUD2_Y 24
#define CLOUD_SPEED 1

// 恐龍方塊結構.
typedef struct {
s16 x, y;
u16 width, height;
u8 isJumping;
s16 jumpCounter;
} Dino;
// 障礙物結構.
typedef struct {
s16 x, y;
u16 width, height;
u8 isActive;
} Obstacle;
// 寶物結構.
typedef struct {
s16 x;
s16 y;
bool isActive;
} Bonus;
// 雲結構.
typedef struct {
s16 x;
s16 y;
} Cloud;

Dino dino;
Obstacle obstacle;
Bonus bonus;
Cloud cloud1;
Cloud cloud2;

// 在main函數開頭添加分數變數.
int score = 0;
// 在main函數開頭添加最高分數變數.
int highScore = 0;

void createDino();
void createObstacle();
void updateDino();
void updateObstacle();
bool checkCollision();
void updateScore();
void updateHighScore();

//----------------------------------------------------------------------------
// 雲 - 初始.
//----------------------------------------------------------------------------
void initClouds() {
cloud1.x = CLOUD1_START_X;
cloud1.y = CLOUD1_Y;
cloud2.x = CLOUD2_START_X;
cloud2.y = CLOUD2_Y;
}

//----------------------------------------------------------------------------
// 雲 - 更新.
//----------------------------------------------------------------------------
void updateCloud(Cloud *cloud) {
// 清除舊雲位置
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), cloud->x / 8, cloud->y / 8);

cloud->x -= CLOUD_SPEED;

// 如果雲超出螢幕左邊邊界,將其重置到右側
if (cloud->x < -16) {
cloud->x = 320;
}

// 繪製新雲位置
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 4), cloud->x / 8, cloud->y / 8);
}

//----------------------------------------------------------------------------
// 初始寶物方塊.
//----------------------------------------------------------------------------
void spawnBonus() {
bonus.x = BONUS_START_X;
bonus.y = BONUS_Y;
bonus.isActive = TRUE;
}

//----------------------------------------------------------------------------
// 更新寶物方塊.
//----------------------------------------------------------------------------
void updateBonus() {
if (!bonus.isActive) {
return;
}

// 清除舊方塊位置
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), bonus.x / 8, bonus.y / 8);

bonus.x -= BONUS_SPEED;

// 如果方塊超出螢幕左邊邊界,將 isActive 設為 FALSE
if (bonus.x < -8) {
bonus.isActive = FALSE;
return;
}

// 檢查恐龍與方塊之間的碰撞
if (dino.y + 16 >= bonus.y && dino.y <= bonus.y + 8 && dino.x + 16 >= bonus.x && dino.x <= bonus.x + 8) {
bonus.isActive = FALSE;
score += 100;
return;
}

// 繪製方塊
if (bonus.isActive) {
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 3), bonus.x / 8, bonus.y / 8);
} else {
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), bonus.x / 8, bonus.y / 8);
}
}

//----------------------------------------------------------------------------
// 初始障礙物.
//----------------------------------------------------------------------------
void initObstacle() {
obstacle.x = 320;
obstacle.y = GROUND_Y - 8;
obstacle.width = 2;
}

//----------------------------------------------------------------------------
// 亂數障礙物高低.
//----------------------------------------------------------------------------
void spawnObstacle() {
obstacle.x = 320;

// 生成隨機高度(1到5格)
obstacle.height = 1 + (random() % 5);

// 繪製障礙物
for (int i = 0; i < obstacle.height; i++) {
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL2, FALSE, FALSE, FALSE, 1), obstacle.x / 8, (obstacle.y - i * 8) / 8, obstacle.width, 1);
}
}

//----------------------------------------------------------------------------
// 程式進入點.
//----------------------------------------------------------------------------
int main() {
// 初始化螢幕
VDP_init();

// 設置解析度
VDP_setScreenWidth320();
VDP_setScreenHeight224();

// 設定背景顏色.
PAL_setColor (0, RGB24_TO_VDPCOLOR(0xa1adff));
// 設定障礙物顏色.
PAL_setColor (1, RGB24_TO_VDPCOLOR(0xde5718));
// 設定恐龍顏色.
PAL_setColor (2, RGB24_TO_VDPCOLOR(0x0600ff));
// 設定寶物顏色.
PAL_setColor (3, RGB24_TO_VDPCOLOR(0xff0000));
// 設定雲顏色.
PAL_setColor (4, RGB24_TO_VDPCOLOR(0xc9c9c9));

// 加入搖桿判斷.
u8 value;
value = JOY_getPortType(PORT_1);
switch (value)
{
case PORT_TYPE_MENACER:
JOY_setSupport(PORT_1, JOY_SUPPORT_MENACER);
break;
case PORT_TYPE_JUSTIFIER:
JOY_setSupport(PORT_1, JOY_SUPPORT_JUSTIFIER_BOTH);
break;
case PORT_TYPE_MOUSE:
JOY_setSupport(PORT_1, JOY_SUPPORT_MOUSE);
break;
case PORT_TYPE_TEAMPLAYER:
JOY_setSupport(PORT_1, JOY_SUPPORT_TEAMPLAYER);
break;
}

createDino();
createObstacle();

// 初始化雲.
initClouds();

while(1)
{
SYS_doVBlankProcess();

// 更新恐龍與障礙物.
updateDino();
updateObstacle();
updateScore(); // 更新分數
updateHighScore();
updateBonus(); // 更新方塊

updateCloud(&cloud1); // 更新雲1
updateCloud(&cloud2); // 更新雲2

if (checkCollision()) {
// 當遊戲結束時,顯示文字提示
VDP_drawTextBG( BG_A, "Game Over!", 15, 12);

while (1) {
SYS_doVBlankProcess();

u16 keys = JOY_readJoypad(JOY_1);
if (keys & BUTTON_START){
// 清除畫面.
VDP_clearPlane (BG_A, TRUE);
// 初始遊戲.
createDino();
createObstacle();
// 初始分數.
score = 0;

break;
}
VDP_waitVSync();
}
}
VDP_waitVSync();
}
return 0;
}

//----------------------------------------------------------------------------
// 建立恐龍方塊.
//----------------------------------------------------------------------------
void createDino() {
dino.x = 32;
dino.y = DINO_Y;
dino.width = DINO_WIDTH;
dino.height = DINO_HEIGHT;
dino.isJumping = FALSE;
dino.jumpCounter = 0;
}

//----------------------------------------------------------------------------
// 建立障礙物.
//----------------------------------------------------------------------------
void createObstacle() {
obstacle.x = 320;
obstacle.y = GROUND_Y - OBSTACLE_HEIGHT;
obstacle.width = OBSTACLE_WIDTH;
obstacle.height = 1;
obstacle.isActive = TRUE;
}

//----------------------------------------------------------------------------
// 更新恐龍方塊.
//----------------------------------------------------------------------------
void updateDino() {
// 清除恐龍之前的位置
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), dino.x / 8, dino.y / 8);
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), dino.x / 8 + 1, dino.y / 8);
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), dino.x / 8, dino.y / 8 + 1);
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), dino.x / 8 + 1, dino.y / 8 + 1);

if (dino.isJumping) {
dino.jumpCounter++;
if (dino.jumpCounter < JUMP_HEIGHT) {
dino.y -= 2;
} else if (dino.jumpCounter < 2 * JUMP_HEIGHT) {
dino.y += 2;
} else {
dino.jumpCounter = 0;
dino.isJumping = FALSE;
}
}

// 根據按鍵狀態更新恐龍
u16 keys = JOY_readJoypad(JOY_1);
if (keys & BUTTON_A && !dino.isJumping) {
dino.isJumping = TRUE;
}

// 根據恐龍與障礙物的距離自動跳起
//if (!dino.isJumping && dino.y == DINO_Y && obstacle.x - (dino.x + DINO_WIDTH) < 32) {
// dino.isJumping = TRUE;
//}

// 恐龍落地後將 dino.isJumping 設定為 FALSE
if (dino.y > DINO_Y) {
dino.y = DINO_Y;
dino.isJumping = FALSE;
}

// 繪製新的恐龍位置
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 2), dino.x / 8, dino.y / 8, 2, 2);
}

//----------------------------------------------------------------------------
// 更新障礙物.
//----------------------------------------------------------------------------
void updateObstacle() {
// 清除障礙物之前的位置
for (int i = 0; i < obstacle.height; i++) {
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 0), obstacle.x / 8, (obstacle.y - i * 8) / 8);
}

// 更新障礙物位置,依照分數設定障礙物一動速度.
if(score>1200){
obstacle.x-=8;
}else if(score>500){
obstacle.x-=4;
}else if(score>50){
obstacle.x-=2;
}else{
obstacle.x--;
}

// 讓恐龍往右每走一格(8像素)就加1分.
if(obstacle.x%8 == 0){
score++;
}

// 檢查是否需要重新生成障礙物
if (obstacle.x + 8 * obstacle.width < 0) {
spawnObstacle();
}

// 繪製新的障礙物位置
for (int i = 0; i < obstacle.height; i++) {
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL0, FALSE, FALSE, FALSE, 1), obstacle.x / 8, (obstacle.y - i * 8) / 8, 1, 2);
}
}

//----------------------------------------------------------------------------
// 判斷恐龍方塊與障礙物碰撞.
//----------------------------------------------------------------------------
bool checkCollision() {
// 如果恐龍在空中並且足夠高以避免碰撞,則返回FALSE.
if (dino.isJumping && dino.y <= (obstacle.y - obstacle.height * 8)) {
return FALSE;
}

// 如果恐龍和障礙物在水平位置上沒有接觸,則返回FALSE.
if (dino.x + 16 < obstacle.x || dino.x > obstacle.x + obstacle.width * 2) {
return FALSE;
}

return TRUE;
}

//----------------------------------------------------------------------------
// 更新分數.
//----------------------------------------------------------------------------
void updateScore() {
static int prevScore = 0;

// 新增修改方塊顯示中就不要再進入執行(!bonus.isActive).
if (score % 100 == 0 && score != prevScore && !bonus.isActive) {
spawnBonus();
prevScore = score;
}

char scoreStr[10];
sprintf(scoreStr, "Score: %d", score);
VDP_drawText(scoreStr, 2, 0);
}

//----------------------------------------------------------------------------
// 更新最高分數.
//----------------------------------------------------------------------------
void updateHighScore() {
if (score > highScore) {
highScore = score;
}
char highScoreStr[15];
sprintf(highScoreStr, "High Score: %d", highScore);
VDP_drawText(highScoreStr, 16, 0);
}
所有程式碼與ChatGPT聊天內容也上傳到Github上

Github

實機試玩影片

後記

大家如果詳細看與ChatGPT聊天內容(文件以上傳至GitHub)會發現有很多ChatGPT寫出來的程式碼無法使用或過多Bugs,所以沒有採納,這部分有很大的原因是因為所輸入的指令不夠精準,文章為了順暢度所以沒將這部分寫入,總歸首次的ChatGPT體驗讓我非常驚訝,AI也走到了所謂的奇異點,依照這個進展相信未來AI也將成為我們生活中的標配,好的!本次的MD小遊戲製作教學也到了尾聲,我們下次見。
對了,如果您熟悉SGDK也可以將遊戲內方塊換成圖片,這樣可以讓畫面看起來更專業,非常期待您的作品。

更新日誌

  • 2023/4/26 上線Youtube影片
即將進入廣告,捲動後可繼續閱讀
為什麼會看到廣告
avatar-img
5會員
16內容數
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
無限升級的沙龍 的其他內容
Hello!大家好我是茄諾,今天要來幫自主研發的掌機TBoy寫款可以連線對戰的遊戲,不知道TBoy的朋友可以連至以下網址看看: 因為要連線對戰,所以需要準備兩台TBoy掌機,請參考以上文章製作;接下來要挑選適合連線對戰的遊戲,個人偏愛Pong,這款由Atari開發的投幣式大型電玩對戰遊戲非常適合與家
前言 嗨,各位懷舊遊戲愛好者!今天要跟大家分享一個有趣的主題:如何利用React和Pixi.js這兩大神兵利器,重塑我們那個年代的經典紅白機打磚塊遊戲! 先跟大家簡單科普一下,React是一個超級火爆的前端框架,能讓我們輕鬆創建可重用的UI組件,組件間的狀態管理也相當方便。。。
想要在復古遊戲機上享受無線遊戲體驗,同時又能炫耀自己現代的遊戲控制器嗎?BlueRetro 來拯救你的童年!這款開源的藍牙遊戲控制器轉接器讓你可以把最新款的 PlayStation、Xbox 和 Switch Pro 控制器連接到那些古老的遊戲機上。誰說過復古不能與現代搭配?
話說ChatGPT+SGDK能不能蹦出MD(SEGA Mega Drive)新遊戲呢?這幾天開通了ChatGPT PLUS想說來試試使用ChatGPT作款能在MD主機上玩的小遊戲,順便記錄一下製作這款遊戲的過程,至於要作哪款遊戲呢?
Hi!大家好,今天來教大家如何玩壞Google Chrome小恐龍,這款Chrome離線小恐龍跑酷遊戲,一般是在開啟Chrome瀏覽器並在離線狀態下可以玩到的遊戲,其實也是可以在連線狀態下玩的。。。
前陣子在某線上拍賣平台買了一支超便宜的有線超任USB搖桿,結果玩不到一個禮拜按鈕就被按壞了,而且發現還沒家裡那台老舊超級任天堂搖桿順手,所以就開始打起老舊超任搖桿的主意。。。。
Hello!大家好我是茄諾,今天要來幫自主研發的掌機TBoy寫款可以連線對戰的遊戲,不知道TBoy的朋友可以連至以下網址看看: 因為要連線對戰,所以需要準備兩台TBoy掌機,請參考以上文章製作;接下來要挑選適合連線對戰的遊戲,個人偏愛Pong,這款由Atari開發的投幣式大型電玩對戰遊戲非常適合與家
前言 嗨,各位懷舊遊戲愛好者!今天要跟大家分享一個有趣的主題:如何利用React和Pixi.js這兩大神兵利器,重塑我們那個年代的經典紅白機打磚塊遊戲! 先跟大家簡單科普一下,React是一個超級火爆的前端框架,能讓我們輕鬆創建可重用的UI組件,組件間的狀態管理也相當方便。。。
想要在復古遊戲機上享受無線遊戲體驗,同時又能炫耀自己現代的遊戲控制器嗎?BlueRetro 來拯救你的童年!這款開源的藍牙遊戲控制器轉接器讓你可以把最新款的 PlayStation、Xbox 和 Switch Pro 控制器連接到那些古老的遊戲機上。誰說過復古不能與現代搭配?
話說ChatGPT+SGDK能不能蹦出MD(SEGA Mega Drive)新遊戲呢?這幾天開通了ChatGPT PLUS想說來試試使用ChatGPT作款能在MD主機上玩的小遊戲,順便記錄一下製作這款遊戲的過程,至於要作哪款遊戲呢?
Hi!大家好,今天來教大家如何玩壞Google Chrome小恐龍,這款Chrome離線小恐龍跑酷遊戲,一般是在開啟Chrome瀏覽器並在離線狀態下可以玩到的遊戲,其實也是可以在連線狀態下玩的。。。
前陣子在某線上拍賣平台買了一支超便宜的有線超任USB搖桿,結果玩不到一個禮拜按鈕就被按壞了,而且發現還沒家裡那台老舊超級任天堂搖桿順手,所以就開始打起老舊超任搖桿的主意。。。。
你可能也想看
Google News 追蹤
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
這篇內容,將教你如何開啟新的GameMaker專案,並調整畫面佈局。也會講解,為何建議用英文語系,來進行遊戲開發。
Thumbnail
這篇內容,將教你如何安裝GameMaker,並更改語言設定。包括GameMaker的下載說明、版本說明、安裝說明、語系更改。
Thumbnail
[懷舊遊戲週報 2024/7-1] 懷舊遊戲週報收集每週與懷舊遊戲相關的新訊,預定每週五出刊。 本週主要的消息包含: ‧MD 新作遊戲《ZPF》展開募資 ‧《閃電出擊V》原聲帶 ‧《時尚魔女》迷你框體模型...等其他資訊
Thumbnail
這次這篇文章資源很多,三大送,送送送。好啦!直接進入正題巴 我先送大家一個ChatGPT的指令寫出Midjourney的咒語
Thumbnail
故事開始,前一陣子找到以前上電腦課時,用MIT App inventor復刻經典的Zelda薩爾達跟Pokemon寶可夢的GBA遊戲,然後突然靈光一閃,決定挑戰在一年內寫完一本小說。 簡單說明工具: 本體內容結構,用AI跟大量閱讀輔助構思劇情 選擇性影響結局,用Twine2構思不同選擇與相對應
工欲善其事,必先利其器,要打造屬於自己的Chat GPT之前,我們先學習怎麼建立Google免費提供的Colab環境,它可以免費使用GPU來加速AI的運算,非常適合沒有錢添購GPU,但又想學習前沿AI技術的人。 第一步:打開Google瀏覽器,並點選右上方的「方格子點點」,接著選擇「雲端硬碟」
Thumbnail
[懷舊遊戲週報 2024/4-4] 懷舊遊戲週報收集每週與懷舊遊戲相關的新訊,預定每週五出刊。 本週主要的消息包含: ‧MD 新作遊戲《Cyber Mission》展開募資 ‧《閃電風暴》R-GRAY1 組合模型再版 ‧《洛克人》FC 卡匣造型收納盒...等其他資訊
Thumbnail
這篇文章分享了一個感覺高科技的AVG製作工具,詳細介紹了TyranoBuilder和TyranoScript之間的區別,並提到了自己對於文字 + 選項的AVG玩法的看法。
Thumbnail
從最初的接觸到深度的投懷送抱,這篇文章對於解決如何安裝text-generation-webui提供了清晰的指引。 它引導讀者通過它的官方方式和Docker方式做到這一點。此外,作者還分享了必要的優化建議。
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
這篇內容,將教你如何開啟新的GameMaker專案,並調整畫面佈局。也會講解,為何建議用英文語系,來進行遊戲開發。
Thumbnail
這篇內容,將教你如何安裝GameMaker,並更改語言設定。包括GameMaker的下載說明、版本說明、安裝說明、語系更改。
Thumbnail
[懷舊遊戲週報 2024/7-1] 懷舊遊戲週報收集每週與懷舊遊戲相關的新訊,預定每週五出刊。 本週主要的消息包含: ‧MD 新作遊戲《ZPF》展開募資 ‧《閃電出擊V》原聲帶 ‧《時尚魔女》迷你框體模型...等其他資訊
Thumbnail
這次這篇文章資源很多,三大送,送送送。好啦!直接進入正題巴 我先送大家一個ChatGPT的指令寫出Midjourney的咒語
Thumbnail
故事開始,前一陣子找到以前上電腦課時,用MIT App inventor復刻經典的Zelda薩爾達跟Pokemon寶可夢的GBA遊戲,然後突然靈光一閃,決定挑戰在一年內寫完一本小說。 簡單說明工具: 本體內容結構,用AI跟大量閱讀輔助構思劇情 選擇性影響結局,用Twine2構思不同選擇與相對應
工欲善其事,必先利其器,要打造屬於自己的Chat GPT之前,我們先學習怎麼建立Google免費提供的Colab環境,它可以免費使用GPU來加速AI的運算,非常適合沒有錢添購GPU,但又想學習前沿AI技術的人。 第一步:打開Google瀏覽器,並點選右上方的「方格子點點」,接著選擇「雲端硬碟」
Thumbnail
[懷舊遊戲週報 2024/4-4] 懷舊遊戲週報收集每週與懷舊遊戲相關的新訊,預定每週五出刊。 本週主要的消息包含: ‧MD 新作遊戲《Cyber Mission》展開募資 ‧《閃電風暴》R-GRAY1 組合模型再版 ‧《洛克人》FC 卡匣造型收納盒...等其他資訊
Thumbnail
這篇文章分享了一個感覺高科技的AVG製作工具,詳細介紹了TyranoBuilder和TyranoScript之間的區別,並提到了自己對於文字 + 選項的AVG玩法的看法。
Thumbnail
從最初的接觸到深度的投懷送抱,這篇文章對於解決如何安裝text-generation-webui提供了清晰的指引。 它引導讀者通過它的官方方式和Docker方式做到這一點。此外,作者還分享了必要的優化建議。