2010-01-30 23:24:23 +00:00
|
|
|
/*
|
|
|
|
* This file is part of MPlayer.
|
|
|
|
*
|
|
|
|
* MPlayer is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* MPlayer is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with MPlayer; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
2007-03-29 17:16:11 +00:00
|
|
|
#include <stdlib.h>
|
2011-05-01 12:57:39 +00:00
|
|
|
#include <assert.h>
|
|
|
|
#include <stdbool.h>
|
2007-03-29 17:16:11 +00:00
|
|
|
#include "osdep/timer.h"
|
|
|
|
#include "input/input.h"
|
2011-04-25 06:43:59 +00:00
|
|
|
#include "input/keycodes.h"
|
2008-04-29 11:52:45 +00:00
|
|
|
#include "mp_fifo.h"
|
2008-04-29 12:12:19 +00:00
|
|
|
#include "talloc.h"
|
2008-04-29 12:44:03 +00:00
|
|
|
#include "options.h"
|
2001-03-16 00:06:50 +00:00
|
|
|
|
2003-04-04 20:02:53 +00:00
|
|
|
|
2008-04-29 12:12:19 +00:00
|
|
|
struct mp_fifo {
|
2008-04-29 12:44:03 +00:00
|
|
|
struct MPOpts *opts;
|
2008-04-29 12:12:19 +00:00
|
|
|
int *data;
|
|
|
|
int readpos;
|
|
|
|
int size;
|
2011-05-01 12:57:39 +00:00
|
|
|
int num_entries;
|
|
|
|
int max_up;
|
|
|
|
int num_up;
|
2011-04-25 11:12:04 +00:00
|
|
|
int last_key_down;
|
|
|
|
unsigned last_down_time;
|
2008-04-29 12:12:19 +00:00
|
|
|
};
|
2003-04-04 20:02:53 +00:00
|
|
|
|
2008-04-29 12:44:03 +00:00
|
|
|
struct mp_fifo *mp_fifo_create(struct MPOpts *opts)
|
2008-04-29 12:12:19 +00:00
|
|
|
{
|
|
|
|
struct mp_fifo *fifo = talloc_zero(NULL, struct mp_fifo);
|
2008-04-29 12:44:03 +00:00
|
|
|
fifo->opts = opts;
|
2011-05-01 12:57:39 +00:00
|
|
|
/* Typical mouse wheel use will generate a sequence repeating 3 events:
|
|
|
|
* down, doubleclick, up, down, doubleclick, up, ...
|
|
|
|
* Normally only one of those event types triggers a command,
|
|
|
|
* so allow opts->key_fifo_size such repeats.
|
|
|
|
*/
|
|
|
|
fifo->max_up = opts->key_fifo_size;
|
|
|
|
fifo->size = opts->key_fifo_size * 3;
|
2008-04-29 12:12:19 +00:00
|
|
|
fifo->data = talloc_array_ptrtype(fifo, fifo->data, fifo->size);
|
|
|
|
return fifo;
|
|
|
|
}
|
|
|
|
|
2011-05-01 12:57:39 +00:00
|
|
|
static bool is_up(int code)
|
|
|
|
{
|
|
|
|
return code > 0 && !(code & MP_KEY_DOWN)
|
|
|
|
&& !(code >= MOUSE_BTN0_DBL && code < MOUSE_BTN_DBL_END);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fifo_peek(struct mp_fifo *fifo, int offset)
|
|
|
|
{
|
|
|
|
return fifo->data[(fifo->readpos + offset) % fifo->size];
|
|
|
|
}
|
|
|
|
|
|
|
|
static int fifo_read(struct mp_fifo *fifo)
|
|
|
|
{
|
|
|
|
int code = fifo_peek(fifo, 0);
|
|
|
|
fifo->readpos += 1;
|
|
|
|
fifo->readpos %= fifo->size;
|
|
|
|
fifo->num_entries--;
|
|
|
|
fifo->num_up -= is_up(code);
|
|
|
|
assert(fifo->num_entries >= 0);
|
|
|
|
assert(fifo->num_up >= 0);
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fifo_write(struct mp_fifo *fifo, int code)
|
|
|
|
{
|
|
|
|
fifo->data[(fifo->readpos + fifo->num_entries) % fifo->size] = code;
|
|
|
|
fifo->num_entries++;
|
|
|
|
fifo->num_up += is_up(code);
|
|
|
|
assert(fifo->num_entries <= fifo->size);
|
|
|
|
assert(fifo->num_up <= fifo->max_up);
|
|
|
|
}
|
|
|
|
|
2008-04-29 12:12:19 +00:00
|
|
|
static void mplayer_put_key_internal(struct mp_fifo *fifo, int code)
|
|
|
|
{
|
2011-05-01 12:57:39 +00:00
|
|
|
// Clear key-down state if we're forced to drop entries
|
|
|
|
if (fifo->num_entries >= fifo->size - 1
|
|
|
|
|| fifo->num_up >= fifo->max_up) {
|
|
|
|
if (fifo_peek(fifo, fifo->num_entries - 1) != MP_INPUT_RELEASE_ALL)
|
|
|
|
fifo_write(fifo, MP_INPUT_RELEASE_ALL);
|
|
|
|
} else
|
|
|
|
fifo_write(fifo, code);
|
2003-04-04 20:02:53 +00:00
|
|
|
}
|
|
|
|
|
2008-04-29 09:55:23 +00:00
|
|
|
int mplayer_get_key(void *ctx, int fd)
|
|
|
|
{
|
2008-04-29 12:12:19 +00:00
|
|
|
struct mp_fifo *fifo = ctx;
|
2011-05-01 12:57:39 +00:00
|
|
|
if (!fifo->num_entries)
|
2008-04-29 12:12:19 +00:00
|
|
|
return MP_INPUT_NOTHING;
|
2011-05-01 12:57:39 +00:00
|
|
|
return fifo_read(fifo);
|
2003-04-04 20:02:53 +00:00
|
|
|
}
|
|
|
|
|
2008-04-29 12:12:19 +00:00
|
|
|
static void put_double(struct mp_fifo *fifo, int code)
|
|
|
|
{
|
2011-04-25 06:43:59 +00:00
|
|
|
if (code >= MOUSE_BTN0 && code < MOUSE_BTN_END)
|
2008-04-29 12:12:19 +00:00
|
|
|
mplayer_put_key_internal(fifo, code - MOUSE_BTN0 + MOUSE_BTN0_DBL);
|
2007-01-18 19:30:02 +00:00
|
|
|
}
|
|
|
|
|
2008-04-29 12:12:19 +00:00
|
|
|
void mplayer_put_key(struct mp_fifo *fifo, int code)
|
|
|
|
{
|
|
|
|
unsigned now = GetTimerMS();
|
2008-04-29 12:44:03 +00:00
|
|
|
int doubleclick_time = fifo->opts->doubleclick_time;
|
2008-04-29 12:12:19 +00:00
|
|
|
// ignore system-doubleclick if we generate these events ourselves
|
|
|
|
if (doubleclick_time
|
|
|
|
&& (code & ~MP_KEY_DOWN) >= MOUSE_BTN0_DBL
|
2011-04-25 06:43:59 +00:00
|
|
|
&& (code & ~MP_KEY_DOWN) < MOUSE_BTN_DBL_END)
|
2008-04-29 12:12:19 +00:00
|
|
|
return;
|
|
|
|
mplayer_put_key_internal(fifo, code);
|
|
|
|
if (code & MP_KEY_DOWN) {
|
|
|
|
code &= ~MP_KEY_DOWN;
|
2011-04-25 11:12:04 +00:00
|
|
|
if (fifo->last_key_down == code
|
|
|
|
&& now - fifo->last_down_time < doubleclick_time)
|
2008-04-29 12:12:19 +00:00
|
|
|
put_double(fifo, code);
|
2011-04-25 11:12:04 +00:00
|
|
|
fifo->last_key_down = code;
|
|
|
|
fifo->last_down_time = now;
|
2008-04-29 12:12:19 +00:00
|
|
|
}
|
2007-01-18 19:30:02 +00:00
|
|
|
}
|