aboutsummaryrefslogtreecommitdiff
path: root/audio/volume.app/volume.app.c.mod
diff options
context:
space:
mode:
Diffstat (limited to 'audio/volume.app/volume.app.c.mod')
-rw-r--r--audio/volume.app/volume.app.c.mod311
1 files changed, 311 insertions, 0 deletions
diff --git a/audio/volume.app/volume.app.c.mod b/audio/volume.app/volume.app.c.mod
new file mode 100644
index 0000000000000..9485ef8103d25
--- /dev/null
+++ b/audio/volume.app/volume.app.c.mod
@@ -0,0 +1,311 @@
+/* volume.app.c */
+
+/* Volume.app -- a simple volume control
+ *
+ * Copyright (C) 2000
+ * Daniel Richard G. <skunk@mit.edu>,
+ * timecop <timecop@japan.co.jp>
+ *
+ * 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, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <getopt.h>
+#include <signal.h>
+#include <unistd.h>
+#include <sys/stat.h>
+#include <X11/X.h>
+#include <X11/Xlib.h>
+
+#include "common.h"
+#include "knob.h"
+#include "misc.h"
+#include "mixer.h"
+
+static int source = DEFAULT_SOURCE;
+static char *mixer_device = NULL;
+static bool list_sources = false;
+
+static Display *display;
+static int display_height;
+
+static double prev_button1_press_time = 0.0;
+static bool button1_pressed = false;
+static int mouse_drag_home_x;
+static int mouse_drag_home_y;
+
+static void
+signal_catch(int sig)
+{
+ switch (sig)
+ {
+ case SIGUSR1:
+ knob_turn(-0.1);
+ break;
+
+ case SIGUSR2:
+ knob_turn(0.1);
+ break;
+
+ default:
+ abort();
+ break;
+ }
+
+ signal(sig, signal_catch);
+}
+
+static void
+button_press_event(XButtonEvent *event)
+{
+ double button_press_time = get_current_time();
+
+ switch (event->button)
+ {
+ /* Left click
+ */
+ case 1:
+ knob_grab();
+ if ((button_press_time - prev_button1_press_time) <= MAX_DOUBLE_CLICK_TIME)
+ {
+ /* Double-click
+ */
+ knob_toggle_mute();
+ prev_button1_press_time = 0.0;
+ } else
+ system("amixer set Master unmute");
+ prev_button1_press_time = button_press_time;
+ button1_pressed = true;
+ mouse_drag_home_x = event->x;
+ mouse_drag_home_y = event->y;
+ break;
+
+ case 3:
+ /*
+ * Right click
+ */
+
+ /* popup menu? */
+
+ break;
+
+ case BUTTON_WHEEL_UP:
+ knob_turn(0.05);
+ break;
+
+ case BUTTON_WHEEL_DOWN:
+ knob_turn(-0.05);
+ break;
+
+ default:
+ break;
+ }
+}
+
+static void
+button_release_event(XButtonEvent *event)
+{
+ if (event->button == 1)
+ {
+ knob_release();
+ button1_pressed = false;
+ }
+}
+
+static void
+mouse_motion_event(XMotionEvent *event)
+{
+ if (button1_pressed)
+ {
+ if ((event->x == mouse_drag_home_x) && (event->y == mouse_drag_home_y))
+ {
+ /* This motion event was generated by an earlier
+ * XWarpPointer() call, so ignore it.
+ */
+ return;
+ }
+
+ if (event->y != mouse_drag_home_y)
+ {
+ int delta_y = mouse_drag_home_y - event->y;
+ float delta_volume = (float)delta_y / (float)display_height;
+ knob_turn(delta_volume);
+ }
+
+ /* Keep mouse pointer on the knob. Note that this call
+ * will generate a bogus motion event (see above)
+ */
+ XWarpPointer(
+ display,
+ None,
+ event->window,
+ event->x, event->y,
+ 0, 0,
+ mouse_drag_home_x, mouse_drag_home_y);
+ }
+}
+
+#define HELP_TEXT \
+ "Volume.app " VERSION "\n" \
+ "usage:\n" \
+ " -c <n> source to control [1]\n" \
+ " (see -l option)\n" \
+ " -d <dev> mixer device [" DEFAULT_MIXER_DEVICE "]\n" \
+ " -h print this help\n" \
+ " -l print list of available sound sources\n"
+
+void
+parse_cli_options(int argc, char **argv)
+{
+ int opt;
+
+ while ((opt = getopt(argc, argv, "c:hl")) != EOF)
+ {
+ switch (opt)
+ {
+ case 'c':
+ if (optarg != NULL)
+ source = strtod(optarg, NULL) - 1;
+ break;
+
+ case 'd':
+ if (optarg != NULL)
+ {
+ if (mixer_device != NULL)
+ free(mixer_device);
+ mixer_device = strdup(optarg);
+ }
+ break;
+
+ case 'h':
+ fputs(HELP_TEXT, stdout);
+ exit(0);
+ break;
+
+ case 'l':
+ list_sources = true;
+ break;
+
+ default:
+ break;
+ }
+ }
+}
+
+int
+main(int argc, char **argv)
+{
+ char *display_name;
+ XEvent event;
+ int idle_tick = 0;
+
+#ifdef DEBUG
+ fputs("**** Volume.app: debug build starting ****\n", stderr);
+#endif
+
+ parse_cli_options(argc, argv);
+
+ display_name = getenv("DISPLAY");
+ display = XOpenDisplay(display_name);
+ if (display == NULL)
+ {
+ if (display_name == NULL)
+ fputs("Unable to open display\n", stderr);
+ else
+ fprintf(stderr, "Unable to open display \"%s\"\n", display_name);
+ return EXIT_FAILURE;
+ }
+ XFlush(display);
+
+ display_height = (float)DisplayHeight(display, DefaultScreen(display));
+
+ if (mixer_device == NULL)
+ mixer_device = strdup(DEFAULT_MIXER_DEVICE);
+ mixer_init(mixer_device);
+ free(mixer_device);
+
+ if (list_sources)
+ {
+ mixer_print_sources();
+ return EXIT_SUCCESS;
+ }
+
+ if ((source < 0) || (source >= mixer_get_source_count()))
+ {
+ fprintf(stderr, "Invalid source number: %d\n", source + 1);
+ return EXIT_FAILURE;
+ }
+ mixer_set_source(source);
+
+ knob_init(display);
+ knob_update();
+
+ signal(SIGUSR1, signal_catch);
+ signal(SIGUSR2, signal_catch);
+
+ /* Main event loop
+ */
+ while (true)
+ {
+ if (button1_pressed || (XPending(display) > 0))
+ {
+ XNextEvent(display, &event);
+
+ switch (event.type)
+ {
+ case Expose:
+ knob_redraw();
+ break;
+
+ case ButtonPress:
+ button_press_event(&event.xbutton);
+ idle_tick = 0;
+ break;
+
+ case ButtonRelease:
+ button_release_event(&event.xbutton);
+ idle_tick = 0;
+ break;
+
+ case MotionNotify:
+ mouse_motion_event(&event.xmotion);
+ idle_tick = 0;
+ break;
+
+ case DestroyNotify:
+ XCloseDisplay(display);
+ goto main_event_loop_exit;
+
+ default:
+ break;
+ }
+ } else {
+ knob_update();
+ usleep(100000);
+ ++idle_tick;
+ }
+ }
+ main_event_loop_exit:
+
+ return EXIT_SUCCESS;
+}
+
+/* end volume.app.c */