How to combine top navigation (navbarPage) and a sidebar menu (sidebarMenu) in shiny

前端 未结 3 1067
长发绾君心
长发绾君心 2021-02-05 16:10

I have a shiny app (using navbarPage) with many tabs and would like to add a sidebarMenu that can be seen no matter which tab is selected. The input values in the sidebar have a

3条回答
  •  猫巷女王i
    2021-02-05 16:56

    This is now possible using bootstraplib

    Github Request to implement this: https://github.com/rstudio/bootstraplib/issues/76

    min reprex:

    # package load ------------------------------------------------------------
    library(shiny)
    library(bootstraplib)
    
    # boot dash layout funs ---------------------------------------------------
    
    
    boot_side_layout <- function(...) {
      div(class = "d-flex wrapper", ...)
    }
    
    boot_sidebar <- function(...) {
      div(
        class = "bg-light border-right sidebar-wrapper",
        div(class = "list-group list-group-flush", ...)
      )
    }
    
    boot_main <- function(...) {
      div(
        class = "page-content-wrapper",
        div(class = "container-fluid", ...)
      )
    }
    
    
    
    # title -------------------------------------------------------------------
    html_title <-
      ''
    
    
    # css ---------------------------------------------------------------------
    
    css_def <- "
    body {
      overflow-x: hidden;
    }
    
    .container-fluid, .container-sm, .container-md, .container-lg, .container-xl {
        padding-left: 0px;
    }
    
    .sidebar-wrapper {
      min-height: 100vh;
      margin-left: -15rem;
      padding-left: 15px;
      padding-right: 15px;
      -webkit-transition: margin .25s ease-out;
      -moz-transition: margin .25s ease-out;
      -o-transition: margin .25s ease-out;
      transition: margin .25s ease-out;
    }
    
    
    .sidebar-wrapper .list-group {
      width: 15rem;
    }
    
    .page-content-wrapper {
      min-width: 100vw;
      padding: 20px;
    }
    
    .wrapper.toggled .sidebar-wrapper {
      margin-left: 0;
    }
    
    .sidebar-wrapper, .page-content-wrapper {
      padding-top: 20px;
    }
    
    .navbar{
      margin-bottom: 0px;
    }
    
    @media (max-width: 768px) {
      .sidebar-wrapper {
        padding-right: 0px;
        padding-left: 0px;
    
      }
    }
    
    @media (min-width: 768px) {
      .sidebar-wrapper {
        margin-left: 0;
      }
    
      .page-content-wrapper {
        min-width: 0;
        width: 100%;
      }
    
      .wrapper.toggled .sidebar-wrapper {
        margin-left: -15rem;
      }
    }
    
    "
    
    
    # app ---------------------------------------------------------------------
    ui <- tagList(
      tags$head(tags$style(HTML(css_def))),
      bootstrap(),
      navbarPage(
        collapsible = TRUE,
        title = HTML(html_title),
        tabPanel(
          "Tab 1",
          boot_side_layout(
            boot_sidebar(
              sliderInput(
                inputId = "bins",
                label = "Number of bins:",
                min = 1,
                max = 50,
                value = 30
              )
            ),
            boot_main(
              fluidRow(column(6, h1("Plot 1")), column(6, h1("Plot 2"))),
              fluidRow(
                column(6, plotOutput(outputId = "distPlot")),
                column(6, plotOutput(outputId = "distPlot2"))
              )
            )
          )
        ),
        tabPanel(
          "Tab 2",
          boot_side_layout(
            boot_sidebar(h1("sidebar input")),
            boot_main(h1("main output"))
          )
        )
      )
    )
    
    server <- function(input, output) {
      output$distPlot <- renderPlot({
        x <- faithful$waiting
        bins <- seq(min(x), max(x), length.out = input$bins + 1)
    
        hist(x,
          breaks = bins, col = "#75AADB", border = "white",
          xlab = "Waiting time to next eruption (in mins)",
          main = "Histogram of waiting times"
        )
      })
    
      output$distPlot2 <- renderPlot({
        x <- faithful$waiting
        bins <- seq(min(x), max(x), length.out = input$bins + 1)
    
        hist(x,
          breaks = bins, col = "#75AADB", border = "white",
          xlab = "Waiting time to next eruption (in mins)",
          main = "Histogram of waiting times"
        )
      })
    }
    
    shinyApp(ui, server)
    

提交回复
热议问题