Added the ability to change bar locations. (#257)

* Added the ability to change bar locations.

* Update dropdown margins

* Make dropdown to bar gap configurable and organized code.
This commit is contained in:
Jas Singh
2024-09-14 23:29:00 -07:00
committed by GitHub
parent 4238a5516e
commit 8c2537b917
21 changed files with 420 additions and 367 deletions

View File

@@ -14,6 +14,8 @@ export type RecursiveOptionsObject = {
[key: string]: RecursiveOptionsObject | Opt<string | number | boolean> | Opt<any>; [key: string]: RecursiveOptionsObject | Opt<string | number | boolean> | Opt<any>;
}; };
export type BarLocation = 'top' | 'bottom';
export type Unit = 'imperial' | 'metric'; export type Unit = 'imperial' | 'metric';
export type PowerOptions = 'sleep' | 'reboot' | 'logout' | 'shutdown'; export type PowerOptions = 'sleep' | 'reboot' | 'logout' | 'shutdown';
export type NotificationAnchor = export type NotificationAnchor =

View File

@@ -36,6 +36,7 @@ import { Attribute, Child } from 'lib/types/widget.js';
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
const { layouts } = options.bar; const { layouts } = options.bar;
const { location } = options.theme.bar;
export type BarWidget = keyof typeof widget; export type BarWidget = keyof typeof widget;
@@ -252,7 +253,7 @@ export const Bar = (() => {
class_name: 'bar', class_name: 'bar',
monitor, monitor,
visible: true, visible: true,
anchor: ['top', 'left', 'right'], anchor: location.bind('value').as((ln) => [ln, 'left', 'right']),
exclusivity: 'exclusive', exclusivity: 'exclusive',
layer: Utils.merge( layer: Utils.merge(
[options.theme.bar.layer.bind('value'), options.tear.bind('value')], [options.theme.bar.layer.bind('value'), options.tear.bind('value')],

View File

@@ -1,191 +0,0 @@
const hyprland = await Service.import('hyprland');
import { DropdownMenuProps } from 'lib/types/dropdownmenu';
import { Attribute, Child, Exclusivity, GtkWidget } from 'lib/types/widget';
import { bash } from 'lib/utils';
import { Widget as TWidget } from 'types/@girs/gtk-3.0/gtk-3.0.cjs';
import { Monitor } from 'types/service/hyprland';
import Box from 'types/widgets/box';
import EventBox from 'types/widgets/eventbox';
import Revealer from 'types/widgets/revealer';
import Window from 'types/widgets/window';
type NestedRevealer = Revealer<Box<TWidget, unknown>, unknown>;
type NestedBox = Box<NestedRevealer, unknown>;
type NestedEventBox = EventBox<NestedBox, unknown>;
export const Padding = (name: string): EventBox<Box<GtkWidget, Attribute>, Attribute> =>
Widget.EventBox({
hexpand: true,
vexpand: true,
can_focus: true,
child: Widget.Box(),
setup: (w) => w.on('button-press-event', () => App.toggleWindow(name)),
});
const moveBoxToCursor = <T extends NestedEventBox>(self: T, fixed: boolean): void => {
if (fixed) {
return;
}
globalMousePos.connect('changed', async ({ value }) => {
const curHyprlandMonitor = hyprland.monitors.find((m) => m.id === hyprland.active.monitor.id);
const dropdownWidth = self.child.get_allocation().width;
let hyprScaling = 1;
try {
const monitorInfo = await bash('hyprctl monitors -j');
const parsedMonitorInfo = JSON.parse(monitorInfo);
const foundMonitor = parsedMonitorInfo.find(
(monitor: Monitor) => monitor.id === hyprland.active.monitor.id,
);
hyprScaling = foundMonitor?.scale || 1;
} catch (error) {
console.error(`Error parsing hyprland monitors: ${error}`);
}
let monWidth = curHyprlandMonitor?.width;
let monHeight = curHyprlandMonitor?.height;
if (monWidth === undefined || monHeight === undefined || hyprScaling === undefined) {
return;
}
// If GDK Scaling is applied, then get divide width by scaling
// to get the proper coordinates.
// Ex: On a 2860px wide monitor... if scaling is set to 2, then the right
// end of the monitor is the 1430th pixel.
const gdkScale = Utils.exec('bash -c "echo $GDK_SCALE"');
if (/^\d+(.\d+)?$/.test(gdkScale)) {
const scale = parseFloat(gdkScale);
monWidth = monWidth / scale;
monHeight = monHeight / scale;
} else {
monWidth = monWidth / hyprScaling;
monHeight = monHeight / hyprScaling;
}
// If monitor is vertical (transform = 1 || 3) swap height and width
const isVertical = curHyprlandMonitor?.transform !== undefined ? curHyprlandMonitor.transform % 2 !== 0 : false;
if (isVertical) {
[monWidth, monHeight] = [monHeight, monWidth];
}
let marginRight = monWidth - dropdownWidth / 2;
marginRight = fixed ? marginRight - monWidth / 2 : marginRight - value[0];
let marginLeft = monWidth - dropdownWidth - marginRight;
const minimumMargin = 0;
if (marginRight < minimumMargin) {
marginRight = minimumMargin;
marginLeft = monWidth - dropdownWidth - minimumMargin;
}
if (marginLeft < minimumMargin) {
marginLeft = minimumMargin;
marginRight = monWidth - dropdownWidth - minimumMargin;
}
const marginTop = 45;
const marginBottom = monHeight - marginTop;
self.set_margin_left(marginLeft);
self.set_margin_right(marginRight);
self.set_margin_bottom(marginBottom);
});
};
// NOTE: We make the window visible for 2 seconds (on startup) so the child
// elements can allocat their proper dimensions.
// Otherwise the width that we rely on for menu positioning is set improperly
// for the first time we open a menu of each type.
const initRender = Variable(true);
setTimeout(() => {
initRender.value = false;
}, 2000);
export default ({
name,
child,
transition,
exclusivity = 'ignore' as Exclusivity,
fixed = false,
...props
}: DropdownMenuProps): Window<Child, Attribute> =>
Widget.Window({
name,
class_names: [name, 'dropdown-menu'],
setup: (w) => w.keybind('Escape', () => App.closeWindow(name)),
visible: initRender.bind('value'),
keymode: 'on-demand',
exclusivity,
layer: 'top',
anchor: ['top', 'left'],
child: Widget.EventBox({
class_name: 'parent-event',
on_primary_click: () => App.closeWindow(name),
on_secondary_click: () => App.closeWindow(name),
child: Widget.Box({
class_name: 'top-eb',
vertical: true,
children: [
Widget.EventBox({
class_name: 'mid-eb event-top-padding-static',
hexpand: true,
vexpand: false,
can_focus: false,
child: Widget.Box(),
setup: (w) => {
w.on('button-press-event', () => App.toggleWindow(name));
w.set_margin_top(1);
},
}),
Widget.EventBox({
class_name: 'mid-eb event-top-padding',
hexpand: true,
vexpand: false,
can_focus: false,
child: Widget.Box(),
setup: (w) => {
w.on('button-press-event', () => App.toggleWindow(name));
w.set_margin_top(1);
},
}),
Widget.EventBox({
class_name: 'in-eb menu-event-box',
on_primary_click: () => {
return true;
},
on_secondary_click: () => {
return true;
},
setup: (self) => {
moveBoxToCursor(self, fixed);
},
child: Widget.Box({
class_name: 'dropdown-menu-container',
css: 'padding: 1px; margin: -1px;',
child: Widget.Revealer({
revealChild: false,
setup: (self) =>
self.hook(App, (_, wname, visible) => {
if (wname === name) self.reveal_child = visible;
}),
transition,
transitionDuration: 350,
child: Widget.Box({
class_name: 'dropdown-menu-container',
can_focus: true,
children: [child],
}),
}),
}),
}),
],
}),
}),
...props,
});

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { activeDevices } from './active/index.js'; import { activeDevices } from './active/index.js';
import { availableDevices } from './available/index.js'; import { availableDevices } from './available/index.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { Devices } from './devices/index.js'; import { Devices } from './devices/index.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,4 +1,4 @@
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { TimeWidget } from './time/index.js'; import { TimeWidget } from './time/index.js';
import { CalendarWidget } from './calendar.js'; import { CalendarWidget } from './calendar.js';
import { WeatherWidget } from './weather/index.js'; import { WeatherWidget } from './weather/index.js';

View File

@@ -1,4 +1,4 @@
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { Profile } from './profile/index.js'; import { Profile } from './profile/index.js';
import { Shortcuts } from './shortcuts/index.js'; import { Shortcuts } from './shortcuts/index.js';
import { Controls } from './controls/index.js'; import { Controls } from './controls/index.js';

View File

@@ -1,4 +1,4 @@
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { EnergyProfiles } from './profiles/index.js'; import { EnergyProfiles } from './profiles/index.js';
import { Brightness } from './brightness/index.js'; import { Brightness } from './brightness/index.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { Media } from './media.js'; import { Media } from './media.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { Ethernet } from './ethernet/index.js'; import { Ethernet } from './ethernet/index.js';
import { Wifi } from './wifi/index.js'; import { Wifi } from './wifi/index.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,5 +1,5 @@
import { Notification } from 'types/service/notifications.js'; import { Notification } from 'types/service/notifications.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
const notifs = await Service.import('notifications'); const notifs = await Service.import('notifications');
import { Controls } from './controls/index.js'; import { Controls } from './controls/index.js';
import { NotificationCard } from './notification/index.js'; import { NotificationCard } from './notification/index.js';

View File

@@ -1,5 +1,5 @@
import { Action } from 'lib/types/power.js'; import { Action } from 'lib/types/power.js';
import PopupWindow from '../PopupWindow.js'; import PopupWindow from '../shared/popup/index.js';
import powermenu from './helpers/actions.js'; import powermenu from './helpers/actions.js';
import icons from '../../icons/index.js'; import icons from '../../icons/index.js';
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import PopupWindow from '../PopupWindow.js'; import PopupWindow from '../shared/popup/index.js';
import powermenu from './helpers/actions.js'; import powermenu from './helpers/actions.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -1,5 +1,5 @@
import Window from 'types/widgets/window.js'; import Window from 'types/widgets/window.js';
import DropdownMenu from '../DropdownMenu.js'; import DropdownMenu from '../shared/dropdown/index.js';
import { PowerButton } from './button.js'; import { PowerButton } from './button.js';
import { Attribute, Child } from 'lib/types/widget.js'; import { Attribute, Child } from 'lib/types/widget.js';

View File

@@ -0,0 +1,33 @@
import { Attribute, BoxWidget } from 'lib/types/widget';
import EventBox from 'types/widgets/eventbox';
import { BarLocation } from 'lib/types/options';
const createEventBox = (className: string, windowName: string): EventBox<BoxWidget, Attribute> => {
return Widget.EventBox({
class_name: className,
hexpand: true,
vexpand: false,
can_focus: false,
child: Widget.Box(),
setup: (w) => {
w.on('button-press-event', () => App.toggleWindow(windowName));
},
});
};
export const barEventMargins = (
windowName: string,
location: BarLocation = 'top',
): [EventBox<BoxWidget, Attribute>, EventBox<BoxWidget, Attribute>] => {
if (location === 'top') {
return [
createEventBox('mid-eb event-top-padding-static', windowName),
createEventBox('mid-eb event-top-padding', windowName),
];
} else {
return [
createEventBox('mid-eb event-bottom-padding', windowName),
createEventBox('mid-eb event-bottom-padding-static', windowName),
];
}
};

View File

@@ -0,0 +1,99 @@
import options from 'options';
import { DropdownMenuProps } from 'lib/types/dropdownmenu';
import { Attribute, Child, Exclusivity } from 'lib/types/widget';
import Window from 'types/widgets/window';
import { moveBoxToCursor } from './locationHandler/index';
import { barEventMargins } from './eventBoxes/index';
const { location } = options.theme.bar;
// NOTE: We make the window visible for 2 seconds (on startup) so the child
// elements can allocat their proper dimensions.
// Otherwise the width that we rely on for menu positioning is set improperly
// for the first time we open a menu of each type.
const initRender = Variable(true);
setTimeout(() => {
initRender.value = false;
}, 2000);
export default ({
name,
child,
transition,
exclusivity = 'ignore' as Exclusivity,
fixed = false,
...props
}: DropdownMenuProps): Window<Child, Attribute> =>
Widget.Window({
name,
class_names: [name, 'dropdown-menu'],
setup: (w) => w.keybind('Escape', () => App.closeWindow(name)),
visible: initRender.bind('value'),
keymode: 'on-demand',
exclusivity,
layer: 'top',
anchor: location.bind('value').as((ln) => [ln, 'left']),
child: Widget.EventBox({
class_name: 'parent-event',
on_primary_click: () => App.closeWindow(name),
on_secondary_click: () => App.closeWindow(name),
child: Widget.Box({
class_name: 'top-eb',
vertical: true,
children: [
Widget.Box({
className: 'event-box-container',
children: location.bind('value').as((lcn) => {
if (lcn === 'top') {
return barEventMargins(name);
} else {
return [];
}
}),
}),
Widget.EventBox({
class_name: 'in-eb menu-event-box',
on_primary_click: () => {
return true;
},
on_secondary_click: () => {
return true;
},
setup: (self) => {
moveBoxToCursor(self, fixed);
},
child: Widget.Box({
class_name: 'dropdown-menu-container',
css: 'padding: 1px; margin: -1px;',
child: Widget.Revealer({
revealChild: false,
setup: (self) =>
self.hook(App, (_, wname, visible) => {
if (wname === name) self.reveal_child = visible;
}),
transition,
transitionDuration: 350,
child: Widget.Box({
class_name: 'dropdown-menu-container',
can_focus: true,
children: [child],
}),
}),
}),
}),
Widget.Box({
className: 'event-box-container',
children: location.bind('value').as((lcn) => {
if (lcn === 'bottom') {
return barEventMargins(name);
} else {
return [];
}
}),
}),
],
}),
}),
...props,
});

View File

@@ -0,0 +1,95 @@
const hyprland = await Service.import('hyprland');
import { bash } from 'lib/utils';
import { Widget as TWidget } from 'types/@girs/gtk-3.0/gtk-3.0.cjs';
import { Monitor } from 'types/service/hyprland';
import Box from 'types/widgets/box';
import EventBox from 'types/widgets/eventbox';
import Revealer from 'types/widgets/revealer';
type NestedRevealer = Revealer<Box<TWidget, unknown>, unknown>;
type NestedBox = Box<NestedRevealer, unknown>;
type NestedEventBox = EventBox<NestedBox, unknown>;
const { location } = options.theme.bar;
export const moveBoxToCursor = <T extends NestedEventBox>(self: T, fixed: boolean): void => {
if (fixed) {
return;
}
globalMousePos.connect('changed', async ({ value }) => {
const curHyprlandMonitor = hyprland.monitors.find((m) => m.id === hyprland.active.monitor.id);
const dropdownWidth = self.child.get_allocation().width;
const dropdownHeight = self.child.get_allocation().height;
let hyprScaling = 1;
try {
const monitorInfo = await bash('hyprctl monitors -j');
const parsedMonitorInfo = JSON.parse(monitorInfo);
const foundMonitor = parsedMonitorInfo.find(
(monitor: Monitor) => monitor.id === hyprland.active.monitor.id,
);
hyprScaling = foundMonitor?.scale || 1;
} catch (error) {
console.error(`Error parsing hyprland monitors: ${error}`);
}
let monWidth = curHyprlandMonitor?.width;
let monHeight = curHyprlandMonitor?.height;
if (monWidth === undefined || monHeight === undefined || hyprScaling === undefined) {
return;
}
// If GDK Scaling is applied, then get divide width by scaling
// to get the proper coordinates.
// Ex: On a 2860px wide monitor... if scaling is set to 2, then the right
// end of the monitor is the 1430th pixel.
const gdkScale = Utils.exec('bash -c "echo $GDK_SCALE"');
if (/^\d+(.\d+)?$/.test(gdkScale)) {
const scale = parseFloat(gdkScale);
monWidth = monWidth / scale;
monHeight = monHeight / scale;
} else {
monWidth = monWidth / hyprScaling;
monHeight = monHeight / hyprScaling;
}
// If monitor is vertical (transform = 1 || 3) swap height and width
const isVertical = curHyprlandMonitor?.transform !== undefined ? curHyprlandMonitor.transform % 2 !== 0 : false;
if (isVertical) {
[monWidth, monHeight] = [monHeight, monWidth];
}
let marginRight = monWidth - dropdownWidth / 2;
marginRight = fixed ? marginRight - monWidth / 2 : marginRight - value[0];
let marginLeft = monWidth - dropdownWidth - marginRight;
const minimumMargin = 0;
if (marginRight < minimumMargin) {
marginRight = minimumMargin;
marginLeft = monWidth - dropdownWidth - minimumMargin;
}
if (marginLeft < minimumMargin) {
marginLeft = minimumMargin;
marginRight = monWidth - dropdownWidth - minimumMargin;
}
self.set_margin_left(marginLeft);
self.set_margin_right(marginRight);
if (location.value === 'top') {
self.set_margin_top(0);
self.set_margin_bottom(monHeight);
} else {
self.set_margin_bottom(0);
self.set_margin_top(monHeight - dropdownHeight);
}
});
};

View File

@@ -12,6 +12,7 @@ import { KbIcon, KbLabelType } from 'lib/types/customModules/kbLayout';
import { import {
ActiveWsIndicator, ActiveWsIndicator,
BarButtonStyles, BarButtonStyles,
BarLocation,
NotificationAnchor, NotificationAnchor,
OSDAnchor, OSDAnchor,
OSDOrientation, OSDOrientation,
@@ -145,6 +146,7 @@ const options = mkOptions(OPTIONS, {
bar: { bar: {
scaling: opt(100), scaling: opt(100),
floating: opt(false), floating: opt(false),
location: opt<BarLocation>('top'),
layer: opt<WindowLayer>('top'), layer: opt<WindowLayer>('top'),
margin_top: opt('0.5em'), margin_top: opt('0.5em'),
opacity: opt(100), opacity: opt(100),
@@ -154,6 +156,7 @@ const options = mkOptions(OPTIONS, {
outer_spacing: opt('1.6em'), outer_spacing: opt('1.6em'),
label_spacing: opt('0.5em'), label_spacing: opt('0.5em'),
transparent: opt(false), transparent: opt(false),
dropdownGap: opt('2.9em'),
background: opt(colors.crust), background: opt(colors.crust),
buttons: { buttons: {
style: opt<BarButtonStyles>('default'), style: opt<BarButtonStyles>('default'),

View File

@@ -1,257 +1,256 @@
.menu-slider { .menu-slider {
trough {
border-radius: 0.3rem;
background: $surface0;
highlight,
progress {
background: $peach;
border-radius: 0.3rem;
}
}
slider {
box-shadow: none;
background-color: transparent;
min-height: 0.6rem;
min-width: 0.6rem;
border: 0rem solid transparent;
border-radius: 0.3rem;
}
&:hover {
trough { trough {
background: $surface0; border-radius: 0.3rem;
background: $surface0;
highlight,
progress {
background: $peach;
border-radius: 0.3rem;
}
} }
slider { slider {
background: $overlay0; box-shadow: none;
box-shadow: none; background-color: transparent;
min-height: 0.6rem;
min-width: 0.6rem;
border: 0rem solid transparent;
border-radius: 0.3rem;
}
&:hover {
trough {
background: $surface0;
}
slider {
background: $overlay0;
box-shadow: none;
}
} }
}
} }
.menu-switch { .menu-switch {
font-size: 1.3em; font-size: 1.3em;
background-color: $surface0; background-color: $surface0;
border-radius: 0.2em;
&:checked {
background: $sky;
}
trough {
highlight,
progress {
background-color: $peach;
border-radius: 0.3em;
}
}
slider {
box-shadow: none;
background-color: $overlay0;
min-height: 1em;
min-width: 1em;
border: 0em solid transparent;
border-radius: 0.2em; border-radius: 0.2em;
margin: 0.1em 0.15em;
}
&:hover { &:checked {
background: $sky;
}
trough { trough {
background: $surface0; highlight,
progress {
background-color: $peach;
border-radius: 0.3em;
}
} }
slider { slider {
background: $overlay0; box-shadow: none;
box-shadow: none; background-color: $overlay0;
min-height: 1em;
min-width: 1em;
border: 0em solid transparent;
border-radius: 0.2em;
margin: 0.1em 0.15em;
} }
}
&:active { &:hover {
background-color: $sky; trough {
} background: $surface0;
}
slider {
background: $overlay0;
box-shadow: none;
}
}
&:active {
background-color: $sky;
}
} }
.menu-separator { .menu-separator {
min-height: .1rem; min-height: 0.1rem;
margin: .6rem 0rem; margin: 0.6rem 0rem;
background: $surface1; background: $surface1;
} }
.menu-items { .menu-items {
background: $crust; background: $crust;
border: $bar-menus-border-size solid $bar-menus-border-color; border: $bar-menus-border-size solid $bar-menus-border-color;
border-radius: $bar-menus-border-radius; border-radius: $bar-menus-border-radius;
color: $text; color: $text;
} }
.menu-items-container { .menu-items-container {
border-radius: 0.4em; border-radius: 0.4em;
font-size: 1.3em; font-size: 1.3em;
} }
.menu-section-container { .menu-section-container {
margin: 1em 0em; margin: 1em 0em;
.menu-label { .menu-label {
color: $text; color: $text;
font-size: 1.1em; font-size: 1.1em;
font-weight: bold; font-weight: bold;
} }
.menu-label-container { .menu-label-container {
background: $base; background: $base;
border-radius: $bar-menus-card_radius; border-radius: $bar-menus-card_radius;
border-bottom-left-radius: 0em; border-bottom-left-radius: 0em;
border-bottom-right-radius: 0em; border-bottom-right-radius: 0em;
margin: 0em 1em; margin: 0em 1em;
min-height: 2em; min-height: 2em;
} }
&:first-child { &:first-child {
margin-bottom: 0em; margin-bottom: 0em;
} }
&:last-child { &:last-child {
margin-top: 0em; margin-top: 0em;
} }
&:nth-child(2) { &:nth-child(2) {
margin-top: 1em; margin-top: 1em;
} }
.menu-items-section { .menu-items-section {
background: $base; background: $base;
border-radius: $bar-menus-card_radius; border-radius: $bar-menus-card_radius;
border-top-left-radius: 0em; border-top-left-radius: 0em;
border-top-right-radius: 0em; border-top-right-radius: 0em;
padding: 0.9em; padding: 0.9em;
margin: 0em 1em; margin: 0em 1em;
} }
} }
.menu-active { .menu-active {
font-size: 0.9em; font-size: 0.9em;
font-weight: bold; font-weight: bold;
margin: 0rem 1em; margin: 0rem 1em;
margin-bottom: 0.9em; margin-bottom: 0.9em;
} }
.menu-active-container { .menu-active-container {
&:first-child { &:first-child {
margin-bottom: 0.5em; margin-bottom: 0.5em;
} }
} }
.menu-active-button { .menu-active-button {
padding: 0.1em; padding: 0.1em;
margin-bottom: -0.2em; margin-bottom: -0.2em;
.menu-active-icon { .menu-active-icon {
font-size: 1.4em; font-size: 1.4em;
font-weight: bold; font-weight: bold;
}
} &.muted image {
color: $maroon;
}
&.muted image { &:hover image {
color: $maroon; color: $maroon;
} }
&:hover image {
color: $maroon;
}
} }
.menu-active-percentage { .menu-active-percentage {
font-size: 0.9em; font-size: 0.9em;
min-width: 2.5em; min-width: 2.5em;
font-weight: bold; font-weight: bold;
} }
.menu-active-slider { .menu-active-slider {
margin-left: 1rem; margin-left: 1rem;
margin-right: 1.5rem; margin-right: 1.5rem;
} }
.menu-active-slider * { .menu-active-slider * {
min-height: 0.85em; min-height: 0.85em;
border-radius: .2em; border-radius: 0.2em;
} }
.menu-slider-container { .menu-slider-container {
margin-bottom: .7rem; margin-bottom: 0.7rem;
} }
.menu-label-dim { .menu-label-dim {
color: $overlay0; color: $overlay0;
margin-right: 1rem; margin-right: 1rem;
font-size: 1em; font-size: 1em;
font-weight: bold; font-weight: bold;
} }
.menu-icon-button { .menu-icon-button {
&:hover { &:hover {
color: $surface2; color: $surface2;
} }
} }
.menu-dropdown-label-container { .menu-dropdown-label-container {
background: $base; background: $base;
border-radius: 0.4em; border-radius: 0.4em;
} }
.menu-button { .menu-button {
margin-bottom: .4em; margin-bottom: 0.4em;
} }
.menu-button-name { .menu-button-name {
font-size: 0.95em; font-size: 0.95em;
font-weight: bold; font-weight: bold;
margin-left: 0.5em; margin-left: 0.5em;
margin-right: 1.2rem; margin-right: 1.2rem;
} }
.menu-button-icon { .menu-button-icon {
font-size: 1.3em; font-size: 1.3em;
font-weight: bold; font-weight: bold;
margin-right: .5rem; margin-right: 0.5rem;
} }
.menu-item-box { .menu-item-box {
margin-bottom: 0.5rem; margin-bottom: 0.5rem;
} }
.dropdown-menu-container { .dropdown-menu-container {
min-height: 10em; min-height: 10em;
} }
.menu-label { .menu-label {
margin: 0.5em 1em; margin: 0.5em 1em;
color: $sky; color: $sky;
} }
.event-top-padding-static * { .event-box-container {
min-height: 0em; min-height: 0em;
margin-top: 2.8em; margin-top: $bar-dropdownGap;
} }
.event-top-padding * { .event-top-padding * {
min-height: 0em; min-height: 0em;
margin-top: if($bar-floating, $bar-margin_top, 0); margin-top: if($bar-floating and $bar-location == 'top', $bar-margin_top, 0);
margin-bottom: if($bar-floating and $bar-location == 'bottom', $bar-margin_bottom, 0);
} }
@keyframes spin { @keyframes spin {
to { to {
-gtk-icon-transform: rotate(1turn); -gtk-icon-transform: rotate(1turn);
} }
} }
image.spinning { image.spinning {
animation-name: spin; animation-name: spin;
animation-duration: 1s; animation-duration: 1s;
animation-timing-function: linear; animation-timing-function: linear;
animation-iteration-count: infinite; animation-iteration-count: infinite;
} }

View File

@@ -29,6 +29,17 @@ export const BarSettings = (): Scrollable<Gtk.Widget, Gtk.Widget> => {
}, },
'bar-layout-input', 'bar-layout-input',
), ),
Option({
opt: options.theme.bar.floating,
title: 'Floating Bar',
type: 'boolean',
}),
Option({
opt: options.theme.bar.location,
title: 'Location',
type: 'enum',
enums: ['top', 'bottom'],
}),
/* /*
****************************** ******************************
@@ -69,11 +80,6 @@ export const BarSettings = (): Scrollable<Gtk.Widget, Gtk.Widget> => {
title: 'Button Radius', title: 'Button Radius',
type: 'string', type: 'string',
}), }),
Option({
opt: options.theme.bar.floating,
title: 'Floating Bar',
type: 'boolean',
}),
Option({ Option({
opt: options.theme.bar.layer, opt: options.theme.bar.layer,
title: 'Layer', title: 'Layer',
@@ -81,6 +87,12 @@ export const BarSettings = (): Scrollable<Gtk.Widget, Gtk.Widget> => {
subtitle: 'Layer determines the Z index of your bar.', subtitle: 'Layer determines the Z index of your bar.',
enums: ['top', 'bottom', 'overlay', 'background'], enums: ['top', 'bottom', 'overlay', 'background'],
}), }),
Option({
opt: options.theme.bar.dropdownGap,
title: 'Dropdown Gap',
subtitle: 'The gap between the dropdown and the bar',
type: 'string',
}),
Option({ Option({
opt: options.theme.bar.margin_top, opt: options.theme.bar.margin_top,
title: 'Margin Top', title: 'Margin Top',