format tile size

This commit is contained in:
2026-03-31 20:41:30 -05:00
parent f217b6af0d
commit ca20cf6e90
3 changed files with 27 additions and 19 deletions

View File

@@ -632,7 +632,7 @@ class SovranHubWindow(Adw.ApplicationWindow):
max_children_per_line=4, max_children_per_line=4,
min_children_per_line=2, min_children_per_line=2,
selection_mode=Gtk.SelectionMode.NONE, selection_mode=Gtk.SelectionMode.NONE,
homogeneous=True, homogeneous=False,
row_spacing=12, row_spacing=12,
column_spacing=12, column_spacing=12,
margin_top=4, margin_top=4,

View File

@@ -19,8 +19,7 @@ LOADING_STATES = {"reloading", "activating", "deactivating", "maintenance"}
ICON_DIR = os.environ.get("SOVRAN_HUB_ICONS", "") ICON_DIR = os.environ.get("SOVRAN_HUB_ICONS", "")
ICON_EXTENSIONS = [".svg", ".png"] ICON_EXTENSIONS = [".svg", ".png"]
TILE_WIDTH = 148 TILE_SIZE = 140
TILE_HEIGHT = 170
class ServiceTile(Gtk.Box): class ServiceTile(Gtk.Box):
@@ -29,14 +28,17 @@ class ServiceTile(Gtk.Box):
icon_name="", enabled=True, **kw): icon_name="", enabled=True, **kw):
super().__init__( super().__init__(
orientation=Gtk.Orientation.VERTICAL, orientation=Gtk.Orientation.VERTICAL,
spacing=4, spacing=2,
halign=Gtk.Align.CENTER, halign=Gtk.Align.CENTER,
valign=Gtk.Align.START, valign=Gtk.Align.START,
width_request=TILE_WIDTH,
height_request=TILE_HEIGHT,
css_classes=["card", "sovran-tile"], css_classes=["card", "sovran-tile"],
**kw, **kw,
) )
# Force exact tile dimensions
self.set_size_request(TILE_SIZE, TILE_SIZE + 30)
self.set_hexpand(False)
self.set_vexpand(False)
self._unit = unit self._unit = unit
self._scope = scope self._scope = scope
self._method = method self._method = method
@@ -44,28 +46,30 @@ class ServiceTile(Gtk.Box):
# ── Icon ───────────────────────────────────────────────── # ── Icon ─────────────────────────────────────────────────
self._logo = Gtk.Image( self._logo = Gtk.Image(
pixel_size=40, pixel_size=36,
margin_top=16, margin_top=14,
halign=Gtk.Align.CENTER, halign=Gtk.Align.CENTER,
) )
self._set_logo(icon_name) self._set_logo(icon_name)
self.append(self._logo) self.append(self._logo)
# ── Name label (wraps, max 2 lines) ────────────────────── # ── Name label (wraps within tile width) ─────────────────
self._name_label = Gtk.Label( self._name_label = Gtk.Label(
label=name, label=name,
css_classes=["heading"], css_classes=["heading", "tile-name"],
halign=Gtk.Align.CENTER, halign=Gtk.Align.CENTER,
justify=Gtk.Justification.CENTER, justify=Gtk.Justification.CENTER,
wrap=True, wrap=True,
wrap_mode=Pango.WrapMode.WORD_CHAR, wrap_mode=Pango.WrapMode.WORD_CHAR,
max_width_chars=16,
lines=2, lines=2,
ellipsize=Pango.EllipsizeMode.END, ellipsize=Pango.EllipsizeMode.END,
margin_start=8, margin_start=6,
margin_end=8, margin_end=6,
margin_top=4, margin_top=4,
) )
# Clamp the label width so it wraps inside the tile
self._name_label.set_size_request(TILE_SIZE - 16, -1)
self._name_label.set_max_width_chars(1) # forces natural width to be small
self.append(self._name_label) self.append(self._name_label)
# ── Status label ───────────────────────────────────────── # ── Status label ─────────────────────────────────────────
@@ -73,11 +77,11 @@ class ServiceTile(Gtk.Box):
label="● …", label="● …",
css_classes=["caption", "dim-label"], css_classes=["caption", "dim-label"],
halign=Gtk.Align.CENTER, halign=Gtk.Align.CENTER,
margin_top=2, margin_top=1,
) )
self.append(self._status_label) self.append(self._status_label)
# ── Spacer to push controls to bottom ──────────────────── # ── Spacer ───────────────────────────────────────────────
spacer = Gtk.Box(vexpand=True) spacer = Gtk.Box(vexpand=True)
self.append(spacer) self.append(spacer)
@@ -86,7 +90,7 @@ class ServiceTile(Gtk.Box):
orientation=Gtk.Orientation.HORIZONTAL, orientation=Gtk.Orientation.HORIZONTAL,
spacing=8, spacing=8,
halign=Gtk.Align.CENTER, halign=Gtk.Align.CENTER,
margin_bottom=12, margin_bottom=10,
) )
self._switch = Gtk.Switch(valign=Gtk.Align.CENTER) self._switch = Gtk.Switch(valign=Gtk.Align.CENTER)
self._switch.connect("state-set", self._on_toggled) self._switch.connect("state-set", self._on_toggled)
@@ -117,7 +121,7 @@ class ServiceTile(Gtk.Box):
path = os.path.join(ICON_DIR, f"{icon_name}{ext}") path = os.path.join(ICON_DIR, f"{icon_name}{ext}")
if os.path.isfile(path): if os.path.isfile(path):
try: try:
pixbuf = GdkPixbuf.Pixbuf.new_from_file_at_scale(path, 40, 40, True) pixbuf = GdkPixbuf.Pixbuf.new_from_file_at_scale(path, 36, 36, True)
texture = Gdk.Texture.new_for_pixbuf(pixbuf) texture = Gdk.Texture.new_for_pixbuf(pixbuf)
self._logo.set_from_paintable(texture) self._logo.set_from_paintable(texture)
return return

View File

@@ -1,15 +1,19 @@
.sovran-tile { .sovran-tile {
border-radius: 16px; border-radius: 16px;
padding: 0px; padding: 0px;
min-width: 148px; min-width: 140px;
max-width: 148px; max-width: 140px;
min-height: 170px; min-height: 170px;
max-height: 170px; max-height: 170px;
overflow: hidden;
transition: box-shadow 200ms ease-in-out; transition: box-shadow 200ms ease-in-out;
} }
.sovran-tile:hover { .sovran-tile:hover {
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.25); box-shadow: 0 2px 8px rgba(0, 0, 0, 0.25);
} }
.tile-name {
font-size: 0.8em;
}
.success { color: #2ec27e; } .success { color: #2ec27e; }
.warning { color: #e5a50a; } .warning { color: #e5a50a; }
.error { color: #e01b24; } .error { color: #e01b24; }