以编程方式触发 R 传单中的标记鼠标单击事件以实现闪亮

Posted

技术标签:

【中文标题】以编程方式触发 R 传单中的标记鼠标单击事件以实现闪亮【英文标题】:Programatically trigger marker mouse click event in R leaflet for shiny 【发布时间】:2022-01-14 05:30:52 【问题描述】:

我的问题与这个问题相同:Trigger marker mouse click event in R leaflet for shiny 但我没有足够的代表来添加评论,并且编辑队列已“满”,所以我无法将我的想法添加到原始问题中。不确定这是否违反社区规则/最佳实践,如果是,请删除!为下面冗长的描述道歉,但我想我可能接近一个 javascript 或闪亮的大师可以立即修复的解决方案!或者,我完全找错了树。感谢阅读!

当我在 R Shiny Web 应用程序的 DT 数据表中选择一行时,我想触发 Leaflet 地图标记点击事件。

这是一个最小的示例应用程序,作为添加此功能的基础:

library(shiny)
library(leaflet)
library(magrittr)
library(shinyjs)

# create js function that triggers a click on a button 'buttona'
jsCode <- 'shinyjs.buttonClick = (function() 
           $("#buttona").click();
           );'

df <- tibble::tibble(id = c(1,2,3,4,5),
                     label = c('One','Two','Three','Four','Five'),
                     lat = c(50,55,60,65,70), lng = c(0,5,-5,10,-10)
                     )

ui <- fluidPage(
    # new lines to enable shinyjs and import custom js function
    shinyjs::useShinyjs(),
    shinyjs::extendShinyjs(text = jsCode, functions = c('buttonClick')),

    leaflet::leafletOutput('map'),
    DT::DTOutput('table'),
    shiny::actionButton('buttona',"Button A") # new button
)

server <- function(input, output, session) 
    
    output$map <- leaflet::renderLeaflet(
        leaflet::leaflet(options = leaflet::leafletOptions(minZoom = 3,maxZoom = 10)) %>%
            leaflet::setView(lng = 10,lat = 60,zoom = 3) %>%
            leaflet::addProviderTiles(provider = leaflet::providers$Esri.OceanBasemap) %>%
            leaflet::addMarkers(data = df,
                                layerId = ~id,
                                group = 'group1',
                                label = ~label,
                                lat = ~lat,
                                lng = ~lng,
                                popup = ~paste("<h3>More Information</h3>",
                                               "<b>Title:</b>",label,sep =" "))
    )
    output$table <- DT::renderDT(df,
                                 selection = 'single',
                                 rownames = FALSE,
                                 editable = FALSE
    )

    # observer looking for datatable row selection and triggering js function
    observeEvent(input$table_rows_selected,
        shinyjs::js$buttonClick()
    )

    # observer looking for button click to trigger modal
    observeEvent(input$buttona,
        showModal(
            modalDialog(title = "Test",
                        size = 'm',
                        h1("Test")
                        
            )
        )
    )
    


# Run the application 
shinyApp(ui = ui, server = server)

我尝试过的事情:

shinyjs 和 javascript

我已经能够成功地使用 shinyjs 包通过按钮创建类似的功能(请参阅上面的示例应用程序),但是当我尝试对标记做同样的事情时,我只是没有找到 js 知识正确的元素。通过在 chrome 中浏览 js 控制台,我可以手动找到它们,但它们位于我不知道如何以编程方式定位的 iframe 中,而且该位置中有一个随机字符串,例如jQuery351022343796258432992。 通过 chrome js 控制台使用手动定位(在此工作之前,我需要使用“元素”选项卡在 iframe 中选择#document)我可以通过以下几行触发我想要的点击事件:

var mymap = document.getElementsByClassName('leaflet');
var els = mymap.map.jQuery351022343796258432992.leafletMap.layerManager._byGroup.group1;
els[0].fire('click'); //note this is the leaflet.js to trigger a marker click event

闪亮的小部件

根据本页底部https://rstudio.github.io/leaflet/morefeatures.html 的传单文档,使用shinywidgets::onRender 可能会有一些问题,但我不知道如何在这种情况下实现它。

再次感谢您的阅读!

【问题讨论】:

【参考方案1】:

使用JS的解决方案

访问 Map 对象后,您需要遍历所有图层以找到具有特定 id 的标记。

我修改了您使用 shinyjs 调用的 JS 函数以遍历所有层并在与 id 匹配的标记上触发事件 click。为避免每次都查找 Map 对象,使用 htmlwidgets::onRender 函数在渲染后检索 Map 对象。作为shinyjs 的替代方案,您可以使用runjs 来执行函数(不在下面的代码中)。

library(shiny)
library(leaflet)
library(magrittr)
library(shinyjs)

# create js function that triggers a click on a marker selected by a row in a DT
jsCode <- 'shinyjs.markerClick = function(id) 
              map.eachLayer(function (layer) 
                if (layer.options.layerId == id) 
                  layer.fire("click");
                
              )
           ;'

df <- tibble::tibble(id = c(1,2,3,4,5),
                     label = c('One','Two','Three','Four','Five'),
                     lat = c(50,55,60,65,70), lng = c(0,5,-5,10,-10)
)

ui <- fluidPage(
  # new lines to enable shinyjs and import custom js function
  shinyjs::useShinyjs(),
  shinyjs::extendShinyjs(text = jsCode, functions = c('markerClick')),
  
  leaflet::leafletOutput('map'),
  DT::DTOutput('table'),
  shiny::actionButton('buttona',"Button A") # new button
)

server <- function(input, output, session) 
  
  output$map <- leaflet::renderLeaflet(
    m <- leaflet::leaflet(options = leaflet::leafletOptions(minZoom = 3,maxZoom = 10)) %>%
      leaflet::setView(lng = 10,lat = 60,zoom = 3) %>%
      leaflet::addProviderTiles(provider = leaflet::providers$Esri.OceanBasemap) %>%
      leaflet::addMarkers(data = df,
                          layerId = ~id,
                          group = 'group1',
                          label = ~label,
                          lat = ~lat,
                          lng = ~lng,
                          popup = ~paste("<h3>More Information</h3>",
                                         "<b>Title:</b>",label,sep =" "))
    
    # assign the leaflet object to variable 'map'
    m <- m %>% 
      htmlwidgets::onRender("
          function(el, x) 
            map = this;
          "
      )                                         
    
  )
  output$table <- DT::renderDT(df,
                               selection = 'single',
                               rownames = FALSE,
                               editable = FALSE
  )
  
  # observer looking for datatable row selection and triggering js function
  observeEvent(input$table_rows_selected,
    rowIndex <- input$table_rows_selected
    df$id[rowIndex]
    shinyjs::js$markerClick(df$id[rowIndex])
  )
  
  # observer looking for button click to trigger modal
  observeEvent(input$buttona,
    showModal(
      modalDialog(title = "Test",
                  size = 'm',
                  h1("Test")
                  
      )
    ) 
  )
  


# Run the application 
shinyApp(ui = ui, server = server)

使用 Leaflet 代理的解决方案

每次用户选择表格中的一行时,只需添加一个新的弹出窗口。使用相同的layerId 自动更新可能已经在地图上的弹出窗口很重要。此外,由于弹出窗口将放置在标记latlng 上,因此需要使用offset 调整像素的相对位置。

library(shiny)
library(leaflet)

df <- tibble::tibble(id = c(1,2,3,4,5),
                     label = c('One','Two','Three','Four','Five'),
                     lat = c(50,55,60,65,70), lng = c(0,5,-5,10,-10)
)

ui <- fluidPage( 
  leaflet::leafletOutput('map'),
  DT::DTOutput('table')
)

server <- function(input, output, session) 
  
  output$map <- leaflet::renderLeaflet(
    m <- leaflet::leaflet(options = leaflet::leafletOptions(minZoom = 3,maxZoom = 10)) %>%
      leaflet::setView(lng = 10,lat = 60,zoom = 3) %>%
      leaflet::addProviderTiles(provider = leaflet::providers$Esri.OceanBasemap) %>%
      leaflet::addMarkers(data = df,
                          layerId = ~id,
                          group = 'group1',
                          label = ~label,
                          lat = ~lat,
                          lng = ~lng,
                          popup = ~paste("<h3>More Information</h3>",
                                         "<b>Title:</b>",label,sep =" "))
    
  )
  
  output$table <- DT::renderDT(df,
                               selection = 'single',
                               rownames = FALSE,
                               editable = FALSE
  )
  
  # observer looking for datatable row selection and use leaflet proxy to add a popup
  observeEvent(input$table_rows_selected,
    rowIndex <- input$table_rows_selected
    df$id[rowIndex]
    proxy <- leafletProxy("map")
    addPopups(
      proxy,
      lng = df$lng[rowIndex],
      lat =df$lat[rowIndex],
      popup = paste("<h3>More Information</h3>",
                    "<b>Title:</b>",df$label[rowIndex],sep =" "),
      layerId = "popup",
      options  = popupOptions(offset = list (x = 0, y = -26))
    )
  )


shinyApp(ui = ui, server = server)

【讨论】:

leaflet我不熟悉,但我们不能更简单地使用代理吗? 是的,你是对的,我只是尝试使用“点击事件...”来回答问题,我使用更简单的代理添加了解决方案。 感谢 Geovany,这是我需要的解决方案。 JS 版本对我来说更好,原因有两个:我只需要定义一次弹出内容,这在我的用例中更可取,并且代理版本创建了两个弹出窗口可以在地图上共存的情况。除此之外,这两种解决方案都很棒,所以还要感谢@Stéphane 的替代建议。只需注意一点,使用 JS 解决方案时,“鼠标悬停”标签会贴在标记上。我会弄清楚的,不过与添加的功能相比,这不是什么大问题! 我认为代理解决方案可能适合这个问题 Geovany ***.com/questions/56962857/… 如果您想将其粘贴到那里。 我很高兴它成功了。我添加了指向您指出的问题的链接。

以上是关于以编程方式触发 R 传单中的标记鼠标单击事件以实现闪亮的主要内容,如果未能解决你的问题,请参考以下文章

R小册子中的标记鼠标点击事件有光泽

对于以编程方式创建的 UIButton,有时不会触发按钮单击事件

如何通过在Vue中单击以编程方式触发可拖动的“移动”事件?

单击反应传单 v.3.x 中的标记时如何动态更改地图缩放?

如何在 Android 中以编程方式触发自定义信息窗口

在 JavaScript 中,我可以以编程方式为文件输入元素触发“点击”事件吗?