Blame alsamixer/mixer_widget.c

Packit 229ac0
/*
Packit 229ac0
 * mixer_widget.c - mixer widget and keys handling
Packit 229ac0
 * Copyright (c) 1998,1999 Tim Janik
Packit 229ac0
 *                         Jaroslav Kysela <perex@perex.cz>
Packit 229ac0
 * Copyright (c) 2009      Clemens Ladisch <clemens@ladisch.de>
Packit 229ac0
 *
Packit 229ac0
 * This program is free software: you can redistribute it and/or modify
Packit 229ac0
 * it under the terms of the GNU General Public License as published by
Packit 229ac0
 * the Free Software Foundation, either version 2 of the License, or
Packit 229ac0
 * (at your option) any later version.
Packit 229ac0
 *
Packit 229ac0
 * This program is distributed in the hope that it will be useful,
Packit 229ac0
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 229ac0
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 229ac0
 * GNU General Public License for more details.
Packit 229ac0
 *
Packit 229ac0
 * You should have received a copy of the GNU General Public License
Packit 229ac0
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 229ac0
 */
Packit 229ac0
Packit 229ac0
#include "aconfig.h"
Packit 229ac0
#include <stdlib.h>
Packit 229ac0
#include <string.h>
Packit 229ac0
#include <errno.h>
Packit 229ac0
#include <alsa/asoundlib.h>
Packit 229ac0
#include "gettext_curses.h"
Packit 229ac0
#include "version.h"
Packit 229ac0
#include "utils.h"
Packit 229ac0
#include "die.h"
Packit 229ac0
#include "mem.h"
Packit 229ac0
#include "colors.h"
Packit 229ac0
#include "widget.h"
Packit 229ac0
#include "textbox.h"
Packit 229ac0
#include "proc_files.h"
Packit 229ac0
#include "card_select.h"
Packit 229ac0
#include "volume_mapping.h"
Packit 229ac0
#include "mixer_controls.h"
Packit 229ac0
#include "mixer_display.h"
Packit 229ac0
#include "mixer_widget.h"
Packit 229ac0
Packit 229ac0
snd_mixer_t *mixer;
Packit 229ac0
char *mixer_device_name;
Packit 229ac0
bool unplugged;
Packit 229ac0
Packit 229ac0
struct widget mixer_widget;
Packit 229ac0
Packit 229ac0
enum view_mode view_mode;
Packit 229ac0
Packit 229ac0
int focus_control_index;
Packit 229ac0
snd_mixer_selem_id_t *current_selem_id;
Packit 229ac0
unsigned int current_control_flags;
Packit 229ac0
Packit 229ac0
bool control_values_changed;
Packit 229ac0
bool controls_changed;
Packit 229ac0
Packit 229ac0
enum channel_mask {
Packit 229ac0
	LEFT = 1,
Packit 229ac0
	RIGHT = 2,
Packit 229ac0
};
Packit 229ac0
Packit 229ac0
static int elem_callback(snd_mixer_elem_t *elem, unsigned int mask)
Packit 229ac0
{
Packit 229ac0
	if (mask == SND_CTL_EVENT_MASK_REMOVE) {
Packit 229ac0
		controls_changed = TRUE;
Packit 229ac0
	} else {
Packit 229ac0
		if (mask & SND_CTL_EVENT_MASK_VALUE)
Packit 229ac0
			control_values_changed = TRUE;
Packit 229ac0
Packit 229ac0
		if (mask & SND_CTL_EVENT_MASK_INFO)
Packit 229ac0
			controls_changed = TRUE;
Packit 229ac0
	}
Packit 229ac0
Packit 229ac0
	return 0;
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static int mixer_callback(snd_mixer_t *mixer, unsigned int mask, snd_mixer_elem_t *elem)
Packit 229ac0
{
Packit 229ac0
	if (mask & SND_CTL_EVENT_MASK_ADD) {
Packit 229ac0
		snd_mixer_elem_set_callback(elem, elem_callback);
Packit 229ac0
		controls_changed = TRUE;
Packit 229ac0
	}
Packit 229ac0
	return 0;
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
void create_mixer_object(struct snd_mixer_selem_regopt *selem_regopt)
Packit 229ac0
{
Packit 229ac0
	int err;
Packit 229ac0
Packit 229ac0
	err = snd_mixer_open(&mixer, 0);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		fatal_alsa_error(_("cannot open mixer"), err);
Packit 229ac0
Packit 229ac0
	mixer_device_name = cstrdup(selem_regopt->device);
Packit 229ac0
	err = snd_mixer_selem_register(mixer, selem_regopt, NULL);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		fatal_alsa_error(_("cannot open mixer"), err);
Packit 229ac0
Packit 229ac0
	snd_mixer_set_callback(mixer, mixer_callback);
Packit 229ac0
Packit 229ac0
	err = snd_mixer_load(mixer);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		fatal_alsa_error(_("cannot load mixer controls"), err);
Packit 229ac0
Packit 229ac0
	err = snd_mixer_selem_id_malloc(&current_selem_id);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		fatal_error("out of memory");
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void set_view_mode(enum view_mode m)
Packit 229ac0
{
Packit 229ac0
	view_mode = m;
Packit 229ac0
	create_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void close_hctl(void)
Packit 229ac0
{
Packit 229ac0
	free_controls();
Packit 229ac0
	if (mixer_device_name) {
Packit 229ac0
		snd_mixer_detach(mixer, mixer_device_name);
Packit 229ac0
		free(mixer_device_name);
Packit 229ac0
		mixer_device_name = NULL;
Packit 229ac0
	}
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void check_unplugged(void)
Packit 229ac0
{
Packit 229ac0
	snd_hctl_t *hctl;
Packit 229ac0
	snd_ctl_t *ctl;
Packit 229ac0
	unsigned int state;
Packit 229ac0
	int err;
Packit 229ac0
Packit 229ac0
	unplugged = FALSE;
Packit 229ac0
	if (mixer_device_name) {
Packit 229ac0
		err = snd_mixer_get_hctl(mixer, mixer_device_name, &hctl);
Packit 229ac0
		if (err >= 0) {
Packit 229ac0
			ctl = snd_hctl_ctl(hctl);
Packit 229ac0
			/* just any random function that does an ioctl() */
Packit 229ac0
			err = snd_ctl_get_power_state(ctl, &state);
Packit 229ac0
			if (err == -ENODEV)
Packit 229ac0
				unplugged = TRUE;
Packit 229ac0
		}
Packit 229ac0
	}
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
void close_mixer_device(void)
Packit 229ac0
{
Packit 229ac0
	check_unplugged();
Packit 229ac0
	close_hctl();
Packit 229ac0
Packit 229ac0
	display_card_info();
Packit 229ac0
	set_view_mode(view_mode);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
bool select_card_by_name(const char *device_name)
Packit 229ac0
{
Packit 229ac0
	int err;
Packit 229ac0
	bool opened;
Packit 229ac0
	char *msg;
Packit 229ac0
Packit 229ac0
	close_hctl();
Packit 229ac0
	unplugged = FALSE;
Packit 229ac0
Packit 229ac0
	opened = FALSE;
Packit 229ac0
	if (device_name) {
Packit 229ac0
		err = snd_mixer_attach(mixer, device_name);
Packit 229ac0
		if (err >= 0)
Packit 229ac0
			opened = TRUE;
Packit 229ac0
		else {
Packit 229ac0
			msg = casprintf(_("Cannot open mixer device '%s'."), device_name);
Packit 229ac0
			show_alsa_error(msg, err);
Packit 229ac0
			free(msg);
Packit 229ac0
		}
Packit 229ac0
	}
Packit 229ac0
	if (opened) {
Packit 229ac0
		mixer_device_name = cstrdup(device_name);
Packit 229ac0
Packit 229ac0
		err = snd_mixer_load(mixer);
Packit 229ac0
		if (err < 0)
Packit 229ac0
			fatal_alsa_error(_("cannot load mixer controls"), err);
Packit 229ac0
	}
Packit 229ac0
Packit 229ac0
	display_card_info();
Packit 229ac0
	set_view_mode(view_mode);
Packit 229ac0
	return opened;
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void show_help(void)
Packit 229ac0
{
Packit 229ac0
	const char *help[] = {
Packit 229ac0
		_("Esc     Exit"),
Packit 229ac0
		_("F1 ? H  Help"),
Packit 229ac0
		_("F2 /    System information"),
Packit 229ac0
		_("F3      Show playback controls"),
Packit 229ac0
		_("F4      Show capture controls"),
Packit 229ac0
		_("F5      Show all controls"),
Packit 229ac0
		_("Tab     Toggle view mode (F3/F4/F5)"),
Packit 229ac0
		_("F6 S    Select sound card"),
Packit 229ac0
		_("L       Redraw screen"),
Packit 229ac0
		"",
Packit 229ac0
		_("Left    Move to the previous control"),
Packit 229ac0
		_("Right   Move to the next control"),
Packit 229ac0
		"",
Packit 229ac0
		_("Up/Down    Change volume"),
Packit 229ac0
		_("+ -        Change volume"),
Packit 229ac0
		_("Page Up/Dn Change volume in big steps"),
Packit 229ac0
		_("End        Set volume to 0%"),
Packit 229ac0
		_("0-9        Set volume to 0%-90%"),
Packit 229ac0
		_("Q W E      Increase left/both/right volumes"),
Packit 229ac0
		/* TRANSLATORS: or Y instead of Z */
Packit 229ac0
		_("Z X C      Decrease left/both/right volumes"),
Packit 229ac0
		_("B          Balance left and right volumes"),
Packit 229ac0
		"",
Packit 229ac0
		_("M          Toggle mute"),
Packit 229ac0
		/* TRANSLATORS: or , . */
Packit 229ac0
		_("< >        Toggle left/right mute"),
Packit 229ac0
		"",
Packit 229ac0
		_("Space      Toggle capture"),
Packit 229ac0
		/* TRANSLATORS: or Insert Delete */
Packit 229ac0
		_("; '        Toggle left/right capture"),
Packit 229ac0
		"",
Packit 229ac0
		_("Authors:"),
Packit 229ac0
		_("  Tim Janik"),
Packit 229ac0
		_("  Jaroslav Kysela <perex@perex.cz>"),
Packit 229ac0
		_("  Clemens Ladisch <clemens@ladisch.de>"),
Packit 229ac0
	};
Packit 229ac0
	show_text(help, ARRAY_SIZE(help), _("Help"));
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
void refocus_control(void)
Packit 229ac0
{
Packit 229ac0
	if (focus_control_index < controls_count) {
Packit 229ac0
		snd_mixer_selem_get_id(controls[focus_control_index].elem, current_selem_id);
Packit 229ac0
		current_control_flags = controls[focus_control_index].flags;
Packit 229ac0
	}
Packit 229ac0
Packit 229ac0
	display_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static struct control *get_focus_control(unsigned int type)
Packit 229ac0
{
Packit 229ac0
	if (focus_control_index >= 0 &&
Packit 229ac0
	    focus_control_index < controls_count &&
Packit 229ac0
	    (controls[focus_control_index].flags & IS_ACTIVE) &&
Packit 229ac0
	    (controls[focus_control_index].flags & type))
Packit 229ac0
		return &controls[focus_control_index];
Packit 229ac0
	else
Packit 229ac0
		return NULL;
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_enum_to_percent(struct control *control, int value)
Packit 229ac0
{
Packit 229ac0
	unsigned int i;
Packit 229ac0
	unsigned int index;
Packit 229ac0
	unsigned int new_index;
Packit 229ac0
	int items;
Packit 229ac0
	int err;
Packit 229ac0
Packit 229ac0
	i = ffs(control->enum_channel_bits) - 1;
Packit 229ac0
	err = snd_mixer_selem_get_enum_item(control->elem, i, &index);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		return;
Packit 229ac0
	new_index = index;
Packit 229ac0
	if (value == 0) {
Packit 229ac0
		new_index = 0;
Packit 229ac0
	} else if (value == 100) {
Packit 229ac0
		items = snd_mixer_selem_get_enum_items(control->elem);
Packit 229ac0
		if (items < 1)
Packit 229ac0
			return;
Packit 229ac0
		new_index = items - 1;
Packit 229ac0
	}
Packit 229ac0
	if (new_index == index)
Packit 229ac0
		return;
Packit 229ac0
	for (i = 0; i <= SND_MIXER_SCHN_LAST; ++i)
Packit 229ac0
		if (control->enum_channel_bits & (1 << i))
Packit 229ac0
			snd_mixer_selem_set_enum_item(control->elem, i, new_index);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_enum_relative(struct control *control, int delta)
Packit 229ac0
{
Packit 229ac0
	int items;
Packit 229ac0
	unsigned int i;
Packit 229ac0
	unsigned int index;
Packit 229ac0
	int new_index;
Packit 229ac0
	int err;
Packit 229ac0
Packit 229ac0
	items = snd_mixer_selem_get_enum_items(control->elem);
Packit 229ac0
	if (items < 1)
Packit 229ac0
		return;
Packit 229ac0
	err = snd_mixer_selem_get_enum_item(control->elem, 0, &index);
Packit 229ac0
	if (err < 0)
Packit 229ac0
		return;
Packit 229ac0
	new_index = (int)index + delta;
Packit 229ac0
	if (new_index < 0)
Packit 229ac0
		new_index = 0;
Packit 229ac0
	else if (new_index >= items)
Packit 229ac0
		new_index = items - 1;
Packit 229ac0
	if (new_index == index)
Packit 229ac0
		return;
Packit 229ac0
	for (i = 0; i <= SND_MIXER_SCHN_LAST; ++i)
Packit 229ac0
		if (control->enum_channel_bits & (1 << i))
Packit 229ac0
			snd_mixer_selem_set_enum_item(control->elem, i, new_index);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_volume_to_percent(struct control *control, int value, unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	int (*set_func)(snd_mixer_elem_t *, snd_mixer_selem_channel_id_t, double, int);
Packit 229ac0
Packit 229ac0
	if (!(control->flags & HAS_VOLUME_1))
Packit 229ac0
		channels = LEFT;
Packit 229ac0
	if (control->flags & TYPE_PVOLUME)
Packit 229ac0
		set_func = set_normalized_playback_volume;
Packit 229ac0
	else
Packit 229ac0
		set_func = set_normalized_capture_volume;
Packit 229ac0
	if (channels & LEFT)
Packit 229ac0
		set_func(control->elem, control->volume_channels[0], value / 100.0, 0);
Packit 229ac0
	if (channels & RIGHT)
Packit 229ac0
		set_func(control->elem, control->volume_channels[1], value / 100.0, 0);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static double clamp_volume(double v)
Packit 229ac0
{
Packit 229ac0
	if (v < 0)
Packit 229ac0
		return 0;
Packit 229ac0
	if (v > 1)
Packit 229ac0
		return 1;
Packit 229ac0
	return v;
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_volume_relative(struct control *control, int delta, unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	double (*get_func)(snd_mixer_elem_t *, snd_mixer_selem_channel_id_t);
Packit 229ac0
	int (*set_func)(snd_mixer_elem_t *, snd_mixer_selem_channel_id_t, double, int);
Packit 229ac0
	double left, right;
Packit 229ac0
	int dir;
Packit 229ac0
Packit 229ac0
	if (!(control->flags & HAS_VOLUME_1))
Packit 229ac0
		channels = LEFT;
Packit 229ac0
	if (control->flags & TYPE_PVOLUME) {
Packit 229ac0
		get_func = get_normalized_playback_volume;
Packit 229ac0
		set_func = set_normalized_playback_volume;
Packit 229ac0
	} else {
Packit 229ac0
		get_func = get_normalized_capture_volume;
Packit 229ac0
		set_func = set_normalized_capture_volume;
Packit 229ac0
	}
Packit 229ac0
	if (channels & LEFT)
Packit 229ac0
		left = get_func(control->elem, control->volume_channels[0]);
Packit 229ac0
	if (channels & RIGHT)
Packit 229ac0
		right = get_func(control->elem, control->volume_channels[1]);
Packit 229ac0
	dir = delta > 0 ? 1 : -1;
Packit 229ac0
	if (channels & LEFT) {
Packit 229ac0
		left = clamp_volume(left + delta / 100.0);
Packit 229ac0
		set_func(control->elem, control->volume_channels[0], left, dir);
Packit 229ac0
	}
Packit 229ac0
	if (channels & RIGHT) {
Packit 229ac0
		right = clamp_volume(right + delta / 100.0);
Packit 229ac0
		set_func(control->elem, control->volume_channels[1], right, dir);
Packit 229ac0
	}
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_control_to_percent(int value, unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	struct control *control;
Packit 229ac0
Packit 229ac0
	control = get_focus_control(TYPE_PVOLUME | TYPE_CVOLUME | TYPE_ENUM);
Packit 229ac0
	if (!control)
Packit 229ac0
		return;
Packit 229ac0
	if (control->flags & TYPE_ENUM)
Packit 229ac0
		change_enum_to_percent(control, value);
Packit 229ac0
	else
Packit 229ac0
		change_volume_to_percent(control, value, channels);
Packit 229ac0
	display_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void change_control_relative(int delta, unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	struct control *control;
Packit 229ac0
Packit 229ac0
	control = get_focus_control(TYPE_PVOLUME | TYPE_CVOLUME | TYPE_ENUM);
Packit 229ac0
	if (!control)
Packit 229ac0
		return;
Packit 229ac0
	if (control->flags & TYPE_ENUM)
Packit 229ac0
		change_enum_relative(control, delta);
Packit 229ac0
	else
Packit 229ac0
		change_volume_relative(control, delta, channels);
Packit 229ac0
	display_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void toggle_switches(unsigned int type, unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	struct control *control;
Packit 229ac0
	unsigned int switch_1_mask;
Packit 229ac0
	int (*get_func)(snd_mixer_elem_t *, snd_mixer_selem_channel_id_t, int *);
Packit 229ac0
	int (*set_func)(snd_mixer_elem_t *, snd_mixer_selem_channel_id_t, int);
Packit 229ac0
	snd_mixer_selem_channel_id_t channel_ids[2];
Packit 229ac0
	int left, right;
Packit 229ac0
	int err;
Packit 229ac0
Packit 229ac0
	control = get_focus_control(type);
Packit 229ac0
	if (!control)
Packit 229ac0
		return;
Packit 229ac0
	if (type == TYPE_PSWITCH) {
Packit 229ac0
		switch_1_mask = HAS_PSWITCH_1;
Packit 229ac0
		get_func = snd_mixer_selem_get_playback_switch;
Packit 229ac0
		set_func = snd_mixer_selem_set_playback_switch;
Packit 229ac0
		channel_ids[0] = control->pswitch_channels[0];
Packit 229ac0
		channel_ids[1] = control->pswitch_channels[1];
Packit 229ac0
	} else {
Packit 229ac0
		switch_1_mask = HAS_CSWITCH_1;
Packit 229ac0
		get_func = snd_mixer_selem_get_capture_switch;
Packit 229ac0
		set_func = snd_mixer_selem_set_capture_switch;
Packit 229ac0
		channel_ids[0] = control->cswitch_channels[0];
Packit 229ac0
		channel_ids[1] = control->cswitch_channels[1];
Packit 229ac0
	}
Packit 229ac0
	if (!(control->flags & switch_1_mask))
Packit 229ac0
		channels = LEFT;
Packit 229ac0
	if (channels & LEFT) {
Packit 229ac0
		err = get_func(control->elem, channel_ids[0], &left);
Packit 229ac0
		if (err < 0)
Packit 229ac0
			return;
Packit 229ac0
	}
Packit 229ac0
	if (channels & RIGHT) {
Packit 229ac0
		err = get_func(control->elem, channel_ids[1], &right);
Packit 229ac0
		if (err < 0)
Packit 229ac0
			return;
Packit 229ac0
	}
Packit 229ac0
	if (channels & LEFT)
Packit 229ac0
		set_func(control->elem, channel_ids[0], !left);
Packit 229ac0
	if (channels & RIGHT)
Packit 229ac0
		set_func(control->elem, channel_ids[1], !right);
Packit 229ac0
	display_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void toggle_mute(unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	toggle_switches(TYPE_PSWITCH, channels);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void toggle_capture(unsigned int channels)
Packit 229ac0
{
Packit 229ac0
	toggle_switches(TYPE_CSWITCH, channels);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void balance_volumes(void)
Packit 229ac0
{
Packit 229ac0
	struct control *control;
Packit 229ac0
	double left, right;
Packit 229ac0
Packit 229ac0
	control = get_focus_control(TYPE_PVOLUME | TYPE_CVOLUME);
Packit 229ac0
	if (!control || !(control->flags & HAS_VOLUME_1))
Packit 229ac0
		return;
Packit 229ac0
	if (control->flags & TYPE_PVOLUME) {
Packit 229ac0
		left = get_normalized_playback_volume(control->elem, control->volume_channels[0]);
Packit 229ac0
		right = get_normalized_playback_volume(control->elem, control->volume_channels[1]);
Packit 229ac0
	} else {
Packit 229ac0
		left = get_normalized_capture_volume(control->elem, control->volume_channels[0]);
Packit 229ac0
		right = get_normalized_capture_volume(control->elem, control->volume_channels[1]);
Packit 229ac0
	}
Packit 229ac0
	left = (left + right) / 2;
Packit 229ac0
	if (control->flags & TYPE_PVOLUME) {
Packit 229ac0
		set_normalized_playback_volume(control->elem, control->volume_channels[0], left, 0);
Packit 229ac0
		set_normalized_playback_volume(control->elem, control->volume_channels[1], left, 0);
Packit 229ac0
	} else {
Packit 229ac0
		set_normalized_capture_volume(control->elem, control->volume_channels[0], left, 0);
Packit 229ac0
		set_normalized_capture_volume(control->elem, control->volume_channels[1], left, 0);
Packit 229ac0
	}
Packit 229ac0
	display_controls();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void on_handle_key(int key)
Packit 229ac0
{
Packit 229ac0
	switch (key) {
Packit 229ac0
	case 27:
Packit 229ac0
	case KEY_CANCEL:
Packit 229ac0
	case KEY_F(10):
Packit 229ac0
		mixer_widget.close();
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(1):
Packit 229ac0
	case KEY_HELP:
Packit 229ac0
	case 'H':
Packit 229ac0
	case 'h':
Packit 229ac0
	case '?':
Packit 229ac0
		show_help();
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(2):
Packit 229ac0
	case '/':
Packit 229ac0
		create_proc_files_list();
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(3):
Packit 229ac0
		set_view_mode(VIEW_MODE_PLAYBACK);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(4):
Packit 229ac0
		set_view_mode(VIEW_MODE_CAPTURE);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(5):
Packit 229ac0
		set_view_mode(VIEW_MODE_ALL);
Packit 229ac0
		break;
Packit 229ac0
	case '\t':
Packit 229ac0
		set_view_mode((enum view_mode)((view_mode + 1) % VIEW_MODE_COUNT));
Packit 229ac0
		break;
Packit 229ac0
	case KEY_F(6):
Packit 229ac0
	case 'S':
Packit 229ac0
	case 's':
Packit 229ac0
		create_card_select_list();
Packit 229ac0
		break;
Packit 229ac0
	case KEY_REFRESH:
Packit 229ac0
	case 12:
Packit 229ac0
	case 'L':
Packit 229ac0
	case 'l':
Packit 229ac0
		clearok(mixer_widget.window, TRUE);
Packit 229ac0
		display_controls();
Packit 229ac0
		break;
Packit 229ac0
	case KEY_LEFT:
Packit 229ac0
	case 'P':
Packit 229ac0
	case 'p':
Packit 229ac0
		if (focus_control_index > 0) {
Packit 229ac0
			--focus_control_index;
Packit 229ac0
			refocus_control();
Packit 229ac0
		}
Packit 229ac0
		break;
Packit 229ac0
	case KEY_RIGHT:
Packit 229ac0
	case 'N':
Packit 229ac0
	case 'n':
Packit 229ac0
		if (focus_control_index < controls_count - 1) {
Packit 229ac0
			++focus_control_index;
Packit 229ac0
			refocus_control();
Packit 229ac0
		}
Packit 229ac0
		break;
Packit 229ac0
	case KEY_PPAGE:
Packit 229ac0
		change_control_relative(5, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_NPAGE:
Packit 229ac0
		change_control_relative(-5, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
#if 0
Packit 229ac0
	case KEY_BEG:
Packit 229ac0
	case KEY_HOME:
Packit 229ac0
		change_control_to_percent(100, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
#endif
Packit 229ac0
	case KEY_LL:
Packit 229ac0
	case KEY_END:
Packit 229ac0
		change_control_to_percent(0, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_UP:
Packit 229ac0
	case '+':
Packit 229ac0
	case 'K':
Packit 229ac0
	case 'k':
Packit 229ac0
	case 'W':
Packit 229ac0
	case 'w':
Packit 229ac0
		change_control_relative(1, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_DOWN:
Packit 229ac0
	case '-':
Packit 229ac0
	case 'J':
Packit 229ac0
	case 'j':
Packit 229ac0
	case 'X':
Packit 229ac0
	case 'x':
Packit 229ac0
		change_control_relative(-1, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case '0': case '1': case '2': case '3': case '4':
Packit 229ac0
	case '5': case '6': case '7': case '8': case '9':
Packit 229ac0
		change_control_to_percent((key - '0') * 10, LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case 'Q':
Packit 229ac0
	case 'q':
Packit 229ac0
		change_control_relative(1, LEFT);
Packit 229ac0
		break;
Packit 229ac0
	case 'Y':
Packit 229ac0
	case 'y':
Packit 229ac0
	case 'Z':
Packit 229ac0
	case 'z':
Packit 229ac0
		change_control_relative(-1, LEFT);
Packit 229ac0
		break;
Packit 229ac0
	case 'E':
Packit 229ac0
	case 'e':
Packit 229ac0
		change_control_relative(1, RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case 'C':
Packit 229ac0
	case 'c':
Packit 229ac0
		change_control_relative(-1, RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case 'M':
Packit 229ac0
	case 'm':
Packit 229ac0
		toggle_mute(LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case 'B':
Packit 229ac0
	case 'b':
Packit 229ac0
	case '=':
Packit 229ac0
		balance_volumes();
Packit 229ac0
		break;
Packit 229ac0
	case '<':
Packit 229ac0
	case ',':
Packit 229ac0
		toggle_mute(LEFT);
Packit 229ac0
		break;
Packit 229ac0
	case '>':
Packit 229ac0
	case '.':
Packit 229ac0
		toggle_mute(RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case ' ':
Packit 229ac0
		toggle_capture(LEFT | RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_IC:
Packit 229ac0
	case ';':
Packit 229ac0
		toggle_capture(LEFT);
Packit 229ac0
		break;
Packit 229ac0
	case KEY_DC:
Packit 229ac0
	case '\'':
Packit 229ac0
		toggle_capture(RIGHT);
Packit 229ac0
		break;
Packit 229ac0
	}
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void create(void)
Packit 229ac0
{
Packit 229ac0
	static const char title[] = " AlsaMixer v" SND_UTIL_VERSION_STR " ";
Packit 229ac0
Packit 229ac0
	widget_init(&mixer_widget, screen_lines, screen_cols, 0, 0,
Packit 229ac0
		    attr_mixer_frame, WIDGET_BORDER);
Packit 229ac0
	if (screen_cols >= (sizeof(title) - 1) + 2) {
Packit 229ac0
		wattrset(mixer_widget.window, attr_mixer_active);
Packit 229ac0
		mvwaddstr(mixer_widget.window, 0, (screen_cols - (sizeof(title) - 1)) / 2, title);
Packit 229ac0
	}
Packit 229ac0
	init_mixer_layout();
Packit 229ac0
	display_card_info();
Packit 229ac0
	set_view_mode(view_mode);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void on_window_size_changed(void)
Packit 229ac0
{
Packit 229ac0
	create();
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
static void on_close(void)
Packit 229ac0
{
Packit 229ac0
	widget_free(&mixer_widget);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
void mixer_shutdown(void)
Packit 229ac0
{
Packit 229ac0
	free_controls();
Packit 229ac0
	if (mixer)
Packit 229ac0
		snd_mixer_close(mixer);
Packit 229ac0
	if (current_selem_id)
Packit 229ac0
		snd_mixer_selem_id_free(current_selem_id);
Packit 229ac0
}
Packit 229ac0
Packit 229ac0
struct widget mixer_widget = {
Packit 229ac0
	.handle_key = on_handle_key,
Packit 229ac0
	.window_size_changed = on_window_size_changed,
Packit 229ac0
	.close = on_close,
Packit 229ac0
};
Packit 229ac0
Packit 229ac0
void create_mixer_widget(void)
Packit 229ac0
{
Packit 229ac0
	create();
Packit 229ac0
}