Перезапустить бесконечный цикл while для события мыши? - PullRequest
0 голосов
/ 26 сентября 2018

Я стремлюсь к простому коду, который выглядит следующим образом:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <poll.h>
#include <linux/input.h>

int main()
{
    int fd, bytes;
    unsigned char data[3];
    fd = open("/dev/input/mice", O_RDWR);
    int left;

    while(1)
    {    
        bytes = read(fd, data, sizeof(data));

        if(bytes > 0)
        {
            left = data[0] & 0x1;
            right = data[0] & 0x2;

            if(left==1){
            goto reset_EVENT;
            }           
        }

        reset_EVENT:

        printf("hey 1");
        sleep(1);   
        printf("hey 2");
        sleep(1);
        printf("hey 3");
        sleep(1);
    }
    return 0;
}

Моя цель здесь - перезапустить последовательность печати с «эй 1» сразу после щелчка левой кнопкой мыши.Однако я не могу достичь этого без использования условий «если» перед каждым сном."goto" не работает для меня в разных функциях тоже.Кто-нибудь может предложить оптимальное решение для этого, пожалуйста?

1 Ответ

0 голосов
/ 26 сентября 2018

Этого можно достичь с помощью какого-то конечного автомата: функции, которая имеет внутреннее статическое состояние, чтобы знать, какой был последний hey напечатанный

void print_event(int reset)
{
    /* by default, print nothing */
    static int state = 4;

    /* if reset, restart counting */
    if (reset) 
        state = 1;

    /* if state under 4, display `hey` */
    if (state <= 3)
        printf("hey %d\n", state++);
}

int main()
{
    int fd, bytes;
    unsigned char data[3];
    fd = open("/dev/input/mice", O_RDWR);
    int left, right;        

    while(1)    
    {   
        /* by default, event is not reset */
        int reset = 0;
        bytes = read(fd, data, sizeof(data));

        if(bytes > 0)
        {
            left = data[0] & 0x1;
            right = data[0] & 0x2;

            if(left==1){
                /* left click: reset event*/
                reset = 1;
            }           
        }

        /* ask function to print if needed */
        print_event(reset);

        sleep(1);               
    }
    return 0;
}

РЕДАКТИРОВАТЬ

одна проблема в вашем цикле состоит в том, что вы будете долго ждать, прежде чем читать новое значение из fd.

. Для этого вы можете использовать сигнал:

  • непрерывно считывая ввод с мышей, без сна
  • при обнаружении левого щелчка, подайте сигнал тревоги для отображения события:

Таким образом, ваш код может быть:

/* print event function: can be called from alarm signal handler
   or directly from main function */
void print_event(int reset)
{
    /* by default, print nothing */
    static int state = 4;

    /* if reset, restart counting */
    if (reset) 
        state = 1;

    /* if state under 4, display `hey` */
    if (state <= 3)
    {
        printf("hey %d", state++);

        /* if all `hey` have not been printed, prepare the next */
        alarm(1);
    }
}

/* alarm signal receiver*/
void handle(int sig) {
    /* print event without reset */
    print_event(0);
}

int main()
{
    int fd, bytes;
    unsigned char data[3];
    fd = open("/dev/input/mice", O_RDWR);
    int left;

    /* tell what function is to be called when alarm signal is raised */
    signal(SIGALRM, handle);

    /* if you want to display one sequence before the first click, uncomment the following line */
    /* print_event(1); */

    while(1)    
    {   
        bytes = read(fd, data, sizeof(data));

        if(bytes > 0)
        {
            left = data[0] & 0x1;                

            if(left==1){
                /* left click: reset event*/
                print_event(1);
            }           
        }
    }
    return 0;
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...