/* * Copyright 2021 elementary, Inc (https://elementary.io) * 2015 Rory J Sanderson * * 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 3 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 . */ public class Gala.Plugins.MaskCorners.Main : Gala.Plugin { private const int DEFAULT_CORNER_RADIUS = 6; private Gala.WindowManager? wm = null; private GLib.Settings settings; private int[] corner_radii; private List[] cornermasks; private Meta.Display display; public override void initialize (Gala.WindowManager wm) { this.wm = wm; display = wm.get_display (); settings = new GLib.Settings (Config.SCHEMA + ".mask-corners"); setup_cornermasks (); settings.changed.connect (resetup_cornermasks); } public override void destroy () { destroy_cornermasks (); } private void setup_cornermasks () { if (!settings.get_boolean ("enable")) { return; } int n_monitors = display.get_n_monitors (); corner_radii = new int[n_monitors]; cornermasks = new List[n_monitors]; for (int m = 0; m < n_monitors; m++) { corner_radii[m] = Utils.scale_to_int (DEFAULT_CORNER_RADIUS, display.get_monitor_scale (m)); } if (settings.get_boolean ("only-on-primary")) { add_cornermasks (display.get_primary_monitor ()); } else { for (int m = 0; m < n_monitors; m++) add_cornermasks (m); } if (settings.get_boolean ("disable-on-fullscreen")) { display.in_fullscreen_changed.connect (fullscreen_changed); } unowned Meta.MonitorManager monitor_manager = display.get_context ().get_backend ().get_monitor_manager (); monitor_manager.monitors_changed.connect (resetup_cornermasks); display.gl_video_memory_purged.connect (resetup_cornermasks); } private void destroy_cornermasks () { display.gl_video_memory_purged.disconnect (resetup_cornermasks); unowned Meta.MonitorManager monitor_manager = display.get_context ().get_backend ().get_monitor_manager (); monitor_manager.monitors_changed.disconnect (resetup_cornermasks); display.in_fullscreen_changed.disconnect (fullscreen_changed); foreach (unowned List list in cornermasks) { foreach (Clutter.Actor actor in list) { actor.destroy (); } } } private void resetup_cornermasks () { destroy_cornermasks (); setup_cornermasks (); } private void fullscreen_changed () { for (int i = 0; i < display.get_n_monitors (); i++) { foreach (Clutter.Actor actor in cornermasks[i]) { if (display.get_monitor_in_fullscreen (i)) { actor.hide (); } else { actor.show (); } } } } private void add_cornermasks (int monitor_no) { var monitor_geometry = display.get_monitor_geometry (monitor_no); var canvas = new Clutter.Canvas (); canvas.set_size (corner_radii[monitor_no], corner_radii[monitor_no]); canvas.draw.connect ((context) => draw_cornermask (context, monitor_no)); canvas.invalidate (); var actor = new Clutter.Actor (); actor.set_content (canvas); actor.set_size (corner_radii[monitor_no], corner_radii[monitor_no]); actor.set_position (monitor_geometry.x, monitor_geometry.y); actor.set_pivot_point ((float) 0.5, (float) 0.5); cornermasks[monitor_no].append (actor); wm.stage.add_child (actor); for (int p = 1; p < 4; p++) { var clone = new Clutter.Clone (actor); clone.rotation_angle_z = p * 90; switch (p) { case 1: clone.set_position (monitor_geometry.x + monitor_geometry.width, monitor_geometry.y); break; case 2: clone.set_position (monitor_geometry.x + monitor_geometry.width, monitor_geometry.y + monitor_geometry.height); break; case 3: clone.set_position (monitor_geometry.x, monitor_geometry.y + monitor_geometry.height); break; } cornermasks[monitor_no].append (clone); wm.stage.add_child (clone); } } private bool draw_cornermask (Cairo.Context context, int monitor_no) requires (corner_radii.length > monitor_no) { var buffer = new Drawing.BufferSurface (corner_radii[monitor_no], corner_radii[monitor_no]); var buffer_context = buffer.context; buffer_context.arc (corner_radii[monitor_no], corner_radii[monitor_no], corner_radii[monitor_no], Math.PI, 1.5 * Math.PI); buffer_context.line_to (0, 0); buffer_context.line_to (0, corner_radii[monitor_no]); buffer_context.set_source_rgb (0, 0, 0); buffer_context.fill (); context.set_operator (Cairo.Operator.CLEAR); context.paint (); context.set_operator (Cairo.Operator.OVER); context.set_source_surface (buffer.surface, 0, 0); context.paint (); return true; } } public Gala.PluginInfo register_plugin () { return { "Mask Corners", "Gala Developers", typeof (Gala.Plugins.MaskCorners.Main), Gala.PluginFunction.ADDITION, Gala.LoadPriority.IMMEDIATE }; }