0

0

Shiny 应用中实现可滚动 Sortable 列表的实践指南

心靈之曲

心靈之曲

发布时间:2025-09-28 15:51:01

|

981人浏览过

|

来源于php中文网

原创

shiny 应用中实现可滚动 sortable 列表的实践指南

本文详细介绍了如何在 Shiny 应用中创建具有滚动功能的 sortable 列表。通过应用 CSS 样式 max-height 和 overflow-y: auto 到 rank_list 容器,用户可以有效管理内容过多的列表,确保界面整洁且用户体验良好。教程将提供完整的代码示例和详细解释,帮助开发者轻松实现这一功能。

理解 Sortable 列表与滚动需求

在 Shiny 应用中,sortable 包提供了一种直观的方式来创建可拖拽排序的列表,极大地增强了用户交互体验。然而,当这些列表包含大量项目时,其内容可能会超出容器的可见区域,导致界面布局混乱,甚至部分内容无法访问。为了解决这一问题,我们需要为 sortable 列表容器添加滚动功能,使其在内容溢出时自动显示滚动条,从而保持界面的整洁和良好的用户体验。

本教程将以一个典型的 sortable 桶列表(bucket list)应用为例,演示如何为源列表(即用户可以从中拖拽项目的列表)实现垂直滚动。

核心解决方案:CSS 样式实现滚动

实现 sortable 列表的滚动功能,主要依赖于 CSS 中的两个关键属性:max-height 和 overflow-y。

  1. max-height: 这个属性用于设置元素的最大高度。当元素内容的高度超过 max-height 所设定的值时,元素的高度将不再增加,而是保持在 max-height 的限制内。
  2. overflow-y: auto: 这个属性定义了当内容溢出元素的垂直边界时,浏览器应如何处理。将其设置为 auto 意味着当内容溢出时,浏览器将自动显示垂直滚动条;如果内容未溢出,则不显示滚动条。

通过将这两个属性应用于 rank_list 组件所生成的 HTML 容器,我们就能有效地控制其高度并在必要时启用滚动。

代码实现与解析

以下是基于原始示例修改后的 Shiny 应用代码,其中包含了实现滚动功能的关键 CSS 样式。

科威旅游管理系统
科威旅游管理系统

该软件是以php+MySQL进行开发的旅游管理网站系统。系统前端采用可视化布局,能自动适应不同尺寸屏幕,一起建站,不同设备使用,免去兼容性烦恼。系统提供列表、表格、地图三种列表显示方式,让用户以最快的速度找到所需行程,大幅提高效率。系统可设置推荐、优惠行程,可将相应行程高亮显示,对重点行程有效推广,可实现网站盈利。系统支持中文、英文,您还可以在后台添加新的语言,关键字单独列出,在后台即可快速翻译。

下载
library(shiny)
library(sortable)

ui <- fluidPage(
  tags$head(
    tags$style(HTML("
      /* 为所有桶列表容器设置最小高度 */
      .bucket-list-container {min-height: 350px;}

      /* 为第一个可拖拽源列表(ID为 rank_list_1)设置最大高度和垂直滚动 */
      #rank_list_1 {
        max-height: 300px; /* 设置最大高度,可根据需要调整 */
        overflow-y: auto;  /* 当内容溢出时显示垂直滚动条 */
        border: 1px solid #ddd; /* 可选:添加边框以便观察滚动效果 */
        padding: 5px; /* 可选:增加内边距 */
      }
      /* 确保内部的列表项不会因为滚动而挤压 */
      #rank_list_1 .sortable-item {
        margin-bottom: 5px; /* 增加列表项之间的间距 */
      }
    "))
  ),
  fluidRow(
    column(
      width = 12,
      # 选择变量列表的单选按钮
      radioButtons(inputId="variableList",
                   label="选择您的变量列表",
                   choices = c("names(mtcars)"="names(mtcars)","state.name"="state.name")),
      # 用于筛选变量名称的文本输入框
      textInput(
        inputId = "subsetChooseListText",
        label = "输入文本以筛选列表",
        value = "c"
      ),
      div(
        class = "bucket-list-container default-sortable",
        "将项目拖拽到任意桶中",
        div(
          class = "default-sortable bucket-list bucket-list-horizontal",
          # uiOutput 将渲染 rank_list_1,它是可滚动的源列表
          uiOutput("selection_list", style="flex:1 0 200px;"),

          # 目标列表 1
          rank_list(
            text = "拖拽到这里",
            labels = list(),
            input_id = "rank_list_2",
            options = sortable_options(group = "mygroup")
          ),
          # 目标列表 2
          rank_list(
            text = "也可以拖拽到这里",
            labels = list(),
            input_id = "rank_list_3",
            options = sortable_options(group = "mygroup")
          )
        )
      )
    )
  ),
  fluidRow(
    column(
      width = 12,
      tags$b("结果"),
      column(
        width = 12,
        tags$p("input$rank_list_1"),
        verbatimTextOutput("results_1"),
        tags$p("input$rank_list_2"),
        verbatimTextOutput("results_2"),
        tags$p("input$rank_list_3"),
        verbatimTextOutput("results_3")
      )
    )
  )
)

server <- function(input,output) {

  # 初始化响应式变量列表
  varList <- reactive({
    req(input$variableList)
    if (input$variableList == "state.name") {
      state.name
    } else {
      # 增加项目数量以确保滚动条出现
      paste0(rep(names(mtcars), 20),"_", 1:220) 
    }
  })

  # 根据文本输入筛选列表
  subsetChooseList <- reactive({
    items <- varList()
    pattern <- input$subsetChooseListText
    if (nchar(pattern) < 1) {
      return(items)
    }
    items[
      grepl(
        x = items,
        pattern = input$subsetChooseListText,
        ignore.case = TRUE
      )
    ]
  })

  # 渲染可拖拽的源列表
  output$selection_list <- renderUI({
    labels <- subsetChooseList()

    # 移除已被选择的项目
    labels <- labels[!(
      labels %in% input$rank_list_2 |
        labels %in% input$rank_list_3
    )]
    rank_list(
      text = "从这里拖拽",
      labels = labels,
      input_id = "rank_list_1", # 关键:此ID对应CSS样式
      options = sortable_options(group = "mygroup")
    )
  })

  # 用于调试的可视化输出
  output$results_1 <- renderPrint(input$rank_list_1)
  output$results_2 <- renderPrint(input$rank_list_2)
  output$results_3 <- renderPrint(input$rank_list_3)

}

shinyApp(ui, server)

代码解析:

  1. tags$head(tags$style(HTML(...))): 这是在 Shiny 应用中嵌入自定义 CSS 样式的标准方法。所有 CSS 规则都定义在这个 HTML() 块中。
  2. #rank_list_1 { ... }:
    • #rank_list_1 是 CSS 选择器,它精确地 targeting 了 input_id = "rank_list_1" 所生成的 div 元素。在 server 函数的 output$selection_list 中,我们通过 rank_list(input_id = "rank_list_1", ...) 创建了这个元素。
    • max-height: 300px;: 将 rank_list_1 容器的最大高度限制为 300 像素。你可以根据你的布局需求调整这个值。
    • overflow-y: auto;: 这是实现垂直滚动的核心。当 rank_list_1 中的内容高度超过 300 像素时,会自动出现垂直滚动条。
    • border: 1px solid #ddd; 和 padding: 5px;: 这些是可选的样式,用于在视觉上更清晰地界定可滚动区域,并提供更好的用户体验。
  3. #rank_list_1 .sortable-item { ... }: 这是一个可选的优化,用于调整 rank_list_1 内部每个拖拽项(.sortable-item)的样式,例如增加它们之间的垂直间距,使滚动列表看起来更整洁。
  4. server 函数中的 varList: 为了更好地演示滚动效果,我们将 names(mtcars) 的项目数量增加了 20 倍 (paste0(rep(names(mtcars), 20),"_", 1:220)),确保在默认情况下源列表有足够多的项目来触发滚动。

通过上述修改,当用户选择 "names(mtcars)" 列表且项目数量过多时,Drag from here 列表将显示一个垂直滚动条,允许用户浏览所有可拖拽的项目,而不会破坏整体页面布局。

注意事项与最佳实践

  1. 选择合适的 max-height 值: max-height 的选择应根据你的应用布局和目标设备的屏幕尺寸进行权衡。过小的高度可能导致滚动过于频繁,而过大则可能无法有效解决内容溢出问题。可以考虑使用相对单位(如 vh 视口高度百分比)来实现更好的响应式表现。
  2. 用户体验: 确保滚动条在视觉上清晰可见,且易于操作。适当的 padding 和 margin 可以改善滚动区域内部项目的视觉效果。
  3. 响应式设计: 在不同的屏幕尺寸下测试你的应用。你可能需要使用媒体查询(Media Queries)来为不同设备调整 max-height 值,以确保在移动设备和桌面设备上都有良好的体验。
  4. 与其他 sortable 选项的兼容性: 通常,添加 max-height 和 overflow-y 不会影响 sortable 包的其他功能,如拖拽、排序和分组。但如果遇到异常行为,请检查是否有其他 CSS 规则或 JavaScript 逻辑与之冲突。
  5. 性能考量: 对于包含成千上万个项目的极长列表,虽然滚动解决了显示问题,但渲染大量 DOM 元素仍可能影响性能。在这种极端情况下,可能需要考虑虚拟滚动(Virtual Scrolling)等更高级的优化技术。

总结

通过简单地在 Shiny 应用的 UI 部分嵌入 CSS 样式,并利用 max-height 和 overflow-y: auto 这两个属性,我们可以轻松地为 sortable 列表添加垂直滚动功能。这不仅解决了内容溢出导致的布局问题,也显著提升了用户在处理大量数据时的交互体验。掌握这一技巧,将使你的 Shiny 应用在功能性和美观性方面更上一层楼。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

394

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

754

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

454

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

554

2023.09.20

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

CSS教程
CSS教程

共754课时 | 21.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号