174 lines
6.4 KiB
Swift
174 lines
6.4 KiB
Swift
//
|
||
// Overlay.swift
|
||
// Adwaita
|
||
//
|
||
// Created by auto-generation on 31.10.25.
|
||
//
|
||
|
||
import CAdw
|
||
import LevenshteinTransformations
|
||
|
||
/// Places “overlay” widgets on top of a single main child.
|
||
///
|
||
///
|
||
///
|
||
/// The position of each overlay widget is determined by its
|
||
/// ``halign(_:)`` and ``valign(_:)``
|
||
/// properties. E.g. a widget with both alignments set to %GTK_ALIGN_START
|
||
/// will be placed at the top left corner of the `GtkOverlay` container,
|
||
/// whereas an overlay with halign set to %GTK_ALIGN_CENTER and valign set
|
||
/// to %GTK_ALIGN_END will be placed a the bottom edge of the `GtkOverlay`,
|
||
/// horizontally centered. The position can be adjusted by setting the margin
|
||
/// properties of the child to non-zero values.
|
||
///
|
||
/// More complicated placement of overlays is possible by connecting
|
||
/// to the `Gtk.Overlay::get-child-position` signal.
|
||
///
|
||
/// An overlay’s minimum and natural sizes are those of its main child.
|
||
/// The sizes of overlay children are not considered when measuring these
|
||
/// preferred sizes.
|
||
///
|
||
///
|
||
public struct Overlay: AdwaitaWidget {
|
||
|
||
/// Additional update functions for type extensions.
|
||
var updateFunctions: [(ViewStorage, WidgetData, Bool) -> Void] = []
|
||
/// Additional appear functions for type extensions.
|
||
var appearFunctions: [(ViewStorage, WidgetData) -> Void] = []
|
||
|
||
/// The accessible role of the given `GtkAccessible` implementation.
|
||
///
|
||
/// The accessible role cannot be changed once set.
|
||
var accessibleRole: String?
|
||
/// The main child widget.
|
||
var child: Body?
|
||
/// Emitted to determine the position and size of any overlay
|
||
/// child widgets.
|
||
///
|
||
/// A handler for this signal should fill @allocation with
|
||
/// the desired position and size for @widget, relative to
|
||
/// the 'main' child of @overlay.
|
||
///
|
||
/// The default handler for this signal uses the @widget's
|
||
/// halign and valign properties to determine the position
|
||
/// and gives the widget its natural size (except that an
|
||
/// alignment of %GTK_ALIGN_FILL will cause the overlay to
|
||
/// be full-width/height). If the main child is a
|
||
/// `GtkScrolledWindow`, the overlays are placed relative
|
||
/// to its contents.
|
||
var getChildPosition: (() -> Void)?
|
||
/// The body for the widget "overlay".
|
||
var overlay: () -> Body = { [] }
|
||
|
||
/// Initialize `Overlay`.
|
||
public init() {
|
||
}
|
||
|
||
/// The view storage.
|
||
/// - Parameters:
|
||
/// - modifiers: Modify views before being updated.
|
||
/// - type: The view render data type.
|
||
/// - Returns: The view storage.
|
||
public func container<Data>(data: WidgetData, type: Data.Type) -> ViewStorage where Data: ViewRenderData {
|
||
let storage = ViewStorage(gtk_overlay_new()?.opaque())
|
||
for function in appearFunctions {
|
||
function(storage, data)
|
||
}
|
||
update(storage, data: data, updateProperties: true, type: type)
|
||
if let childStorage = child?.storage(data: data, type: type) {
|
||
storage.content["child"] = [childStorage]
|
||
gtk_overlay_set_child(storage.opaquePointer, childStorage.opaquePointer?.cast())
|
||
}
|
||
|
||
var overlayStorage: [ViewStorage] = []
|
||
for view in overlay() {
|
||
overlayStorage.append(view.storage(data: data, type: type))
|
||
gtk_overlay_add_overlay(storage.opaquePointer, overlayStorage.last?.opaquePointer?.cast())
|
||
}
|
||
storage.content["overlay"] = overlayStorage
|
||
return storage
|
||
}
|
||
|
||
/// Update the stored content.
|
||
/// - Parameters:
|
||
/// - storage: The storage to update.
|
||
/// - modifiers: Modify views before being updated
|
||
/// - updateProperties: Whether to update the view's properties.
|
||
/// - type: The view render data type.
|
||
public func update<Data>(_ storage: ViewStorage, data: WidgetData, updateProperties: Bool, type: Data.Type) where Data: ViewRenderData {
|
||
if let getChildPosition {
|
||
storage.connectSignal(name: "get-child-position", argCount: 2) {
|
||
getChildPosition()
|
||
}
|
||
}
|
||
storage.modify { widget in
|
||
|
||
if let widget = storage.content["child"]?.first {
|
||
child?.updateStorage(widget, data: data, updateProperties: updateProperties, type: type)
|
||
}
|
||
|
||
if let overlayStorage = storage.content["overlay"] {
|
||
for (index, view) in overlay().enumerated() {
|
||
if let storage = overlayStorage[safe: index] {
|
||
view.updateStorage(
|
||
storage,
|
||
data: data,
|
||
updateProperties: updateProperties,
|
||
type: type
|
||
)
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
}
|
||
for function in updateFunctions {
|
||
function(storage, data, updateProperties)
|
||
}
|
||
if updateProperties {
|
||
storage.previousState = self
|
||
}
|
||
}
|
||
|
||
/// The accessible role of the given `GtkAccessible` implementation.
|
||
///
|
||
/// The accessible role cannot be changed once set.
|
||
public func accessibleRole(_ accessibleRole: String?) -> Self {
|
||
modify { $0.accessibleRole = accessibleRole }
|
||
}
|
||
|
||
/// The main child widget.
|
||
public func child(@ViewBuilder _ child: () -> Body) -> Self {
|
||
modify { $0.child = child() }
|
||
}
|
||
|
||
/// Emitted to determine the position and size of any overlay
|
||
/// child widgets.
|
||
///
|
||
/// A handler for this signal should fill @allocation with
|
||
/// the desired position and size for @widget, relative to
|
||
/// the 'main' child of @overlay.
|
||
///
|
||
/// The default handler for this signal uses the @widget's
|
||
/// halign and valign properties to determine the position
|
||
/// and gives the widget its natural size (except that an
|
||
/// alignment of %GTK_ALIGN_FILL will cause the overlay to
|
||
/// be full-width/height). If the main child is a
|
||
/// `GtkScrolledWindow`, the overlays are placed relative
|
||
/// to its contents.
|
||
public func getChildPosition(_ getChildPosition: @escaping () -> Void) -> Self {
|
||
var newSelf = self
|
||
newSelf.getChildPosition = getChildPosition
|
||
return newSelf
|
||
}
|
||
|
||
/// Set the body for "overlay".
|
||
/// - Parameter body: The body.
|
||
/// - Returns: The widget.
|
||
public func overlay(@ViewBuilder _ body: @escaping () -> Body) -> Self {
|
||
var newSelf = self
|
||
newSelf.overlay = body
|
||
return newSelf
|
||
}
|
||
}
|