问题描述
我正在使用ncurses制作一个小型益智游戏,该游戏允许用户敲击任何箭头键来移动一个包含每个数字的框,这些框都在一个大框中,并按升序排列。这是一个4x4的框,一个子框为空白,以便其他子框可以移动。
现在,我在访问函数中创建的WINDOW(不是指针)数组时遇到问题,这样我就可以销毁所有sub_Box,更改数字数组并根据该数组重新创建子框。
#include <stdio.h>
#include <ncurses.h>
#include <math.h>
#include <string.h>
#define HEIGHT 16
#define WIDTH 32
#define STARTY ((LInes - HEIGHT) / 2)
#define STARTX ((COLS - WIDTH) / 2)
#define QEXIT 81
#define qEXIT 113
WINDOW *create_win(int height,int width,int starty,int startx){
WINDOW *local_win;
local_win = newwin(height,width,starty,startx);
Box(local_win,0);
wrefresh(local_win);
return local_win;
};
void destroy_win(WINDOW *local_win){
wborder(local_win,' ',' ');
wrefresh(local_win);
delwin(local_win);
};
void draw_list(WINDOW *local_win,WINDOW *num_wins,int arr[][4],int length){
for(int i=0; i < length; i++){
for(int j=0; j < length; j++){
refresh();
WINDOW *sub_win;
int height = HEIGHT / length,width = WIDTH / length,starty = STARTY + i*height,startx = STARTX + j*width;
sub_win = create_win(height,startx);
//wprintw(stdscr,"%d |",sizeof(*sub_win));
*(num_wins + i*4+j) = *sub_win;
//wprintw(stdscr,"%d - ",i*4+j);
//overwrite(sub_win,(num_wins + i*4+j));
//memcpy(num_wins + i*4+j,sub_win,sizeof(*sub_win));
int digity = height / 2,digitx = width / 2 - 1;
if(arr[i][j])
mvwprintw(sub_win,digity,digitx,"%d ",arr[i][j]);
else
mvwprintw(sub_win,"X ");
wrefresh(sub_win);
};
};
};
void destroy_list(WINDOW *num_wins){
int length = 16;
for(int i=0; i < length; i++){
wprintw(stdscr,i);
wborder((num_wins + i),' ');
wrefresh((num_wins + i));
delwin((num_wins + i));
};
};
void play(){
int arr[4][4] = {1,4,15,7,8,10,2,11,14,3,6,13,12,9,5,0};
int length = 4;
WINDOW *win;
int starty,startx;
//Placement
starty = STARTY;
startx = STARTX;
printw("Press Q or q to exit\n");
//printw("%d - %d\n%d - %d\n",LInes,COLS,HEIGHT,WIDTH);
printw("\n");
refresh();
// Create original window (Maybe it is useless)
win = create_win(HEIGHT,WIDTH,startx);
// Create sub windows for numbers
WINDOW sub_wins[16];
draw_list(win,sub_wins,arr,length);
refresh();
getch();
destroy_win(&sub_wins[0]);
refresh();
//for(int i=0; i < 16; i++){
// wprintw(stdscr,"%d -",sizeof(sub_wins[i]));
//destroy_win(sub_wins + i);
//};
wrefresh(win);
int ch = getch();
while(ch != QEXIT && ch != qEXIT){
switch(ch){
case KEY_UP:
destroy_win(win);
win = create_win(HEIGHT,--starty,startx);
//destroy_list(sub_wins);
break;
case KEY_RIGHT:
destroy_win(win);
win = create_win(HEIGHT,++startx);
break;
case KEY_DOWN:
destroy_win(win);
win = create_win(HEIGHT,++starty,startx);
break;
case KEY_LEFT:
destroy_win(win);
win = create_win(HEIGHT,--startx);
break;
};
ch = getch();
};
};
int main(void){
//printList(arr,length);
printf("\n");
// Init ncurses
initscr();
cbreak();
noecho();
keypad(stdscr,TRUE);
// Game mechanics
play();
endwin();
return 0;
};
我在代码中抛出了多个问题,而某些部分与我的主要问题并不完全相关。抱歉。但我想主要解决有关破坏子窗口的问题。
谢谢。
解决方法
您不能将一个窗口分为两个窗口。如果您复制窗口结构,则只需使用一个实例即可引用该窗口。
*(num_wins + i*4+j) = *sub_win;
在这里复制窗口。窗口现在位于两个位置,sub_win
和num_wins
数组中。您没有两个窗口,只有一个窗口。因此,您需要选择其中之一。
if(arr[i][j])
mvwprintw(sub_win,digity,digitx,"%d ",arr[i][j]);
else
mvwprintw(sub_win,"X ");
wrefresh(sub_win);
好的,所以您修改sub_win
。这意味着您制作的副本不是窗口。请记住,只有一个窗口。由于您修改了sub_win
,因此它是窗口。因此,在修改窗口之前复制窗口的目的是什么,不再是窗口了。
void destroy_list(WINDOW *num_wins){
int length = 16;
for(int i=0; i < length; i++){
wprintw(stdscr,i);
wborder((num_wins + i),' ',' ');
wrefresh((num_wins + i));
delwin((num_wins + i));
};
};
您在这里做什么?您正在修改窗口之前对其进行操作的副本。但是您对其进行了修改-因此该窗口不再存在。这段代码没有意义。
一个窗口是一个东西,只有一个东西。如果制作WINDOW
结构的副本,则可以使用任何一个副本来表示窗口。但是您不能同时使用两者-只有一个窗口。您的代码同时使用。那是无效的。
摘要:复制该窗口,然后修改该窗口(不复制)。这意味着该副本是您修改窗口之前的副本,没有 更长 存在,因为您修改了复制的对象。因此,尝试访问该副本是一个错误,因为它是不再存在的副本。
有点像这样做:
- 您拿一张白纸。
- 您要复制它。
- 您在原始纸上写“ 1”。
- 您尝试删除副本上的“ 1”。
复制后,您可以将任一面都视为原件,因为它们在 点相同。但是您也不能与对方混淆,否则您将感到非常困惑。