mirror of
https://github.com/alsa-project/alsa-utils
synced 2025-01-10 08:26:42 +01:00
269 lines
5.7 KiB
C
269 lines
5.7 KiB
C
|
/*
|
||
|
* card_select.c - select a card by list or device name
|
||
|
* Copyright (c) Clemens Ladisch <clemens@ladisch.de>
|
||
|
*
|
||
|
* This program 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.
|
||
|
*
|
||
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#include "aconfig.h"
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <assert.h>
|
||
|
#include <alsa/asoundlib.h>
|
||
|
#include <menu.h>
|
||
|
#include "gettext_curses.h"
|
||
|
#include "die.h"
|
||
|
#include "mem.h"
|
||
|
#include "utils.h"
|
||
|
#include "colors.h"
|
||
|
#include "widget.h"
|
||
|
#include "mixer_widget.h"
|
||
|
#include "device_name.h"
|
||
|
#include "card_select.h"
|
||
|
|
||
|
struct card {
|
||
|
struct card *next;
|
||
|
char *indexstr;
|
||
|
char *name;
|
||
|
char *device_name;
|
||
|
};
|
||
|
|
||
|
static struct widget list_widget;
|
||
|
static struct card first_card;
|
||
|
static ITEM **items;
|
||
|
static MENU *menu;
|
||
|
static ITEM *initial_item;
|
||
|
|
||
|
static void on_key_enter(void)
|
||
|
{
|
||
|
ITEM *item = current_item(menu);
|
||
|
if (item) {
|
||
|
struct card *card = item_userptr(item);
|
||
|
if (card->device_name) {
|
||
|
if (select_card_by_name(card->device_name))
|
||
|
list_widget.close();
|
||
|
} else {
|
||
|
create_device_name_form();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void on_menu_key(int key)
|
||
|
{
|
||
|
static const struct {
|
||
|
int key;
|
||
|
int request;
|
||
|
} key_map[] = {
|
||
|
{ KEY_DOWN, REQ_DOWN_ITEM },
|
||
|
{ KEY_UP, REQ_UP_ITEM },
|
||
|
{ KEY_HOME, REQ_FIRST_ITEM },
|
||
|
{ KEY_NPAGE, REQ_SCR_DPAGE },
|
||
|
{ KEY_PPAGE, REQ_SCR_UPAGE },
|
||
|
{ KEY_BEG, REQ_FIRST_ITEM },
|
||
|
{ KEY_END, REQ_LAST_ITEM },
|
||
|
};
|
||
|
unsigned int i;
|
||
|
|
||
|
for (i = 0; i < ARRAY_SIZE(key_map); ++i)
|
||
|
if (key_map[i].key == key) {
|
||
|
menu_driver(menu, key_map[i].request);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void on_handle_key(int key)
|
||
|
{
|
||
|
switch (key) {
|
||
|
case 27:
|
||
|
case KEY_CANCEL:
|
||
|
case 'q':
|
||
|
case 'Q':
|
||
|
list_widget.close();
|
||
|
break;
|
||
|
case 10:
|
||
|
case 13:
|
||
|
case KEY_ENTER:
|
||
|
on_key_enter();
|
||
|
break;
|
||
|
default:
|
||
|
on_menu_key(key);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static bool create(void)
|
||
|
{
|
||
|
int rows, columns;
|
||
|
const char *title;
|
||
|
|
||
|
if (screen_lines < 3 || screen_cols < 10) {
|
||
|
beep();
|
||
|
list_widget.close();
|
||
|
return FALSE;
|
||
|
}
|
||
|
scale_menu(menu, &rows, &columns);
|
||
|
rows += 2;
|
||
|
columns += 2;
|
||
|
if (rows > screen_lines)
|
||
|
rows = screen_lines;
|
||
|
if (columns > screen_cols)
|
||
|
columns = screen_cols;
|
||
|
|
||
|
widget_init(&list_widget, rows, columns, SCREEN_CENTER, SCREEN_CENTER,
|
||
|
attr_menu, WIDGET_BORDER | WIDGET_SUBWINDOW);
|
||
|
|
||
|
title = _("Sound Card");
|
||
|
mvwprintw(list_widget.window, 0, (columns - 2 - get_mbs_width(title)) / 2, " %s ", title);
|
||
|
set_menu_win(menu, list_widget.window);
|
||
|
set_menu_sub(menu, list_widget.subwindow);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static void on_window_size_changed(void)
|
||
|
{
|
||
|
unpost_menu(menu);
|
||
|
if (!create())
|
||
|
return;
|
||
|
post_menu(menu);
|
||
|
}
|
||
|
|
||
|
static void on_close(void)
|
||
|
{
|
||
|
unsigned int i;
|
||
|
struct card *card, *next_card;
|
||
|
|
||
|
unpost_menu(menu);
|
||
|
free_menu(menu);
|
||
|
for (i = 0; items[i]; ++i)
|
||
|
free_item(items[i]);
|
||
|
free(items);
|
||
|
for (card = first_card.next; card; card = next_card) {
|
||
|
next_card = card->next;
|
||
|
free(card->indexstr);
|
||
|
free(card->name);
|
||
|
free(card->device_name);
|
||
|
free(card);
|
||
|
}
|
||
|
widget_free(&list_widget);
|
||
|
}
|
||
|
|
||
|
void close_card_select_list(void)
|
||
|
{
|
||
|
on_close();
|
||
|
}
|
||
|
|
||
|
static struct widget list_widget = {
|
||
|
.handle_key = on_handle_key,
|
||
|
.window_size_changed = on_window_size_changed,
|
||
|
.close = on_close,
|
||
|
};
|
||
|
|
||
|
static int get_cards(void)
|
||
|
{
|
||
|
int count, number, err;
|
||
|
snd_ctl_t *ctl;
|
||
|
snd_ctl_card_info_t *info;
|
||
|
char buf[16];
|
||
|
struct card *card, *prev_card;
|
||
|
|
||
|
first_card.indexstr = "-";
|
||
|
first_card.name = _("(default)");
|
||
|
first_card.device_name = "default";
|
||
|
count = 1;
|
||
|
|
||
|
snd_ctl_card_info_alloca(&info);
|
||
|
prev_card = &first_card;
|
||
|
number = -1;
|
||
|
for (;;) {
|
||
|
err = snd_card_next(&number);
|
||
|
if (err < 0)
|
||
|
fatal_alsa_error(_("cannot enumerate sound cards"), err);
|
||
|
if (number < 0)
|
||
|
break;
|
||
|
sprintf(buf, "hw:%d", number);
|
||
|
err = snd_ctl_open(&ctl, buf, 0);
|
||
|
if (err < 0)
|
||
|
continue;
|
||
|
err = snd_ctl_card_info(ctl, info);
|
||
|
snd_ctl_close(ctl);
|
||
|
if (err < 0)
|
||
|
continue;
|
||
|
card = ccalloc(1, sizeof *card);
|
||
|
sprintf(buf, "%d", number);
|
||
|
card->indexstr = cstrdup(buf);
|
||
|
card->name = cstrdup(snd_ctl_card_info_get_name(info));
|
||
|
sprintf(buf, "hw:%d", number);
|
||
|
card->device_name = cstrdup(buf);
|
||
|
prev_card->next = card;
|
||
|
prev_card = card;
|
||
|
++count;
|
||
|
}
|
||
|
|
||
|
card = ccalloc(1, sizeof *card);
|
||
|
card->indexstr = cstrdup(" ");
|
||
|
card->name = cstrdup(_("enter device name..."));
|
||
|
prev_card->next = card;
|
||
|
++count;
|
||
|
|
||
|
return count;
|
||
|
}
|
||
|
|
||
|
static void create_list_items(int cards)
|
||
|
{
|
||
|
int i;
|
||
|
struct card *card;
|
||
|
ITEM *item;
|
||
|
|
||
|
initial_item = NULL;
|
||
|
items = ccalloc(cards + 1, sizeof(ITEM*));
|
||
|
i = 0;
|
||
|
for (card = &first_card; card; card = card->next) {
|
||
|
item = new_item(card->indexstr, card->name);
|
||
|
if (!item)
|
||
|
fatal_error("cannot create menu item");
|
||
|
set_item_userptr(item, card);
|
||
|
items[i++] = item;
|
||
|
if (!initial_item &&
|
||
|
mixer_device_name &&
|
||
|
(!card->device_name ||
|
||
|
!strcmp(card->device_name, mixer_device_name)))
|
||
|
initial_item = item;
|
||
|
}
|
||
|
assert(i == cards);
|
||
|
}
|
||
|
|
||
|
void create_card_select_list(void)
|
||
|
{
|
||
|
int cards;
|
||
|
|
||
|
cards = get_cards();
|
||
|
create_list_items(cards);
|
||
|
|
||
|
menu = new_menu(items);
|
||
|
if (!menu)
|
||
|
fatal_error("cannot create menu");
|
||
|
set_menu_fore(menu, attr_menu_selected);
|
||
|
set_menu_back(menu, attr_menu);
|
||
|
set_menu_mark(menu, NULL);
|
||
|
if (initial_item)
|
||
|
set_current_item(menu, initial_item);
|
||
|
set_menu_spacing(menu, 2, 1, 1);
|
||
|
menu_opts_on(menu, O_SHOWDESC);
|
||
|
|
||
|
if (!create())
|
||
|
return;
|
||
|
|
||
|
post_menu(menu);
|
||
|
}
|