Migrating to Breakpoints
Migrating to Breakpoints
Libadwaita 1.4 introduces AdwBreakpoint and new adaptive widgets that can
integrate with it, while deprecating the old widgets.
Start Using AdwToolbarView
AdwToolbarView is a widget that can be used instead of GtkBox for
widgets like AdwHeaderBar. It’s required to have the correct header bar
style, especially in sidebars.
Replace your GtkBox with AdwToolbarView and add your header bar and other
toolbars (e.g. GtkSearchBar or AdwTabBar) as top bars using
adw_toolbar_view_add_top_bar() or <child type="top"> in a UI file.
If you have any toolbars at the bottom of the window, e.g. GtkActionBar
or AdwViewSwitcherBar, add them using
adw_toolbar_view_add_bottom_bar() or <child type="bottom">.
Finally, use AdwToolbarView:content to set the content widget. Unlike in
GtkBox, there’s no need to set GtkWidget:vexpand to TRUE, so
that can be removed.
Example:
<object class="GtkBox">
  <property name="orientation">vertical</property>
  <child>
    <object class="AdwHeaderBar">
      <!-- ... -->
    </object>
  </child>
  <child>
    <object class="...">
      <property name="vexpand">True</property>
      <!-- ... -->
    </object>
  </child>
</object>
becomes this:
<object class="AdwToolbarView">
  <child type="top">
    <object class="AdwHeaderBar">
      <!-- ... -->
    </object>
  </child>
  <property name="content">
    <object class="...">
      <!-- ... -->
    </object>
  </property>
</object>
AdwToolbarView defaults to flat header bars, replacing the .flat style
class. To use a raised style, set the AdwToolbarView:top-bar-style
and/or AdwToolbarView:bottom-bar-style properties to
ADW_TOOLBAR_RAISED.
Subsequent sections will assume that you’re using AdwToolbarView.
Start Using AdwWindow or AdwApplicationWindow
All of the new adaptive widgets require using breakpoints, which can only be
added into AdwWindow or AdwApplicationWindow. Alternatively, they can
be added into AdwBreakpointBin when only a specific part of the window
needs to be adaptive.
To migrate from GtkWindow to AdwWindow, or from GtkApplicationWindow to
AdwApplicationWindow, put your header bar and window content into an
AdwToolbarView and set that as the window’s content.
Example:
<object class="GtkWindow">
  <property name="titlebar">
    <!-- titlebar -->
  </property>
  <property name="child">
    <!-- content -->
  </property>
</object>
becomes this:
<object class="AdwWindow">
  <property name="content">
    <object class="AdwToolbarView">
      <child type="top">
        <!-- titlebar -->
      </child>
      <property name="content">
        <!-- content -->
    </object>
  </property>
</object>
Using breakpoints also requires the window to have a minimum size. It can be set
using gtk_widget_set_size_request(), or the
GtkWidget:width-request and GtkWidget:height-request properties.
The target window size that would work on phones both in portrait and landscape orientations is 360x294 pixels:
<object class="AdwWindow">
  <property name="width-request">360</property>
  <property name="height-request">294</property>
  <property name="content">
    <object class="AdwToolbarView">
      <child type="top">
        <!-- titlebar -->
      </child>
      <property name="content">
        <!-- content -->
    </object>
  </property>
</object>
The child widget must completely fit into your minimum size. If it doesn’t, then as soon as you add a breakpoint and resize the window to that size, the contents will overflow and a warning message will be printed.
When checking if the contents fit, consider translations and text scale factor changes. Make sure to leave enough space for text labels, and enable ellipsizing or wrapping if they might not fit.
For GtkLabel this can be done via GtkLabel:ellipsize, or
via GtkLabel:wrap together with GtkLabel:wrap-mode.
For buttons, use GtkButton:can-shrink,
GtkMenuButton:can-shrink, AdwSplitButton:can-shrink, or
AdwButtonContent:can-shrink.
Subsequent sections will assume that you’re using AdwWindow or
AdwApplicationWindow.
Start using AdwHeaderBar
AdwHeaderBar provides additional integration with some of the widgets
listed below, compared to GtkHeaderBar, and can make them noticeably
easier to use.
<object class="GtkHeaderBar">
  <property name="show-title-buttons">False</property>
</object>
Replace your header bar with AdwHeaderBar, and
GtkHeaderBar:show-title-buttons with a combination of
AdwHeaderBar:show-start-title-buttons and
AdwHeaderBar:show-end-title-buttons.
<object class="AdwHeaderBar">
  <property name="show-start-title-buttons">False</property>
  <property name="show-end-title-buttons">False</property>
</object>
Replace AdwLeaflet
AdwLeaflet can be replaced by either AdwNavigationView or
AdwNavigationSplitView, depending on how it’s used.
Navigation
Leaflets that have their AdwLeaflet:can-unfold property set to FALSE
can be replaced using AdwNavigationView.
A typical use case with two pages looks as follows:
<object class="AdwLeaflet">
  <property name="can-unfold">False</property>
  <property name="can-navigate-back">True</property>
  <child>
    <object class="AdwLeafletPage">
      <property name="name">page-1</property>
      <property name="child">
        <object class="AdwToolbarView">
          <child type="top">
            <object class="GtkHeaderBar">
              <property name="title-widget">
                <object class="AdwWindowTitle">
                  <property name="title" translatable="yes">Page 1</property>
                </object>
              </property>
            </object>
          </child>
          <property name="content">
            <!-- ... -->
          </property>
        </object>
      </property>
    </object>
  </child>
  <child>
    <object class="AdwLeafletPage">
      <property name="name">page-2</property>
      <property name="child">
        <object class="AdwToolbarView">
          <child type="top">
            <object class="GtkHeaderBar">
              <child type="start">
                <object class="GtkButton">
                  <property name="icon-name">go-previous-symbolic</property>
                  <property name="tooltip-text" translatable="yes">Back</property>
                  <signal name="clicked" handler="back_clicked_cb" swapped="yes"/>
                </object>
              </child>
              <property name="title-widget">
                <object class="AdwWindowTitle">
                  <property name="title" translatable="yes">Page 2</property>
                </object>
              </property>
            </object>
          </child>
          <property name="content">
            <!-- ... -->
          </property>
        </object>
      </property>
    </object>
  </child>
</object>
Replace the leaflet with AdwNavigationView and AdwLeafletPage with
AdwNavigationPage. AdwNavigationView requires all children to be
AdwNavigationPage, so if you didn’t explicitly specify AdwLeafletPage,
wrap your child widgets into AdwNavigationPage instead.
Use AdwNavigationPage:title instead of manually adding titles to your
header bars. Even if you need a subtitle, AdwNavigationPage:title should
always be set to something meaningful, as it will be used as a tooltip for the
back button and label in its context menu, as well as read out by the screen reader.
Replace AdwLeafletPage:name with AdwNavigationPage:tag and
remove the back button as AdwHeaderBar already provides one. If back
buttons are unwanted, set AdwHeaderBar:show-back-button on your header
bar to FALSE.
AdwNavigationView always has back gestures and shortcuts enabled. To disable
them (as well as the back button) on a specific page, set its
AdwNavigationPage:can-pop property to FALSE.
If you’re using GtkHeaderBar, replace it with AdwHeaderBar as well:
<object class="AdwNavigationView">
  <child>
    <object class="AdwNavigationPage">
      <property name="title" translatable="yes">Page 1</property>
      <property name="tag">page-1</property>
      <property name="child">
        <object class="AdwToolbarView">
          <child type="top">
            <object class="AdwHeaderBar"/>
          </child>
          <property name="content">
            <!-- ... -->
          </property>
        </object>
      </property>
    </object>
  </child>
  <child>
    <object class="AdwNavigationPage">
      <property name="title" translatable="yes">Page 2</property>
      <property name="tag">page-2</property>
      <property name="child">
        <object class="AdwToolbarView">
          <child type="top">
            <object class="AdwHeaderBar"/>
          </child>
          <property name="content">
            <!-- ... -->
          </property>
        </object>
      </property>
    </object>
  </child>
</object>
Replace adw_leaflet_navigate() calls with
adw_navigation_view_push() for ADW_NAVIGATION_DIRECTION_FORWARD and
adw_navigation_view_pop() for ADW_NAVIGATION_DIRECTION_BACK. It’s also
possible to push a page using the navigation.push action and the page’s tag as
parameter, or pop the visible page using the navigation.pop action.
If you were using nested AdwLeaflets or GtkStack with pages inside it
to arrange a non-linear navigation structure (for example, page 1 that can lead
to page 2 and page 3, both of which lead back to page 1), AdwNavigationView
can support that directly.
To replace AdwLeaflet:can-navigate-forward, connect to the
AdwNavigationView::get-next-page signal.
Sidebar
Leaflets that implement adaptive sidebar layouts can be replaced with
AdwNavigationSplitView.
A typical use case looks as follows:
<object class="AdwWindow">
  <property name="content">
    <object class="AdwLeaflet" id="leaflet">
      <property name="can-navigate-back">True</property>
      <child>
        <object class="AdwToolbarView">
          <child type="top">
            <object class="AdwHeaderBar">
              <property name="show-end-title-buttons"
                              bind-object="leaflet"
                              bind-property="folded"
                              bind-flags="sync-create"/>
              <property name="title-widget">
                <object class="AdwWindowTitle">
                  <property name="title" translatable="yes">Sidebar</property>
                </object>
              </property>
            </object>
          </child>
          <property name="content">
            <!-- sidebar -->
          </property>
        </object>
      </child>
      <child>
        <object class="AdwLeafletPage">
          <property name="navigatable">False</property>
          <property name="child">
            <object class="GtkSeparator"/>
          </property>
        </object>
      </child>
      <child>
        <object class="AdwToolbarView">
          <property name="hexpand">True</property>
          <child type="top">
            <object class="AdwHeaderBar">
              <property name="show-start-title-buttons"
                              bind-object="leaflet"
                              bind-property="folded"
                              bind-flags="sync-create"/>
              <child type="start">
                <object class="GtkButton">
                  <property name="visible"
                                  bind-object="leaflet"
                                  bind-property="folded"
                                  bind-flags="sync-create"/>
                  <property name="icon-name">go-previous-symbolic</property>
                </object>
              </child>
              <property name="title-widget">
                <object class="AdwWindowTitle">
                  <property name="title" translatable="yes">Content</property>
                </object>
              </property>
            </object>
          </child>
          <property name="content">
            <!-- content -->
          </property>
        </object>
      </child>
    </object>
  </property>
</object>
Replace AdwLeaflet with AdwNavigationSplitView and AdwLeafletPage with
AdwNavigationPage. AdwNavigationSplitView requires all children to be
AdwNavigationPage, so if you didn’t explicitly specify AdwLeafletPage,
wrap your child widgets into AdwNavigationPage instead.
Remove the separator child, AdwNavigationSplitView provides a separator as
part of its styling.
Stop binding window button visibility to the leaflet’s folded state, as
AdwHeaderBar handles that automatically when inside AdwNavigationSplitView.
Use AdwNavigationPage:title instead of manually adding titles to your
header bars. Even if you need a subtitle, AdwNavigationPage:title should
always be set to something meaningful, as it will be used as a tooltip for the
back button, as well as read out by the screen reader.
Replace AdwLeafletPage:name with AdwNavigationPage:tag and
remove the back button from your content header bar, as AdwHeaderBar
already provides one. If back buttons are unwanted, set
AdwHeaderBar:show-back-button on your header bar to FALSE.
It’s also possible to disable the back button, as well as related shortcuts and
actions on the content page by setting its AdwNavigationPage:can-pop
property to FALSE.
Add a breakpoint with a max-width condition to your window. To accommodate
the Large Text setting, the width value should be using sp unit, e.g. 400sp.
Add a setter to your breakpoint, setting the
AdwNavigationSplitView:collapsed property to TRUE.
<object class="AdwWindow">
  <property name="width-request">360</property>
  <property name="height-request">200</property>
  <child>
    <object class="AdwBreakpoint">
      <condition>max-width: 400sp</condition>
      <setter object="split_view" property="collapsed">True</setter>
    </object>
  </child>
  <property name="content">
    <object class="AdwNavigationSplitView" id="split_view">
      <property name="sidebar">
        <object class="AdwNavigationPage">
          <property name="title" translatable="yes">Sidebar</property>
          <property name="child">
            <object class="AdwToolbarView">
              <child type="top">
                <object class="AdwHeaderBar"/>
              </child>
              <property name="content">
                <!-- sidebar -->
              </property>
            </object>
          </property>
        </object>
      </property>
      <property name="content">
        <object class="AdwNavigationPage">
          <property name="title" translatable="yes">Content</property>
          <property name="child">
            <object class="AdwToolbarView">
              <child type="top">
                <object class="AdwHeaderBar"/>
              </child>
              <property name="content">
                <!-- content -->
              </property>
            </object>
          </property>
        </object>
      </property>
    </object>
  </property>
</object>
Replace AdwLeaflet:visible-child and/or
AdwLeaflet:visible-child-name uses with
AdwNavigationSplitView:show-content. It’s also
possible to show content using using the navigation.push action with the
content page’s tag as a parameter, or show sidebar using the navigation.pop
action.
By default, AdwNavigationSplitView will dynamically resize the sidebar
depending on its own width. If that behavior is unwanted, set
AdwNavigationSplitView:min-sidebar-width and
AdwNavigationSplitView:max-sidebar-width to 0.
For triple-pane layouts, see the adaptive layouts page.
Other Uses
Other uses of AdwLeaflet, e.g. in vertical orientation, have no direct
replacement, but can be replicated using e.g. GtkBox, the
GtkWidget:visible property and breakpoints.
Replace AdwFlap
AdwFlap can be used for multiple different things, and is generally replaced
with AdwOverlaySplitView. That widget has similar API to AdwFlap and its
common uses can be directly replaced.
Utility pane
The main AdwFlap use case is
utility panes.
A common case looks as follows:
<object class="AdwWindow">
  <property name="content">
    <object class="AdwToolbarView">
      <property name="top-bar-style">raised</property>
      <child type="top">
        <object class="AdwHeaderBar">
          <object class="GtkToggleButton" id="toggle_pane_button">
            <property name="icon-name">sidebar-show-symbolic</property>
            <property name="active">True</property>
          </object>
        </object>
      </child>
      <property name="content">
        <object class="AdwFlap">
          <property name="reveal-flap"
                    bind-source="toggle_pane_button"
                    bind-property="active"
                    bind-flags="sync-create|bidirectional"/>
          <property name="flap">
            <!-- utility pane -->
          </property>
          <property name="separator">
            <object class="GtkSeparator"/>
          </property>
          <property name="content">
            <!-- main view -->
          </property>
        </object>
      </property>
    </object>
  </property>
</object>
Replace AdwFlap with AdwOverlaySplitView, the flap property with
AdwOverlaySplitView:sidebar and the reveal-flap property with
AdwOverlaySplitView:show-sidebar.
Remove the separator child, AdwOverlaySplitView provides a separator as part
of its styling.
Add an ID to your split view if you’re using UI files.
Add a breakpoint with a max-width condition to your window. To accommodate
the Large Text setting, the width value should be using sp unit, e.g. 400sp.
Add a setter to your breakpoint, setting the
AdwOverlaySplitView:collapsed property to TRUE.
<object class="AdwWindow">
  <property name="width-request">360</property>
  <property name="height-request">200</property>
  <child>
    <object class="AdwBreakpoint">
      <condition>max-width: 400sp</condition>
      <setter object="split_view" property="collapsed">True</setter>
    </object>
  </child>
  <property name="content">
    <object class="AdwToolbarView">
      <property name="top-bar-style">raised</property>
      <child type="top">
        <object class="AdwHeaderBar">
          <object class="GtkToggleButton" id="toggle_pane_button">
            <property name="icon-name">sidebar-show-symbolic</property>
            <property name="active">True</property>
          </object>
        </object>
      </child>
      <property name="content">
        <object class="AdwOverlaySplitView" id="split_view">
          <property name="show-sidebar"
                    bind-source="toggle_pane_button"
                    bind-property="active"
                    bind-flags="sync-create|bidirectional"/>
          <property name="sidebar">
            <!-- utility pane -->
          </property>
          <property name="content">
            <!-- main view -->
          </property>
        </object>
      </property>
    </object>
  </property>
</object>
The AdwFlap:locked property can be replaced with
AdwOverlaySplitView:pin-sidebar.
Sidebar
Sidebar-style AdwFlap use cases, with split header bars, are handled similarly
to utility panes. One additional difference is that, like in
AdwNavigationSplitView, AdwHeaderBar can manage window button visibility
automatically when using AdwOverlaySplitView, so there’s no need to bind their
visibility to the AdwFlap:folded property anymore.
While AdwOverlaySplitView doesn’t require using AdwNavigationPage, it can
still be used to provide header bar titles insead of using AdwWindowTitle.
Fullscreen Header Bar
A vertical AdwFlap can also be used to handle header bars in fullscreen,
specifically for either overlaying a header bar above content in fullscreen,
or showing it next to the content otherwise.
This can be replaced with AdwToolbarView and the
AdwToolbarView:extend-content-to-top-edge property.
Other Uses
Other use cases of AdwFlap have no direct replacement but can be replicated
using e.g. GtkBox, GtkRevealer, the
GtkWidget:visible property, and breakpoints.
Replace AdwViewSwitcherTitle
AdwViewSwitcherTitle can be replaced by using AdwViewSwitcher together
with a breakpoint.
A typical AdwViewSwitcherTitle use looks as follows:
<object class="AdwWindow">
  <property name="width-request">360</property>
  <property name="height-request">294</property>
  <property name="content">
    <object class="AdwToolbarView">
      <child type="top">
        <object class="AdwHeaderBar">
          <property name="centering-policy">strict</property>
          <property name="title-widget">
            <object class="AdwViewSwitcherTitle" id="title">
              <property name="stack">stack</property>
              <property name="title" translatable="yes">Title</property>
            </object>
          </property>
        </object>
      </child>
      <property name="content">
        <object class="AdwViewStack" id="stack">
          <!-- ... -->
        </object>
      </property>
      <child type="bottom">
        <object class="AdwViewSwitcherBar">
          <property name="stack">stack</property>
          <property name="reveal"
                    bind-source="title"
                    bind-property="title-visible"
                    bind-flags="sync-create"/>
        </object>
      </child>
    </object>
  </property>
</object>
Replace your AdwViewSwitcherTitle with a regular AdwViewSwitcher and set
the ADW_VIEW_SWITCHER_POLICY_WIDE policy on it.
Also remove the AdwViewSwitcherBar:reveal property binding on your
switcher bar, and stop setting AdwHeaderBar:centering-policy on your
header bar.
Add an ID to your header bar and switcher bar if you’re using UI files.
<object class="AdwHeaderBar" id="header_bar">
  <!-- ... -->
  <property name="title-widget">
    <object class="AdwViewSwitcher">
      <property name="policy">wide</property>
      <property name="stack">stack</property>
    </object>
  </property>
  <!-- ... -->
</object>
<!-- ... -->
<object class="AdwViewSwitcherBar" id="switcher_bar">
  <property name="stack">stack</property>
</object>
Add a breakpoint with a max-width condition to your window. To accommodate
the Large Text setting, the width value should be using sp unit.
Recommended width values depending on the number of pages:
- 2 pages: 400sp
- 3 pages: 450sp
- 4 pages: 550sp
You may need to tweak the value to match your application’s layout. It needs to be large enough that the view switcher is not yet ellipsized at that width, but small enough that the view switcher doesn’t move to the bottom yet at common window sizes.
The rest of this section will assume 550sp as the threshold.
Add two setters to your breakpoint, unsetting the header bar’s titlebar widget,
and setting the switcher bar’s reveal property to TRUE.
Specify the header bar’s title using GtkWindow:title or
AdwNavigationPage:title.
<object class="AdwWindow">
  <property name="title" translatable="yes">Title</property>
  <child>
    <object class="AdwBreakpoint">
      <condition>max-width: 550sp</condition>
      <setter object="header_bar" property="title-widget"/>
      <setter object="switcher_bar" property="reveal">True</setter>
    </object>
  </child>
  <property name="child">
    <object class="AdwToolbarView">
      <child type="top">
        <object class="AdwHeaderBar" id="header_bar">
          <property name="title-widget">
            <object class="AdwViewSwitcher">
              <property name="policy">wide</property>
              <property name="stack">stack</property>
            </object>
          </property>
        </object>
      </child>
      <property name="content">
        <object class="AdwViewStack" id="stack">
          <!-- ... -->
        </object>
      </property>
      <child type="bottom">
        <object class="AdwViewSwitcherBar" id="switcher_bar">
          <property name="stack">stack</property>
        </object>
      </child>
    </object>
  </property>
</object>
Subtitle
If you were using AdwViewSwitcherTitle:subtitle, use a GtkStack
containing an AdwViewSwitcher and AdwWindowTitle, and switch the stack’s
visible page from your breakpoint.
<object class="AdwWindow">
  <child>
    <object class="AdwBreakpoint">
      <condition>max-width: 550sp</condition>
      <setter object="title_stack" property="visible-child">window_title</property>
      <setter object="switcher_bar" property="reveal">True</property>
    </object>
  </child>
  <property name="child">
    <object class="AdwToolbarView">
      <child type="top">
        <object class="AdwHeaderBar">
          <property name="title-widget">
            <object class="GtkStack" id="title_stack">
              <child>
                <object class="AdwViewSwitcher">
                  <property name="policy">wide</property>
                  <property name="stack">stack</property>
                </object>
              </child>
              <child>
                <object class="AdwWindowTitle" id="window_title">
                  <property name="title" translatable="yes">Title</property>
                  <property name="subtitle" translatable="yes">Subtitle</property>
                </object>
              </child>
            </object>
          </property>
        </object>
      </child>
      <property name="content">
        <object class="AdwViewStack" id="stack">
          <!-- ... -->
        </object>
      </property>
      <child type="bottom">
        <object class="AdwViewSwitcherBar" id="switcher_bar">
          <property name="stack">stack</property>
        </object>
      </child>
    </object>
  </property>
</object>
Replace AdwSqueezer
AdwSqueezer can be replaced with breakpoints.
For example, to migrate the following example:
<object class="AdwSqueezer">
  <property name="homogeneous">False</property>
  <child>
    <object class="GtkBox">
      <property name="spacing">6</property>
      <child>
        <object class="GtkButton">
          <property name="label">Button 1</property>
        </object>
      </child>
      <child>
        <object class="GtkButton">
          <property name="label">Button 2</property>
        </object>
      </child>
    </object>
  </child>
  <child>
    <object class="GtkBox">
      <property name="orientation">vertical</property>
      <property name="spacing">6</property>
      <child>
        <object class="GtkButton">
          <property name="label">Button 1</property>
        </object>
      </child>
      <child>
        <object class="GtkButton">
          <property name="label">Button 2</property>
        </object>
      </child>
    </object>
  </child>
</object>
use a single GtkBox, and a breakpoint toggling the box’s
GtkOrientable:orientation property:
<object class="GtkBox" id="box">
  <property name="spacing">6</property>
  <child>
    <object class="GtkButton">
      <property name="label">Button 1</property>
    </object>
  </child>
  <child>
    <object class="GtkButton">
      <property name="label">Button 2</property>
    </object>
  </child>
</object>
<!-- ... -->
<object class="AdwBreakpoint">
  <condition>max-width: 400sp</condition>
  <setter object="box" property="orientation">vertical</property>
</object>
Alternatively, a GtkStack or the GtkWidget:visible
property can be used to switch between separate widgets, like in AdwSqueezer.
Migrate AdwPreferencesWindow Subpages
AdwPreferencesWindow can now use AdwNavigationView for its subpages instead
of AdwLeaflet. As such, adw_preferences_window_present_subpage() and
adw_preferences_window_close_subpage() have been deprecated in favor of
adw_preferences_window_push_subpage() and
adw_preferences_window_pop_subpage().
Wrap your subpages into AdwNavigationPage, stop setting the header bar
titles manually and use AdwNavigationPage:title instead, remove your
back buttons if you had any. If back buttons are unwanted, set
AdwHeaderBar:show-back-button on your header bar to FALSE.
AdwPreferencesWindow always has back gestures and shortcuts enabled for the
new subpage API. To disable them (as well as the back button) on a specific
page, set its AdwNavigationPage:can-pop property to FALSE.
Unlike adw_preferences_window_present_subpage(), repeatedly calling
adw_preferences_window_push_subpage() with different pages won’t replace the
current page, so if you had an AdwLeaflet inside your subpage, you can use
adw_preferences_window_push_subpage() directly instead.