/* * card_select.c - select a card by list or device name * Copyright (c) Clemens Ladisch * * 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 . */ #include "aconfig.h" #include #include #include #include #include #include "gettext_curses.h" #include "die.h" #include "mem.h" #include "utils.h" #include "colors.h" #include "widget.h" #include "menu_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_handle_key(int key) { switch (menu_widget_handle_key(menu, key)) { case KEY_ENTER: on_key_enter(); break; case KEY_CANCEL: list_widget.close(); 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); }