ncurses滑鼠事件:mousemask(),ALL_MOUSE_EVENTS,KEY_MOUSE,getmouse(),mouse_grafo(),wmouse_trafo()

Koma_Wong發表於2018-06-22
  • Copyright(C) NCURSES Programming HOWTO

使用滑鼠

你現在已經知道如何取得鍵盤的輸入了,那現在讓我們也來取得滑鼠的輸入。因為很多使用者介面程式都支援使用鍵盤和滑鼠的共同操作。

基礎知識

在使用滑鼠之前,首先要呼叫mousemask( )這個函式來啟用你想要接收的滑鼠事件。

mousemask( mmask_t newmask, /* 你想要監聽的滑鼠事件掩碼*/
mmask_t *oldmask ) /* 舊版本使用的滑鼠事件掩碼*/

上述函式中的第一個引數,就是你所要監聽的事件的位掩碼,預設情況下,在使用該函式之前,所有滑鼠事件的接收狀態都是未啟用的。位掩碼ALL_MOUSE_EVENTS 可以讓滑鼠接收所有的事件。 下面是NCURSES 定義的位掩碼清單:(注意:不同的滑鼠按鍵號碼設定不同,使用前需要測試。一般情況下左鍵為1 號,右鍵為2 號)

掩碼                      對應事件
BUTTON1_PRESSED         滑鼠1 號鍵      按下
BUTTON1_RELEASED        滑鼠1 號鍵      釋放
BUTTON1_CLICKED         滑鼠1 號鍵      單擊
BUTTON1_DOUBLE_CLICKED  滑鼠1 號鍵      雙擊
BUTTON1_TRIPLE_CLICKED  滑鼠1 號鍵      三擊
BUTTON2_PRESSED         滑鼠2 號鍵      按下
BUTTON2_RELEASED        滑鼠2 號鍵      釋放
BUTTON2_CLICKED         滑鼠2 號鍵      單擊
BUTTON2_DOUBLE_CLICKED  滑鼠2 號鍵      雙擊
BUTTON2_TRIPLE_CLICKED  滑鼠2 號鍵      三擊
BUTTON3_PRESSED         滑鼠3 號鍵      按下
BUTTON3_RELEASED        滑鼠3 號鍵      釋放
BUTTON3_CLICKED         滑鼠3 號鍵      單擊
BUTTON3_DOUBLE_CLICKED  滑鼠3 號鍵      雙擊
BUTTON3_TRIPLE_CLICKED  滑鼠3 號鍵      三擊
BUTTON4_PRESSED         滑鼠4 號鍵      按下
BUTTON4_RELEASED        滑鼠4 號鍵      釋放
BUTTON4_CLICKED         滑鼠4 號鍵      單擊
BUTTON4_DOUBLE_CLICKED  滑鼠4 號鍵      雙擊
BUTTON4_TRIPLE_CLICKED  滑鼠4 號鍵      三擊
BUTTON_SHIFT            在滑鼠事件發生時,伴隨Shift 鍵按下
BUTTON_CTRL             在滑鼠事件發生時,伴隨Ctrl 鍵按下
BUTTON_ALT              在滑鼠事件發生時,伴隨Alt 鍵按下
ALL_MOUSE_EVENTS        報告所有的滑鼠事件
REPORT_MOUSE_POSITION   報告滑鼠移動位置

取得滑鼠事件

當所有的滑鼠監聽事件被啟用後。getch()一類的函式在每次接收到的滑鼠事件時可以返回KEY_MOUSE。然後通過getmouse()函式可以取得這些事件。 程式碼大概看起來是這樣:

MEVENT event;
ch = getch();
if(ch == KEY_MOUSE)
if(getmouse(&event)== OK)
{
    /* 處理這個事件的程式碼*/
}

getmouse()函式將這個事件返回一個相應的指標。這個指標結構是這樣的:

typedef struct
{
    short id; /* ID 用來辨別不同的裝置*/
    int x, y, z; /* 事件發生的座標*/
    mmask_t bstate; /* 滑鼠按鍵狀態*/
}

Bstate 是我們關注的最主要變數,它返回了當前滑鼠按鍵的狀態。下面的這段程式碼可以讓我們看看按下滑鼠左鍵會出現什麼:

if(event.bstate & BUTTON1_PRESSED)
    printw("Left Button Pressed"

把它們放在一起

能夠使用滑鼠操作的程式是非常棒的,讓我們做用滑鼠操作的選單程式。為了讓例子看起來更有針對性,這個程式中去掉了鍵盤操作:

  • 這個程式碼是書中的程式碼,可能有些問題,在文章最後新增了另外一個網友的程式碼。
/*
Compile: gcc main.c -lncurses
*/
#include <stdio.h>
#include <ncurses.h>
#include <string.h>
#include <stdlib.h>

#define WIDTH 30
#define HEIGHT 10

int startx = 0;
int starty = 0;

char *choices[] = {
            "Choice 1",
            "Choice 2",
            "Choice 3",
            "Choice 4",
            "Exit",};
            
int n_choices = sizeof(choices) / sizeof(char *);
void print_menu(WINDOW *menu_win, int highlight);
void report_choice(int mouse_x, int mouse_y, int *p_choice);

int main()
{ 
    int c, choice = 0;
    WINDOW *menu_win;
    MEVENT event;
    
    /* 初始化curses */
    initscr();
    clear();
    noecho();
    cbreak();   /* 禁用行緩衝,直接傳遞所有的訊號*/
    
    /* 將視窗放在螢幕中央*/
    //startx = (80-WIDTH)/ 2;
    //starty = (24-HEIGHT)/ 2;
    
    attron(A_REVERSE);
    
    mvprintw(23, 1, "Click on Exit to quit (Works best in a virtual console)");
    
    refresh();
    attroff(A_REVERSE);
    
    /* 首先顯示選單*/
    menu_win = newwin(HEIGHT, WIDTH, starty, startx);
    print_menu(menu_win,1);
    
    /* 監聽所有的滑鼠事件*/
    mousemask(ALL_MOUSE_EVENTS, 0);
        
        
    while(1)
    { 
        c = wgetch(menu_win);
        
        switch(c)
        { 
            case KEY_MOUSE:
                getmouse(&event);
                if (!wenclose(menu_win, event.y, event.x)) 
                    break; /*do nothing if not in window*/
                
                report_choice(event.x, event.y, &choice);
                if(choice == -1)
                    /* 退出選項*/
                    goto end;
                mvprintw(22, 1, "%d, %d", event.x, event.y);
                refresh();
                
                print_menu(menu_win,choice);
                break;
                
            case 'q' :
                goto end;
                break;
                
            case 'a':
                print_menu(menu_win,2);
                break;
                
            default :
                break;
        }
    }
    
end:
    endwin();
    return 0;
}

void print_menu(WINDOW *menu_win, int highlight)
{
    int x, y, i;
    
    x = 2;
    y = 2;
    
    box(menu_win, 0, 0);
    
    for(i = 0; i < n_choices; ++i)
    { 
        if(highlight == i + 1)
        { 
            wattron(menu_win,A_REVERSE);
            mvwprintw(menu_win, y, x, "%s", choices[i]);
            wattroff(menu_win,A_REVERSE);
        }
        else
            mvwprintw(menu_win, y, x, "%s", choices[i]);
        
        ++y;
    }
    wrefresh(menu_win);
}
/* 報告滑鼠所在位置的選單選項*/
void report_choice(int mouse_x, int mouse_y, int *p_choice)
{ 
    int choice;

    
    for(choice = 0; choice < n_choices; ++choice)
        if(mouse_y == choice 
            && mouse_x >= 0 
            && mouse_x <= strlen(choices[choice]))
        {
            if(choice == n_choices-1)
                *p_choice = -1;
            else
                *p_choice = choice + 1;
            break;
        }
}

一些輔助函式說明

mouse_trafo() 函式和wmouse_trafo()函式用來轉換滑鼠座標到相對應的螢幕座標。想得到詳細資料可以閱讀curs_mouse(3X)的聯機man 文件。 mouseinterval()函式用來設定識別滑鼠單擊的間隔時間(即按鍵按下和彈起的時間,按千分之一秒計), 並返回上一次設定的間隔時間, 預設的間隔時間是五分之一秒, 即mouseinterval(200)

另一位網友的程式碼,內附網址

/*
https://blog.csdn.net/ccccce/article/details/51096167
Compile: gcc main.c -lncurses
*/
#include <curses.h>
#include <unistd.h>

void showstar(WINDOW * win, const int line);

int main(void)
{
    int key;
    int quit = 0;
    MEVENT mouse;
    WINDOW * win;

    initscr();
    raw();
    win = newwin(10, 50, 5, 3);     /*this must do before keypad*/
    keypad(win, TRUE);              /*then use win.*/
    mousemask(BUTTON1_CLICKED | BUTTON2_CLICKED, 0);    /*set actions*/
    box(win, '|', '-');
    mvwaddch(win, 1, 48, 'X');
    mvwaddstr(win, 3, 3, "Test 1");
    mvwaddstr(win, 4, 3, "Test 2");
    mvwaddstr(win, 5, 3, "Test 3");
    wrefresh(win);

    while (!quit)
    {
        key = wgetch(win);
        switch(key)
        {
            case KEY_MOUSE : 
                getmouse(&mouse); /*get mouse action*/

                if (!wenclose(win, mouse.y, mouse.x)) 
                    break; /*do nothing if not in window*/

                wmouse_trafo(win, &mouse.y, &mouse.x, FALSE);

                if ((3 <= mouse.x && mouse.x <= 8) 
                 && (3 <= mouse.y && mouse.y <= 5))
                    showstar(win, mouse.y);

                if (1 == mouse.y && 48 == mouse.x) /*Clicked 'X'*/
                    quit = 1;

                break;
            case 'q' :
                quit = 1;
                break;
            default :
                break;
        }
    }

    delwin(win);
    endwin();
    return 0;
}

void showstar(WINDOW * win, const int line)
{
    mvwaddch(win, line, 2, '*');
    wrefresh(win);
}

相關文章