ClickFlash: Mouse Click Highlighter for Linux
ClickFlash is a sleek, Python-driven Linux tutorial video tool crafted to enhance your screencasts with flair.
This lightweight command-line mouse click tool adds a vibrant, short-lived colored circle at your cursor’s position with every click, making it a perfect mouse click visualizer to spotlight your GUI interactions.
I’ve been loyal to FVWM2 as my X11 window manager since 1997, resisting the pull of other window managers or desktop environments like Gnome or KDE.
I cherish FVWM2’s speed, efficiency, and full control—qualities that make it a lightweight Linux tool tailored to my workflow.
Yet, it doesn’t offer a native way to highlight mouse clicks, leaving a gap that ClickFlash Linux tool fills effortlessly.
Whether you’re using FVWM2 or another X11 setup, this Python mouse click highlighter brings your tutorials to life.
Usage: please specify: size, duration and color size Diameter of the circle in pixels. duration How long the flash circle should be visible. color Color of the flash circle as hex color code, i.e. FF0000. Example: ./ClickFlash.py 33 0.1 FF0000
ClickFlash.py - Download
#!/usr/bin/python3 ################################################################################ # # ClickFlash v1.00 # # ClickFlash is a lightweight, Python-powered command-line tool designed to # make your tutorial videos pop. # With every mouse click, it flashes a vibrant, short-lived colored circle at # your cursor’s position, spotlighting exactly where you’re interacting in a GUI. # Perfect for educators, developers, or screencast creators. # # # 2025-03-22 bitman@bitmania.de https://bitmania.de # initial version # ################################################################################ from Xlib import X, display from Xlib.ext import shape from PIL import Image, ImageDraw import time import os import sys def create_circle_image(size, color): img = Image.new('RGBA', (size, size), (0, 0, 0, 0)) draw = ImageDraw.Draw(img) draw.ellipse( [0, 0, size-1, size-1], fill=color, # Custom color (R, G, B, A) outline=None ) return img def flash(posX, posY, size=50, duration=0.5, color=(255, 0, 0, 255)): d = display.Display() screen = d.screen() root = screen.root img = create_circle_image(size, color) width, height = img.size window = root.create_window( posX, posY, width, height, 0, screen.root_depth, X.InputOutput, screen.root_visual, colormap=screen.default_colormap, background_pixel=screen.black_pixel, event_mask=X.ExposureMask | X.StructureNotifyMask, override_redirect=True ) colormap = screen.default_colormap # Convert RGBA color to RGB for Xlib (ignore alpha) rgb_color = colormap.alloc_color( color[0] * 257, # Convert 0-255 to 0-65535 color[1] * 257, color[2] * 257 ) gc = window.create_gc(foreground=rgb_color.pixel, background=screen.black_pixel) mask = window.create_pixmap(width, height, 1) mask_gc = mask.create_gc(foreground=1, background=0) alpha = img.split()[3].convert('1') mask.put_pil_image(mask_gc, 0, 0, alpha) window.shape_mask(0, 0, 0, 0, mask) window.map() d.sync() drawn = False while True: event = d.next_event() if event.type == X.Expose or event.type == X.MapNotify: window.clear_area(0, 0, width, height) window.fill_rectangle(gc, 0, 0, width, height) d.sync() drawn = True break if drawn: time.sleep(duration) # Custom duration window.unmap() window.destroy() d.close() # Main execution if __name__ == "__main__": script_name = os.path.basename(__file__) # the name of this script (ommiting path) flash_color = (255, 0, 0, 255) # Default color: red (R, G, B, A) if len(sys.argv) <4: print('Usage:') print('') print('please specify: size, duration and color') print('') print('size Diameter of the circle in pixels.') print('') print('duration How long the flash circle should be visible.') print('') print('color Color of the flash circle as hex color code, i.e. FF0000.') print('') print('Example:') print(f' ./{script_name} 33 0.1 FF0000 ') print('') print('') sys.exit() else: flash_size = int(sys.argv[1]) print(f'flash_size : {flash_size}') flash_duration = float(sys.argv[2]) print(f'flash_duration : {flash_duration}') flash_color = sys.argv[3] print(f'flash_color : {flash_color}') if len(flash_color)!=6: print(f'invalid color {flash_color}') sys.exit() r = int(flash_color[0:2], 16) g = int(flash_color[2:4], 16) b = int(flash_color[4:6], 16) flash_color = (r, g, b, 255) print(f'flash_color : {flash_color}') # Mouse Event Listener with os.popen('xinput test-xi2 --root') as cmd: motion = False RawButtonRelease = False RawButtonPress = False posX, posY = 0, 0 for line in cmd: line = line.strip() #print(f'line: {line}') if 'EVENT type 6 (Motion)' in line: motion = True if motion and 'event:' in line: pos = line.replace(' ', '/').split('/')[1:3] posX, posY = int(float(pos[0])), int(float(pos[1])) motion = False if '(RawButtonRelease)' in line: RawButtonRelease = True if '(RawButtonPress)' in line: RawButtonPress = True #if not motion: # if RawButtonRelease: # print(f'---- {line}') # if 'detail: 1' in line: # flash(posX - int(flash_size/2), posY - int(flash_size/2), flash_size, flash_duration, (0, 255, 0, 255) ) # RawButtonRelease = False # if 'detail: 3' in line: # flash(posX - int(flash_size/2), posY - int(flash_size/2), flash_size, flash_duration, (0, 255, 0, 255) ) # RawButtonRelease = False if not motion: if RawButtonPress: #print(f'++++ {line}') if 'detail: 1' in line: flash(posX - int(flash_size/2), posY - int(flash_size/2), flash_size, flash_duration, flash_color) RawButtonPress = False if 'detail: 3' in line: flash(posX - int(flash_size/2), posY - int(flash_size/2), flash_size, flash_duration, flash_color) RawButtonPress = False