#include #include #include #include #include #define VOLUME_MAX UINT16_MAX #define VOLUME_BAR_LEN 50 #define WIDTH 80 #define HEIGHT 10 static void context_state_callback(pa_context*, void *); static void get_sink_input_info_callback(pa_context *, const pa_sink_input_info*, int, void *); static void change_callback(pa_context* c, int success, void* userdate); static void print_sinks(void); static void print_volume(pa_volume_t, int); int cmp_sink_input_list(const void *, const void *); void get_input(void); void quit(void); typedef struct _sink_input_info { uint32_t index; uint32_t sink; char *name; char *pid; pa_volume_t vol; } sink_input_info; typedef struct _sink_info { uint32_t sink; char* name; pa_volume_t vol; } sink_info; static sink_info** sink_list = NULL; int sink_counter; uint32_t sink_max; static sink_input_info** sink_input_list = NULL; int sink_input_counter; int sink_input_max; static pa_mainloop_api *mainloop_api = NULL; static pa_context *context = NULL; // ncurses WINDOW *menu_win; int chooser; int startx; int starty; int main(int argc, char** argv) { sink_input_counter = 0; sink_input_max = 1; sink_counter = 0; sink_max = 1; sink_input_list = (sink_input_info**) calloc(sink_input_max, sizeof(sink_input_info*)); sink_list = (sink_info**) calloc(sink_max, sizeof(sink_info*)); chooser = 0; initscr(); clear(); noecho(); cbreak(); /* Line buffering disabled. pass on everything */ startx = (80 - WIDTH) / 2; starty = (24 - HEIGHT) / 2; menu_win = newwin(HEIGHT, WIDTH, starty, startx); keypad(menu_win, TRUE); mvprintw(0, 0, "Use arrow keys to go up and down, Press enter to select a choice"); refresh(); pa_mainloop *m = NULL; int ret = 1; if(!(m = pa_mainloop_new())) { printf("error: pa_mainloop_new() failed.\n"); return -1; } mainloop_api = pa_mainloop_get_api(m); if (!(context = pa_context_new(mainloop_api, "pa-sink-ctl"))) { printf("error: pa_context_new() failed.\n"); return -1; } pa_context_set_state_callback(context, context_state_callback, NULL); if (pa_context_connect(context, "tcp:127.0.0.1:4713", PA_CONTEXT_NOAUTOSPAWN, NULL)) { printf("error: pa_context_connect() failed.\n"); } if (pa_mainloop_run(m, &ret) < 0) { printf("error: pa_mainloop_run() failed.\n"); return -1; } return ret; } static void context_state_callback(pa_context *c, void *userdata) { switch (pa_context_get_state(c)) { case PA_CONTEXT_CONNECTING: // printf("connecting...\n"); break; case PA_CONTEXT_AUTHORIZING: // printf("authorizing...\n"); break; case PA_CONTEXT_SETTING_NAME: // printf("setting name\n"); break; case PA_CONTEXT_READY: // printf("Menue\n"); pa_operation_unref(pa_context_get_sink_input_info_list(c, get_sink_input_info_callback, NULL)); break; default: printf("unknown state\n"); break; } } static void get_sink_input_info_callback(pa_context *c, const pa_sink_input_info *i, int is_last, void *userdata) { char t[32], k[32]; //,cv[PA_CVOLUME_SNPRINT_MAX]; if (is_last < 0) { printf("Failed to get sink input information: %s\n", pa_strerror(pa_context_errno(c))); return; } if (is_last) { print_sinks(); get_input(); return; } if (!(i->client != PA_INVALID_INDEX)) return; snprintf(t, sizeof(t), "%u", i->owner_module); snprintf(k, sizeof(k), "%u", i->client); /* printf( "Sink Input #%u" "\tClient: %s" "\tSink: %u" "\tMute: %d" "\tVolume: %s" "\tname: %s" "\tpid: %s\n", i->index, i->client != PA_INVALID_INDEX ? k : "n/a", i->sink, i->mute, pa_cvolume_snprint(cv, sizeof(cv), &i->volume), pa_proplist_gets(i->proplist, "application.name"), pa_proplist_gets(i->proplist, "application.process.id")); */ const char *name = pa_proplist_gets(i->proplist, "application.name"); if (i->sink > sink_max) sink_max = i->sink; ++sink_input_counter; if (sink_input_counter >= sink_input_max) { sink_input_max*=2; sink_input_list = (sink_input_info**) realloc(sink_input_list, sizeof(sink_input_info*) * sink_input_max); } sink_input_list[sink_input_counter-1] = (sink_input_info*) calloc(1, sizeof(sink_input_info)); sink_input_list[sink_input_counter-1]->name = (char*) calloc(strlen(name) + 1, sizeof(char)); sink_input_list[sink_input_counter-1]->index = i->index; sink_input_list[sink_input_counter-1]->sink = i->sink; strncpy(sink_input_list[sink_input_counter-1]->name, name, strlen(name)); sink_input_list[sink_input_counter-1]->vol = pa_cvolume_avg(&i->volume); } void print_sinks(void) { int x, y, i; x = 2; y = 2; box(menu_win, 0, 0); // printf("print sinks: %d\n", sink_input_counter); // qsort(sink_input_list, sink_input_counter, sizeof(sink_input_info*), cmp_sink_input_list); for(i = 0; i < sink_input_counter; ++i) { if (i == chooser) wattron(menu_win, A_REVERSE); mvwprintw(menu_win, y+i, x, "%d\t%s\t", sink_input_list[i]->sink, sink_input_list[i]->name); if (i == chooser) wattroff(menu_win, A_REVERSE); print_volume(sink_input_list[i]->vol, y+i); } } void print_volume(pa_volume_t volume, int y) { int x = 20; unsigned int vol = (unsigned int) ( (((double)volume) / ((double)VOLUME_MAX)) * VOLUME_BAR_LEN ); mvwprintw(menu_win, y, x - 1 , "["); for (int i = 0; i < vol; ++i) mvwprintw(menu_win, y, x + i, "="); for (int i = vol; i < VOLUME_BAR_LEN - vol; ++i) mvwprintw(menu_win, y, x + i, " "); mvwprintw(menu_win, y, x + VOLUME_BAR_LEN, "]"); } int cmp_sink_input_list(const void *a, const void *b) { sink_input_info* sinka = *((sink_input_info**) a); sink_input_info* sinkb = *((sink_input_info**) b); if (sinka->sink < sinkb->sink) return -1; else if (sinka->sink > sinkb->sink) return 1; else return 0; } void get_input(void) { int c; uint32_t sink; c = wgetch(menu_win); switch (c) { case KEY_UP: if (chooser > 0) --chooser; break; case KEY_DOWN: if (chooser < sink_input_counter - 1) ++chooser; break; case KEY_LEFT: break; case KEY_RIGHT: break; case 32: if (sink_input_list[chooser]->sink < sink_max) sink = sink_input_list[chooser]->sink + 1; else sink = 0; pa_operation_unref( pa_context_move_sink_input_by_index( context, sink_input_list[chooser]->index, sink, change_callback, NULL)); return; break; default: printf("key: %d\n", c); quit(); break; } sink_input_counter = 0; pa_operation_unref(pa_context_get_sink_input_info_list(context, get_sink_input_info_callback, NULL)); } void quit(void) { clrtoeol(); refresh(); endwin(); exit(0); } static void change_callback(pa_context* c, int success, void* userdate) { sink_input_counter = 0; pa_operation_unref(pa_context_get_sink_input_info_list(context, get_sink_input_info_callback, NULL)); }