2015-09-16 16:23:38 +00:00
|
|
|
/*
|
|
|
|
* http://code.arp242.net/find-cursor
|
2017-03-23 05:02:56 +00:00
|
|
|
* Copyright © 2015-2017 Martin Tournoij <martin@arp242.net>
|
2015-09-16 16:23:38 +00:00
|
|
|
* See below for full copyright
|
|
|
|
*/
|
|
|
|
|
2016-03-07 20:19:25 +00:00
|
|
|
#include <getopt.h>
|
|
|
|
#include <stdint.h>
|
2015-09-16 16:23:38 +00:00
|
|
|
#include <stdio.h>
|
2016-03-07 20:19:25 +00:00
|
|
|
#include <stdlib.h>
|
2015-09-16 16:23:38 +00:00
|
|
|
#include <string.h>
|
2016-03-07 20:19:25 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2015-09-16 16:23:38 +00:00
|
|
|
#include <X11/Xlib.h>
|
2016-03-07 20:19:25 +00:00
|
|
|
#include <X11/Xatom.h>
|
2015-09-18 14:42:06 +00:00
|
|
|
#include <X11/Xutil.h>
|
2016-03-07 20:19:25 +00:00
|
|
|
#include <X11/extensions/Xdamage.h>
|
|
|
|
#include <X11/extensions/Xrender.h>
|
|
|
|
#include <X11/extensions/shape.h>
|
|
|
|
|
|
|
|
void usage(char *name);
|
2017-03-23 05:02:56 +00:00
|
|
|
int parse_num(int ch, char *opt, char *name);
|
2016-03-07 20:19:25 +00:00
|
|
|
void draw(char *name, int size, int step, int speed, int line_width, char *color_name);
|
|
|
|
|
|
|
|
static struct option longopts[] = {
|
|
|
|
{"help", no_argument, NULL, 'h'},
|
|
|
|
{"size", required_argument, NULL, 's'},
|
|
|
|
{"step", required_argument, NULL, 't'},
|
|
|
|
{"speed", required_argument, NULL, 'p'},
|
|
|
|
{"line-width", required_argument, NULL, 'l'},
|
|
|
|
{"color", required_argument, NULL, 'c'},
|
|
|
|
{NULL, 0, NULL, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
void usage(char *name) {
|
|
|
|
printf("Usage: %s [-stplc]\n\n", name);
|
|
|
|
printf(" -h, --help Show this help.\n");
|
|
|
|
printf(" -s, --size Size in pixels.\n");
|
|
|
|
printf(" -t, --step ???.\n");
|
|
|
|
printf(" -p, --speed Animation speed.\n");
|
|
|
|
printf(" -l, --line-width Width of the lines in pixels.\n");
|
|
|
|
printf(" -c, --color Color; can either be an X11 color name or RGB as hex (i.e. #ff0055).\n");
|
|
|
|
printf("\n");
|
|
|
|
printf("The defaults:\n");
|
|
|
|
printf(" %s --size 220 --step 40 --speed 400 --line-width 2 --color black\n", name);
|
|
|
|
printf("Draw a circle\n");
|
|
|
|
printf(" %s --size 100 --step 1 --speed 20 --line-width 1 --color black\n", name);
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2017-03-23 05:02:56 +00:00
|
|
|
// Parse number from commandline, or show error and exit if it's not a number.
|
|
|
|
int parse_num(int ch, char *opt, char *name) {
|
2016-03-07 20:19:25 +00:00
|
|
|
char *end;
|
|
|
|
long result = strtol(optarg, &end, 10);
|
|
|
|
if (*end) {
|
|
|
|
fprintf(stderr, "%s: %d must be a number\n", name, ch);
|
|
|
|
usage(name);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2015-09-16 16:23:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char* argv[]) {
|
2016-03-07 20:19:25 +00:00
|
|
|
// Parse options
|
|
|
|
int size = 220;
|
|
|
|
int step = 40;
|
|
|
|
int speed = 400;
|
|
|
|
int line_width = 2;
|
|
|
|
char color_name[64] = "black";
|
|
|
|
|
|
|
|
int ch;
|
|
|
|
while ((ch = getopt_long(argc, argv, "hs:t:p:l:c:r:", longopts, NULL)) != -1)
|
|
|
|
switch (ch) {
|
|
|
|
case 's':
|
2017-03-23 05:02:56 +00:00
|
|
|
size = parse_num(ch, optarg, argv[0]);
|
2016-03-07 20:19:25 +00:00
|
|
|
break;
|
|
|
|
case 't':
|
2017-03-23 05:02:56 +00:00
|
|
|
step = parse_num(ch, optarg, argv[0]);
|
2016-03-07 20:19:25 +00:00
|
|
|
break;
|
|
|
|
case 'p':
|
2017-03-23 05:02:56 +00:00
|
|
|
speed = parse_num(ch, optarg, argv[0]);
|
2016-03-07 20:19:25 +00:00
|
|
|
break;
|
|
|
|
case 'l':
|
2017-03-23 05:02:56 +00:00
|
|
|
line_width = parse_num(ch, optarg, argv[0]);
|
2016-03-07 20:19:25 +00:00
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
strncpy(color_name, optarg, sizeof(color_name));
|
|
|
|
break;
|
|
|
|
case 'h':
|
|
|
|
usage(argv[0]);
|
|
|
|
exit(0);
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "%s: Unknown option: %c\n", argv[0], (char) ch);
|
|
|
|
usage(argv[0]);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
argc -= optind;
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
draw(argv[0], size, step, speed, line_width, color_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void draw(char *name, int size, int step, int speed, int line_width, char *color_name) {
|
2015-09-16 16:23:38 +00:00
|
|
|
// Setup display and such
|
|
|
|
char *display_name = getenv("DISPLAY");
|
|
|
|
if (!display_name) {
|
2016-03-07 20:19:25 +00:00
|
|
|
fprintf(stderr, "%s: cannot connect to X server '%s'\n", name, display_name);
|
2015-09-16 16:23:38 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
Display *display = XOpenDisplay(display_name);
|
|
|
|
int screen = DefaultScreen(display);
|
|
|
|
|
2017-03-23 05:02:56 +00:00
|
|
|
int shape_event_base, shape_error_base;
|
|
|
|
if (!XShapeQueryExtension(display, &shape_event_base, &shape_error_base)) {
|
|
|
|
fprintf(stderr, "%s: no XShape extension for display '%s'\n", name, display_name);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2015-09-16 16:23:38 +00:00
|
|
|
// Get the mouse cursor position
|
|
|
|
int win_x, win_y, root_x, root_y = 0;
|
|
|
|
unsigned int mask = 0;
|
|
|
|
Window child_win, root_win;
|
|
|
|
XQueryPointer(display, XRootWindow(display, screen),
|
|
|
|
&child_win, &root_win,
|
|
|
|
&root_x, &root_y, &win_x, &win_y, &mask);
|
|
|
|
|
|
|
|
// Create a window at the mouse position
|
|
|
|
XSetWindowAttributes window_attr;
|
|
|
|
window_attr.override_redirect = 1;
|
|
|
|
Window window = XCreateWindow(display, XRootWindow(display, screen),
|
|
|
|
root_x - size/2, root_y - size/2, // x, y position
|
|
|
|
size, size, // width, height
|
2017-03-23 05:02:56 +00:00
|
|
|
4, // border width
|
2015-09-16 16:23:38 +00:00
|
|
|
DefaultDepth(display, screen), // depth
|
|
|
|
CopyFromParent, // class
|
|
|
|
DefaultVisual(display, screen), // visual
|
|
|
|
CWOverrideRedirect, // valuemask
|
|
|
|
&window_attr // attributes
|
|
|
|
);
|
2017-03-23 05:02:56 +00:00
|
|
|
|
|
|
|
// Make round shaped window.
|
|
|
|
XGCValues xgcv;
|
|
|
|
Pixmap shape_mask = XCreatePixmap(display, window, size, size, 1);
|
|
|
|
GC part_shape = XCreateGC(display, shape_mask, 0, &xgcv);
|
|
|
|
XSetForeground(display, part_shape, 0);
|
|
|
|
XFillRectangle(display, shape_mask, part_shape, 0, 0, size, size);
|
|
|
|
XSetForeground(display, part_shape, 1);
|
|
|
|
XFillArc(display, shape_mask, part_shape,
|
|
|
|
0, 0, // x, y position
|
|
|
|
size, size, // Size
|
|
|
|
0, 360 * 64); // Make it a full circle
|
|
|
|
|
|
|
|
XShapeCombineMask(display, window, ShapeBounding, 0,0, shape_mask, ShapeSet);
|
|
|
|
XShapeCombineMask(display, window, ShapeClip, 0,0, shape_mask, ShapeSet);
|
|
|
|
XFreePixmap(display, shape_mask);
|
|
|
|
|
|
|
|
// Input region is small so we can pass input events.
|
|
|
|
XRectangle rect;
|
|
|
|
XserverRegion region = XFixesCreateRegion(display, &rect, 1);
|
|
|
|
XFixesSetWindowShapeRegion(display, window, ShapeInput, 0, 0, region);
|
|
|
|
XFixesDestroyRegion(display, region);
|
|
|
|
|
|
|
|
// Set attrs
|
2015-09-16 16:23:38 +00:00
|
|
|
XMapWindow(display, window);
|
2015-09-18 14:42:06 +00:00
|
|
|
XStoreName(display, window, "find-cursor");
|
|
|
|
|
|
|
|
XClassHint *class = XAllocClassHint();
|
|
|
|
class->res_name = "find-cursor";
|
|
|
|
class->res_class = "find-cursor";
|
|
|
|
XSetClassHint(display, window, class);
|
|
|
|
XFree(class);
|
2015-09-16 16:23:38 +00:00
|
|
|
|
|
|
|
// Keep the window on top
|
|
|
|
XEvent e;
|
|
|
|
memset(&e, 0, sizeof(e));
|
|
|
|
e.xclient.type = ClientMessage;
|
|
|
|
e.xclient.message_type = XInternAtom(display, "_NET_WM_STATE", False);
|
|
|
|
e.xclient.display = display;
|
|
|
|
e.xclient.window = window;
|
|
|
|
e.xclient.format = 32;
|
|
|
|
e.xclient.data.l[0] = 1;
|
|
|
|
e.xclient.data.l[1] = XInternAtom(display, "_NET_WM_STATE_STAYS_ON_TOP", False);
|
|
|
|
XSendEvent(display, XRootWindow(display, screen), False, SubstructureRedirectMask, &e);
|
|
|
|
|
2017-03-23 05:02:56 +00:00
|
|
|
// Make sure compositing window managers don't apply shadows and such.
|
2016-03-07 20:19:25 +00:00
|
|
|
// https://specifications.freedesktop.org/wm-spec/1.4/ar01s05.html
|
|
|
|
// https://wiki.archlinux.org/index.php/Compton
|
|
|
|
Atom type_util = XInternAtom(display, "_NET_WM_WINDOW_TYPE", 0);
|
|
|
|
Atom type = XInternAtom(display, "_NET_WM_WINDOW_TYPE_MENU", 0);
|
2017-03-23 05:02:56 +00:00
|
|
|
XChangeProperty(display, window, type, XA_ATOM, 32, PropModeReplace,
|
|
|
|
(unsigned char *)&type_util, 1);
|
2016-03-07 20:19:25 +00:00
|
|
|
|
2015-09-16 16:23:38 +00:00
|
|
|
XRaiseWindow(display, window);
|
|
|
|
XFlush(display);
|
|
|
|
|
|
|
|
// Prepare to draw on this window
|
|
|
|
XGCValues values = { .graphics_exposures = False };
|
|
|
|
unsigned long valuemask = 0;
|
|
|
|
GC gc = XCreateGC(display, window, valuemask, &values);
|
|
|
|
|
|
|
|
Colormap colormap = DefaultColormap(display, screen);
|
|
|
|
XColor color;
|
|
|
|
XAllocNamedColor(display, colormap, color_name, &color, &color);
|
|
|
|
XSetForeground(display, gc, color.pixel);
|
|
|
|
XSetLineAttributes(display, gc, line_width, LineSolid, CapButt, JoinBevel);
|
|
|
|
|
|
|
|
// Draw the circles
|
|
|
|
for (int i=1; i<=size; i+=step) {
|
|
|
|
XDrawArc(display, window, gc,
|
|
|
|
size/2 - i/2, size/2 - i/2, // x, y position
|
|
|
|
i, i, // Size
|
|
|
|
0, 360 * 64); // Make it a full circle
|
|
|
|
|
|
|
|
XSync(display, False);
|
|
|
|
usleep(speed * 100);
|
|
|
|
}
|
2017-03-23 05:02:56 +00:00
|
|
|
|
2015-09-18 14:42:06 +00:00
|
|
|
XFreeGC(display, gc);
|
2015-09-16 16:23:38 +00:00
|
|
|
XCloseDisplay(display);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The MIT License (MIT)
|
2017-03-23 05:02:56 +00:00
|
|
|
*
|
|
|
|
* Copyright © 2015-2017 Martin Tournoij
|
|
|
|
*
|
2015-09-16 16:23:38 +00:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
2017-03-23 05:02:56 +00:00
|
|
|
*
|
2015-09-16 16:23:38 +00:00
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
2017-03-23 05:02:56 +00:00
|
|
|
*
|
2015-09-16 16:23:38 +00:00
|
|
|
* The software is provided "as is", without warranty of any kind, express or
|
|
|
|
* implied, including but not limited to the warranties of merchantability,
|
|
|
|
* fitness for a particular purpose and noninfringement. In no event shall the
|
|
|
|
* authors or copyright holders be liable for any claim, damages or other
|
|
|
|
* liability, whether in an action of contract, tort or otherwise, arising
|
|
|
|
* from, out of or in connection with the software or the use or other dealings
|
|
|
|
* in the software.
|
|
|
|
*/
|