⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ch16.htm

📁 ncurses中文说明
💻 HTM
📖 第 1 页 / 共 2 页
字号:
<html>
<head>
<title>第十六章:面板库</title>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312">
<meta name="keywords" content="ncurses,curses,面板库">
</head>
<body leftmargin=40 bgcolor="#DAFFFF" text="black" link="#0000ff" alink="blue" vlink="#940084">
<br>
<div align="center"><a href="index.htm"><font size="5">
    <strong>
    回  目  录
    </strong>
  </font></a></div>
  <div align="right">
  <font size="5">
  <a href="ch15.htm">
  <strong>
  上 一 章
  </strong>
  </a>
  <a href="ch17.htm">
    <strong>
    下 一 章
	</strong>
	</a>
	</font>
	</div>
	<br>
<br>
<br>
<h1 align="center">
第十六章:面板库
</h1>
<br>
<br>
<br>
<div align="left"><font size="4">
<p>
<dd>
现在你应该已经很精通curses库了,你可能会想试着做一些更大的项目。为了让界面看起来更专业,于是你就创建了许多重叠的窗口。但很不幸的是,你很快会发现它们变得难以管理,多次的更新窗口使开发变成了一场恶梦。如果你没有按照适当的顺序刷新那些窗口的话,它们就会给你带来很多麻烦。
</dd>
</p>
<p>
<dd>
不过别失望,面板库(Panel Library)提供了一个很好的解决方案。用ncureses 的开发者的话来说就是:
</dd>
</p>
<P>
<dd>
<i>
如果你的界面设计需要使窗口在运行的时候置底或者置顶,你就会发现不光显示正确的结果很困难,而且为此付出得代价很大。这时候Panels库就可以派上用场了。
</i>
</dd>
</P>
<p>
<dd>
如果你要处理很多重叠的窗口,选择panels 库绝对没错。通过一系列的wnoutrefresh()函数和doupdate()函数调用可以大大减少工作量。面板库实时控制着窗口的顺序信息,并且可以处理好重叠顺序,而且可以适时的更新屏幕。读到这里,你是不是又开始激动了呢?那还等什么呢,让我们继续!
</dd>
</p>
<br>
<br>
<h3 align="left">
<a name="basis">
16.1 基础知识
</a>
</h3>
<br>
<p>
<dd>
面板对象实际上也是一个窗口,它被认为是一块包含所有面板对象的一个平台。这个平台可以看作是一个栈,栈顶的面板是完全可见的,其它的面板就被“压”在下面。所有面板根据所处栈的位置不同从而决定它是否可见。基本思想就是:创建一个栈来保存那些重叠的面板,然后使用面板库来正确的显示它们。可以调用一个类似于 refresh()的函数来以正确的显示顺序显示这些面板。 面板库提供了一些可以隐藏、显示、移动和改变其大小等操作的函数。这样,我们就可以通过调用面板库提供的函数来解决重叠的窗口的问题了。
</p>
<p>
一般的,一个面板程序的设计流程如下:
</p>
<p>
1、	创建要添加到面板里面去的窗口(使用newwin())。
</p>
<p>
2、	创建一个有合适可见顺序的面板,根据需要的可见顺序把它们压进栈。函数new_panel() 可以用来创建面板。
</p>
<p>
3、	调用update_panels() 把面板按正确的顺序写到虚拟屏幕上,调用doupdate()让它在现实器上显示。
</p>
<p>
4、	调用show_panel(), hide_panel(), move_panel()等函数来对面板进行操作。可以使用一些例如panel_hidden() 和 panel_window()之类的辅助函数。你也可以使用用户指针来存储为一个定制面板的数据。而且可以使用函数 set_panel_userptr() 和 panel_userptr() 来设置和取得一个面板的用户指针。
</p>
<p>
5、	所有的工作都做完后用del_panel() 删除这个面板。
</p>
<p>
<dd>
接下来我们通过一些程序来加深一下对这些概念的理解。下面的将要看到的程序创建了3个重叠的面板并把它们按次序显示在屏幕上。
</dd>
</p>
<br>
<br>
<h3 align="left">
<a name="compile">
16.2  编译包含面板库的程序
</a>
</h3>
<br>
<p>
<dd>
要使用面板库里的函数,你首先要把panel.h这个头文件包含到你的代码中。如果要把编译并连接与面板库相关的程序的话就要同时添加-lpanel和 –lncurses两个参数。
</dd>
</p>
<font color="Maroon">
<pre>
#include &lt;panel.h&gt;
    .
    .
    .
	编译和连接: gcc <program file> -lpanel –lncurses

</pre>
</font>
<p>
例14.一个有关面板库的基础例子
</p>
<font color="Maroon">
<pre>
#include &lt;panel.h&gt;

int main()
{	
	WINDOW *my_wins[3];
	PANEL  *my_panels[3];
	int lines = 10, cols = 40, y = 2, x = 4, i;

	initscr();
	cbreak();
	noecho();
	/* 为每个面板创建窗口 */
	my_wins[0] = newwin(lines, cols, y, x);
	my_wins[1] = newwin(lines, cols, y + 1, x + 5);
	my_wins[2] = newwin(lines, cols, y + 2, x + 10);

	/* 为窗口添加创建边框以便你能看到面板的效果 */
	for(i = 0; i < 3; +++i)
		box(my_wins[i], 0, 0);

	/* 按自底向上的顺序,把给每个窗口添加进一个面板 */ 	
	my_panels[0] = new_panel(my_wins[0]);/* 把面板0压进栈, 叠放顺序: stdscr-0 */
	my_panels[1] = new_panel(my_wins[1]);/* 把面板1压进栈, 叠放顺序: stdscr-0-1 */
	my_panels[2] = new_panel(my_wins[2]);/* 把面板2压进栈, 叠放顺序: stdscr-0-1-2 */

	/* 更新栈的顺序。把面板2置于栈顶  */
	update_panels();

	/* 在屏幕上显示 */
	doupdate();
	
	getch();
	endwin();
}

</pre>
</font>
<p>
<dd>
如你所见,上面的这个程序就是按照前面所讲的那个流程进行的。用newwin() 创建窗口,然后用new_panel()把它们添加到面板库里面去,当我们把那些面板一个一个压进栈的时候,panels栈也就随之更新了。调用update_panels() 和 doupdate() 就可以让它们在屏幕上显示出来。
</dd>
</p>
<br>
<br>
<h3 align="left">
<a name="browse">
16.3 面板窗口浏览
</a>
</h3>
<p>
<dd>
下面给出一个稍微复杂点的例子。这个程序创建了3个窗口,通过使用<TAB>键可以使它们循环置顶显示。让我们来看一下代码:
</dd>
</p>
<p>
例15 一个面板窗口浏览的例子
</p>
<font color="Maroon">
<pre>
#include &lt;panel.h&gt;
#define NLINES 10
#define NCOLS 40

void init_wins(WINDOW **wins, int n);
void win_show(WINDOW *win, char *label, int label_color);
void print_in_middle(WINDOW *win, int starty, int startx, int width, char *string, chtype color);

int main()
{	
	WINDOW *my_wins[3];
	PANEL  *my_panels[3];
	PANEL  *top;
	int ch;

	/*初始化curses */
	initscr();
	start_color();
	cbreak();
	noecho();
	keypad(stdscr, TRUE);

	/* 初始化所有的颜色 */
	init_pair(1, COLOR_RED, COLOR_BLACK);
	init_pair(2, COLOR_GREEN, COLOR_BLACK);
	init_pair(3, COLOR_BLUE, COLOR_BLACK);
	init_pair(4, COLOR_CYAN, COLOR_BLACK);

	init_wins(my_wins, 3);
	
	/* 按自底向上的顺序,把每个窗口添加进一个面板  */
	my_panels[0] = new_panel(my_wins[0]);	/* 把面板0压入栈, 顺序: stdscr-0 */
	my_panels[1] = new_panel(my_wins[1]); 	/* 把面板1压入栈,顺序: stdscr-0-1 */
	my_panels[2] = new_panel(my_wins[2]); 	/* 把面板2压入栈,顺序: stdscr-0-1-2 */

	/* 为下一个面板建立用户指针 */
	set_panel_userptr(my_panels[0], my_panels[1]);
	set_panel_userptr(my_panels[1], my_panels[2]);
	set_panel_userptr(my_panels[2], my_panels[0]);

	/* 更新面板栈的顺序。把面板2置于栈顶*/
	update_panels();

	/* 在屏幕上显示*/
	attron(COLOR_PAIR(4));
	mvprintw(LINES - 2, 0, "Use tab to browse through the windows (F1 to Exit)");
	attroff(COLOR_PAIR(4));
	doupdate();

	top = my_panels[2];
	while((ch = getch()) != KEY_F(1))
	{	switch(ch)
		{	case 9:
				top = (PANEL *)panel_userptr(top);
				top_panel(top);
				break;
		}
		update_panels();
		doupdate();
	}
	endwin();
	return 0;
}

/* 显示所有的窗口 */
void init_wins(WINDOW **wins, int n)
{	
	int x, y, i;
	char label[80];

	y = 2;
	x = 10;
	for(i = 0; i < n; ++i)
	{	wins[i] = newwin(NLINES, NCOLS, y, x);
		sprintf(label, "Window Number %d", i + 1);
		win_show(wins[i], label, i + 1);
		y += 3;
		x += 7;
	}
}

/* 用一个边框和控件显示所有的窗口 */
void win_show(WINDOW *win, char *label, int label_color)
{	
	int startx, starty, height, width;

	getbegyx(win, starty, startx);
	getmaxyx(win, height, width);

	box(win, 0, 0);
	mvwaddch(win, 2, 0, ACS_LTEE); 
	mvwhline(win, 2, 1, ACS_HLINE, width - 2); 
	mvwaddch(win, 2, width - 1, ACS_RTEE); 
	
	print_in_middle(win, 1, 0, width, label, COLOR_PAIR(label_color));
}

void print_in_middle(WINDOW *win, int starty, int startx, int width, char *string, chtype color)
{	
	int length, x, y;
	float temp;

	if(win == NULL)
		win = stdscr;
	getyx(win, y, x);
	if(startx != 0)
		x = startx;
	if(starty != 0)
		y = starty;
	if(width == 0)
		width = 80;

	length = strlen(string);
	temp = (width - length)/ 2;
	x = startx + (int)temp;
	wattron(win, color);
	mvwprintw(win, y, x, "%s", string);
	wattroff(win, color);
	refresh();
}

</pre>
</font>
<br>
<br>
<h3 align="left">
<a name="user's pointer">
16.4 使用用户指针
</a>
</h3>
<br>
<p>
<dd>
在上面的例子中,为了在这个循环例查找下一个窗口我使用了用户指针。我们可以通过指定一个用户指针给面板增加自定义信息,这个指针可以指向你想要存储的任何信息。在这个例子中,我往指针里面存储了循环中的下一个面板。一个面板的用户指针可以用函数set_panel_userptr() 来设定。用面板作为函数panel_userptr() 参变量可以访问它,函数将返回这个面板的用户指针。在循环中查找下一个面板结束后,通过使用函数top_panel() 可以将其置于顶层。用给定的面板作为参变量,这个函数将会把这个面板置于面板栈的顶层。
</dd>
</p>
<br>
<br>
<h3 align="left">
<a name="move">
16.5 移动面板和改变面板的大小
</a>
</h3>
<br>
<p>
<dd>
函数move_panel()可以将面板移动到你想移动的地方。它不改变面板在栈里的位置。务必使用move_panel()而不是用来移动窗口的函数mvwin() 来移动面板。
</dd>
</p>
<p>
<dd>
改变面板的大小有一点点复杂,没有哪个函数可以用来直接改变和面板关联的窗口的大小。要改变面板的大小,只有创建一个大小为你所需的新窗口。用函数replace_panel()可以替换和相应面板关联的窗口。但是别忘了要删除旧的窗口。和面板关联的窗口可以使用函数panel_window()来创建。
</dd>
</p>
<p>
<dd>
下面这个程序就用简单的方式体现了这个概念。同样,你可以用<TAB>键来使它们循环。要改变大小或移动当前的面板你可以按下 ‘ r’ 或 ‘ m ’键。然后使用方向键来改变大小和移动。在具体操作时,为获得需要的结果这个例子利用了用户已有的数据。
</dd>
</p>
<p>
例16、 一个移动和改变面板大小的例子
</p>
<font color="Maroon">
<pre>
#include &lt;panel.h&gt;

typedef struct _PANEL_DATA {
	int x, y, w, h;
	char label[80]; 
	int label_color;
	PANEL *next;
}PANEL_DATA;

#define NLINES 10
#define NCOLS 40

void init_wins(WINDOW **wins, int n);
void win_show(WINDOW *win, char *label, int label_color);
void print_in_middle(WINDOW *win, int starty, int startx, int width, char *string, chtype color);
void set_user_ptrs(PANEL **panels, int n);

int main()
{	WINDOW *my_wins[3];
	PANEL  *my_panels[3];
	PANEL_DATA  *top;
	PANEL *stack_top;
	WINDOW *temp_win, *old_win;
	int ch;
	int newx, newy, neww, newh;
	int size = FALSE, move = FALSE;

	/* 初始化curses */
	initscr();
	start_color();
	cbreak();
	noecho();
	keypad(stdscr, TRUE);

	/* 初始化所有的颜色 */
	init_pair(1, COLOR_RED, COLOR_BLACK);
	init_pair(2, COLOR_GREEN, COLOR_BLACK);
	init_pair(3, COLOR_BLUE, COLOR_BLACK);
	init_pair(4, COLOR_CYAN, COLOR_BLACK);

	init_wins(my_wins, 3);
	
	/* 更新面板栈的顺序。把面板2置于栈顶*/
	my_panels[0] = new_panel(my_wins[0]); 	/* 把面板0压入栈,顺序: stdscr-0 */
	my_panels[1] = new_panel(my_wins[1]); 	/* 把面板1压入栈。顺序: stdscr-0-1 */
	my_panels[2] = new_panel(my_wins[2]); 	/* 把面板2压入栈,顺序: stdscr-0-1-2 */

	set_user_ptrs(my_panels, 3);
	/* 更新面板栈的顺序。把面板2置于栈顶 */
	update_panels();

	/* 在屏幕上显示出来 */
	attron(COLOR_PAIR(4));
	mvprintw(LINES - 3, 0, "Use 'm' for moving, 'r' for resizing");
	mvprintw(LINES - 2, 0, "Use tab to browse through the windows (F1 to Exit)");
	attroff(COLOR_PAIR(4));
	doupdate();

	stack_top = my_panels[2];
	top = (PANEL_DATA *)panel_userptr(stack_top);
	newx = top->x;
	newy = top->y;
	neww = top->w;
	newh = top->h;
	while((ch = getch()) != KEY_F(1))

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -