Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat!: Introduce navbar_options() #1141

Merged
merged 34 commits into from
Dec 2, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
01ff848
chore: code style
gadenbuie Nov 26, 2024
9ed36bb
feat!(navbar_options): Introduce `navbar_options()`, deprecate navbar…
gadenbuie Nov 26, 2024
0612d30
feat: Add `underline` to `navbar_options()`
gadenbuie Nov 26, 2024
c726c6b
feat!(page_navbar): Use `navbar_options()`
gadenbuie Nov 26, 2024
284d10f
docs: Move `navbar_options()` to its own docs page
gadenbuie Nov 26, 2024
8abc2b7
chore(page_navbar): Move deprecated arguments to the end
gadenbuie Nov 26, 2024
388720b
docs: Update navset examples
gadenbuie Nov 26, 2024
33732ae
Resave data (GitHub Action)
gadenbuie Nov 26, 2024
f24c0c6
feat: Prevent deprecation warnings from `shiny::navbarPage()`
gadenbuie Nov 26, 2024
43e01c6
chore: small edits/fixes
gadenbuie Nov 26, 2024
15c42d4
chore: only warn if values are different
gadenbuie Nov 26, 2024
85b282a
chore: restore legacy code for minimal diff
gadenbuie Nov 26, 2024
a5b2868
chore(navbar_options): Expose defaults in function signature
gadenbuie Nov 27, 2024
5e1f0e2
Resave data (GitHub Action)
gadenbuie Nov 27, 2024
b017dc5
Update website deps (GitHub Action)
gadenbuie Nov 27, 2024
7d473ee
refactor: simplify navbar_options()
gadenbuie Nov 27, 2024
fcd6d5c
Revert "Update website deps (GitHub Action)"
gadenbuie Nov 27, 2024
3be0434
chore: use version 2 for internal data
gadenbuie Nov 27, 2024
bd43c4d
Resave data (GitHub Action)
gadenbuie Nov 27, 2024
a85ff8a
feat: Make `underline = TRUE` the default
gadenbuie Nov 27, 2024
0fc9985
chore: style
gadenbuie Nov 27, 2024
bf0f402
refactor: consolidate deprecation warning into single message
gadenbuie Nov 27, 2024
6f66bb7
tests: update snapshots
gadenbuie Nov 27, 2024
76a9692
`devtools::document()` (GitHub Actions)
gadenbuie Nov 27, 2024
60b84ce
docs: Add news item
gadenbuie Nov 27, 2024
3f0d9bd
docs: Add a note about `navset_bar()` in the examples page
gadenbuie Nov 27, 2024
7028fba
`devtools::document()` (GitHub Actions)
gadenbuie Nov 27, 2024
bc87599
chore: add todo note to remember to remove code when fully deprecated
gadenbuie Nov 27, 2024
cf8d423
chore: Include `...` for future expansion
gadenbuie Nov 27, 2024
ee683e7
refactor: Use attributes on `navbar_options()` to track source
gadenbuie Dec 2, 2024
b919ae6
chore: Apply changes from review
gadenbuie Dec 2, 2024
963a1eb
chore: edit wording in comment
gadenbuie Dec 2, 2024
b163ffa
feat: resolved navbar options equivalent to `navbar_options()`
gadenbuie Dec 2, 2024
01a273c
chore: Error if `...` includes unused options
gadenbuie Dec 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion DESCRIPTION
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ License: MIT + file LICENSE
URL: https://rstudio.github.io/bslib/, https://github.com/rstudio/bslib
BugReports: https://github.com/rstudio/bslib/issues
Depends:
R (>= 2.10)
R (>= 3.5)
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
Imports:
base64enc,
cachem,
Expand Down
2 changes: 2 additions & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ S3method(is_fillable_container,default)
S3method(is_fillable_container,htmlwidget)
S3method(print,bslib_breakpoints)
S3method(print,bslib_fragment)
S3method(print,bslib_navbar_options)
S3method(print,bslib_page)
S3method(print,bslib_showcase_layout)
S3method(print,bslib_value_box_theme)
Expand Down Expand Up @@ -105,6 +106,7 @@ export(nav_remove)
export(nav_select)
export(nav_show)
export(nav_spacer)
export(navbar_options)
export(navs_bar)
export(navs_hidden)
export(navs_pill)
Expand Down
254 changes: 236 additions & 18 deletions R/navs-legacy.R
Original file line number Diff line number Diff line change
Expand Up @@ -105,36 +105,254 @@ navset_hidden <- function(..., id = NULL, selected = NULL,
#' character vector, matching the `value` of [nav_panel()]s to be filled, may
#' also be provided. Note that, if a `sidebar` is provided, `fillable` makes
#' the main content portion fillable.
#' @param bg a CSS color to use for the navbar's background color.
#' @param inverse Either `TRUE` for a light text color or `FALSE` for a dark
#' text color. If `"auto"` (the default), the best contrast to `bg` is chosen.
#' @param navbar_options Options to control the appearance and behavior of the
#' navbar. Use [navbar_options()] to create the list of options.
#' @param position `r lifecycle::badge("deprecated")` Please use
#' [`navbar_options = navbar_options(position=)`][navbar_options] instead.
#' @param collapsible `r lifecycle::badge("deprecated")` Please use
#' [`navbar_options = navbar_options(collapsible=)`][navbar_options] instead.
#' @param bg `r lifecycle::badge("deprecated")` Please use
#' [`navbar_options = navbar_options(bg=)`][navbar_options] instead.
#' @param inverse `r lifecycle::badge("deprecated")` Please use
#' [`navbar_options = navbar_options(inverse=)`][navbar_options] instead.
#'
#' @export
#' @rdname navset
navset_bar <- function(..., title = NULL, id = NULL, selected = NULL,
sidebar = NULL, fillable = TRUE,
gap = NULL, padding = NULL,
# TODO: add sticky-top as well?
position = c("static-top", "fixed-top", "fixed-bottom"),
header = NULL, footer = NULL,
bg = NULL, inverse = "auto",
collapsible = TRUE, fluid = TRUE) {
navset_bar <- function(
...,
title = NULL,
id = NULL,
selected = NULL,
sidebar = NULL,
fillable = TRUE,
gap = NULL,
padding = NULL,
header = NULL,
footer = NULL,
fluid = TRUE,
navbar_options = NULL,
position = deprecated(),
bg = deprecated(),
inverse = deprecated(),
collapsible = deprecated()
) {
padding <- validateCssPadding(padding)
gap <- validateCssUnit(gap)

navs_bar_(
..., title = title, id = id, selected = selected,
sidebar = sidebar, fillable = fillable,
gap = gap, padding = padding,
.navbar_options <- navbar_options_resolve_deprecated(
options_user = navbar_options,
position = position,
header = header, footer = footer,
bg = bg, inverse = inverse,
collapsible = collapsible, fluid = fluid,
bg = bg,
inverse = inverse,
collapsible = collapsible
)

navs_bar_(
...,
title = title,
id = id,
selected = selected,
sidebar = sidebar,
fillable = fillable,
gap = gap,
padding = padding,
header = header,
footer = footer,
fluid = fluid,
position = .navbar_options$position,
bg = .navbar_options$bg,
inverse = .navbar_options$inverse,
collapsible = .navbar_options$collapsible,
underline = .navbar_options$underline %||% FALSE,
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
# theme is only used to determine whether legacy style markup should be used
# (and, at least at the moment, we don't need legacy markup for this exported function)
theme = bs_theme()
)
}

#' Create a set of navbar options
#'
#' A `navbar_options()` object captures options specific to the appearance and
#' behavior of the navbar, independent from the content displayed on the page.
#' This helper should be used to create the list of options expected by
#' `navbar_options` in [page_navbar()] and [navset_bar()].
#'
#' ## Changelog
#'
#' This function was introduced in \pkg{bslib} v0.9.0, replacing the `position`,
#' `bg`, `inverse`, `collapsible` and `underline` arguments of [page_navbar()]
#' and [navset_bar()]. Those arguments are deprecated with a warning and will be
#' removed in a future version of \pkg{bslib}.
#'
#' @examples
#' navbar_options(position = "static-top", bg = "#2e9f7d", underline = FALSE)
#'
#' @inheritParams shiny::navbarPage
#' @param bg a CSS color to use for the navbar's background color.
#' @param inverse Either `TRUE` for a light text color or `FALSE` for a dark
#' text color. If `"auto"` (the default), the best contrast to `bg` is chosen.
#' @param underline Whether or not to add underline styling to page links when
#' active or focused.
#'
#' @returns Returns a list of navbar options.
#'
#' @export
navbar_options <- function(
position = c("static-top", "fixed-top", "fixed-bottom"),
bg = NULL,
inverse = "auto",
collapsible = TRUE,
underline = NULL
) {
arg_missing <- list(
position = missing(position),
bg = missing(bg),
inverse = missing(inverse),
collapsible = missing(collapsible),
underline = missing(underline)
)

maybe_default <- function(arg, value) {
if (isFALSE(arg_missing[[arg]])) return(value)
as_bslib_default(value)
}

if (!is.null(position)) {
# TODO: add sticky-top as well?
position <- rlang::arg_match(position)
}

opts <- list(
position = position,
bg = bg,
inverse = inverse,
collapsible = collapsible,
underline = underline
)
opts <- dropNulls(opts)
for (opt in names(opts)) {
opts[[opt]] <- maybe_default(opt, opts[[opt]])
}
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved

structure(opts, class = c("bslib_navbar_options", "list"))
}

as_bslib_default <- function(x) {
if (is.null(x)) return(x)
attr(x, "bslib_default") <- TRUE
attr(x, "waldo_opts") <- list(ignore_attr = TRUE)
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
x
}

is_bslib_default <- function(x) {
isTRUE(attr(x, "bslib_default"))
}

navbar_options_resolve_deprecated <- function(
options_user = NULL,
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
position = deprecated(),
bg = deprecated(),
inverse = deprecated(),
collapsible = deprecated(),
underline = deprecated(),
.fn_caller = "navset_bar",
.warn_deprecated = TRUE
) {
fn_arg <- function(arg) sprintf("%s(%s=)", .fn_caller, arg)

warn_deprecated <- function(arg) {
if (!.warn_deprecated) return()
lifecycle::deprecate_warn(
"0.9.0",
fn_arg(arg),
details = paste(
"Navbar options have been consolidated into a single `options` argument.",
sprintf(
"Use the `%s` argument of `navbar_options()` instead.",
arg
)
)
)
}

old_opts <- list()
if (lifecycle::is_present(position)) {
warn_deprecated("position")
old_opts$position <- position
}
if (lifecycle::is_present(bg)) {
warn_deprecated("bg")
old_opts$bg <- bg
}
if (lifecycle::is_present(inverse)) {
warn_deprecated("inverse")
old_opts$inverse <- inverse
}
if (lifecycle::is_present(collapsible)) {
warn_deprecated("collapsible")
old_opts$collapsible <- collapsible
}
if (lifecycle::is_present(underline)) {
warn_deprecated("underline")
old_opts$underline <- underline
}

# Consolidate `navbar_options` (options_user) with direction options taking
# the direct option if the user option is a default value, warning if
# otherwise ignored.
options_user <- options_user %||% list()
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
ignored <- c()
for (opt in names(old_opts)) {
if (opt %in% names(options_user) && !is_bslib_default(options_user[[opt]])) {
if (identical(old_opts[[opt]], options_user[[opt]])) {
next
}
ignored <- c(ignored, opt)
} else {
options_user[[opt]] <- old_opts[[opt]]
}
}

if (length(ignored) > 0) {
rlang::warn(
c(
sprintf(
"`%s` %s provided twice: once directly and once in `navbar_options`.",
paste(ignored, collapse = "`, `"),
if (length(ignored) == 1) "was" else "were"
),
"The deprecated direct option(s) will be ignored and the values from `navbar_options` will be used."
),
call = rlang::caller_call()
)
}

ret <- rlang::exec(navbar_options, !!!options_user)
# strip bslib_default attributes now that we've resolved defaults
lapply(ret, c)
gadenbuie marked this conversation as resolved.
Show resolved Hide resolved
}

#' @export
print.bslib_navbar_options <- function(x, ...) {
cat("<bslib_navbar_options>\n")

if (length(x) == 0) {
return(invisible(x))
}

fields <- names(dropNulls(x))
opt_w <- max(nchar(fields))
for (opt in fields) {
value <- x[[opt]]
if (is_bslib_default(value)) {
value <- sprintf("(%s)", value)
}
cat(sprintf("%*s", opt_w, opt), ": ", value, "\n", sep = "")
}

invisible(x)
}

# This internal version of navs_bar() exists so both it and page_navbar()
# (and thus shiny::navbarPage()) can use it. And in the page_navbar() case,
# we can use addition theme information as an indication of whether we need
Expand Down
63 changes: 43 additions & 20 deletions R/page.R
Original file line number Diff line number Diff line change
Expand Up @@ -333,8 +333,8 @@ maybe_page_sidebar <- function(x) {
#'
#' @param fillable_mobile Whether or not `fillable` pages should fill the viewport's
#' height on mobile devices (i.e., narrow windows).
#' @param underline Whether or not to add underline styling to page links when
#' active or focused.
#' @param underline `r lifecycle::badge("deprecated")` Please use
#' [`navbar_options = navbar_options(underline=)`][navbar_options] instead.
#' @param window_title the browser window title. The default value, `NA`, means
#' to use any character strings that appear in `title` (if none are found, the
#' host URL of the page is displayed by default).
Expand Down Expand Up @@ -397,31 +397,44 @@ page_navbar <- function(
fillable_mobile = FALSE,
gap = NULL,
padding = NULL,
position = c("static-top", "fixed-top", "fixed-bottom"),
header = NULL,
footer = NULL,
bg = NULL,
inverse = "auto",
underline = TRUE,
collapsible = TRUE,
navbar_options = NULL,
fluid = TRUE,
theme = bs_theme(),
window_title = NA,
lang = NULL
lang = NULL,
position = deprecated(),
bg = deprecated(),
inverse = deprecated(),
underline = deprecated(),
collapsible = deprecated()
) {

sidebar <- maybe_page_sidebar(sidebar)

padding <- validateCssPadding(padding)
gap <- validateCssUnit(gap)

# Change behavior when called by Shiny
# TODO: Coordinate with next bslib version bump in Shiny to use the new interface
was_called_by_shiny <- isNamespaceLoaded("shiny") && identical(rlang::caller_fn(), shiny::navbarPage)

.navbar_options <- navbar_options_resolve_deprecated(
options_user = navbar_options,
position = position,
bg = bg,
inverse = inverse,
collapsible = collapsible,
underline = underline,
.fn_caller = "page_navbar",
.warn_deprecated = !was_called_by_shiny
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would really really like to decouple shiny::navbarPage() from bslib::page_navbar(). I'm thinking it'd be an update to add page_navbar_legacy() that we'd export but mark as internal so it doesn't show up in docs and indices.

What I want out of that change is to have a clear line around what parts of the navbar API we shouldn't touch without thinking carefully about backwards compatibility or API changes and which parts are owned by bslib. Currently it feels too entangled and over time entropy wins leading to more entanglement.

)

# Default to fillable = F when this is called from shiny::navbarPage()
# TODO: update shiny::navbarPage() to set fillable = FALSE and get rid of this hack
if (missing(fillable)) {
isNavbarPage <- isNamespaceLoaded("shiny") && identical(rlang::caller_fn(), shiny::navbarPage)
if (isNavbarPage) {
fillable <- FALSE
}
if (missing(fillable) && was_called_by_shiny) {
fillable <- FALSE
}

# If a sidebar is provided, we want the layout_sidebar(fill = TRUE) component
Expand All @@ -439,13 +452,23 @@ page_navbar <- function(
class = "bslib-page-navbar",
class = if (!is.null(sidebar)) "has-page-sidebar",
navs_bar_(
..., title = title, id = id, selected = selected,
sidebar = sidebar, fillable = fillable,
gap = gap, padding = padding,
position = match.arg(position), header = header,
footer = footer, bg = bg, inverse = inverse,
underline = underline, collapsible = collapsible,
fluid = fluid, theme = theme
...,
title = title,
id = id,
selected = selected,
sidebar = sidebar,
fillable = fillable,
gap = gap,
padding = padding,
header = header,
footer = footer,
position = .navbar_options$position,
bg = .navbar_options$bg,
inverse = .navbar_options$inverse,
underline = .navbar_options$underline %||% TRUE,
collapsible = .navbar_options$collapsible,
fluid = fluid,
theme = theme
)
)
}
Expand Down
Binary file modified R/sysdata.rda
Binary file not shown.
Loading