Agent-almanac build-shiny-module
install
source · Clone the upstream repo
git clone https://github.com/pjt222/agent-almanac
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/pjt222/agent-almanac "$T" && mkdir -p ~/.claude/skills && cp -r "$T/i18n/ja/skills/build-shiny-module" ~/.claude/skills/pjt222-agent-almanac-build-shiny-module-96d76a && rm -rf "$T"
manifest:
i18n/ja/skills/build-shiny-module/SKILL.mdsource content
Shinyモジュールの構築
適切な名前空間分離、リアクティブな通信、および構成可能性を持つShiny UI/サーバーモジュールペアを作成します。
使用タイミング
- 成長するShinyアプリから再利用可能コンポーネントを抽出するとき
- 複数の場所で使われるUIウィジェットを構築するとき
- 複雑なリアクティブロジックをクリーンなインターフェースの裏に隠すとき
- 大きなアプリケーションを小さくテスト可能な単位から構成するとき
入力
- 必須: モジュールの目的と機能の説明
- 必須: 入出力の仕様(モジュールが受け取るものと返すもの)
- オプション: モジュールが他のモジュールをネストするか(デフォルト:いいえ)
- オプション: フレームワークのコンテキスト(golem、rhino、またはvanilla)
手順
ステップ1: モジュールインターフェースの定義
コードを書く前に、モジュールが受け取るものと返すものを定義します:
モジュール: data_filter 入力: リアクティブデータセット、フィルタリングする列名 出力: リアクティブなフィルタリング済みデータセット UI: フィルターコントロール(selectInput、sliderInput、dateRangeInput)
期待結果: リアクティブな入力、リアクティブな出力、UIエレメントを指定した明確な仕様。
失敗時: インターフェースが不明確な場合、モジュールがおそらく広すぎます。単一の責任を持つより小さなモジュールに分割してください。
ステップ2: モジュールUI関数の作成
#' Data Filter Module UI #' #' @param id Module namespace ID #' @return A tagList of filter controls #' @export dataFilterUI <- function(id) { ns <- NS(id) tagList( selectInput( ns("column"), "Filter column", choices = NULL ), uiOutput(ns("filter_control")), actionButton(ns("apply"), "Apply Filter", class = "btn-primary") ) }
主要なルール:
- 関数名は
規則に従う<name>UI - 最初の引数は常に
id - 先頭に
を作成するns <- NS(id) - すべての
とinputId
をoutputId
でラップするns() - 柔軟な配置を可能にするために
を返すtagList()
期待結果: 名前空間化されたinput/outputエレメントを作成するUI関数。
失敗時: モジュールを2回使用するとIDが衝突する場合は、すべてのIDが
ns()でラップされているか確認してください。よくある見落とし:renderUI()またはuiOutput()内のID — これらもns()が必要です。
ステップ3: モジュールサーバー関数の作成
#' Data Filter Module Server #' #' @param id Module namespace ID #' @param data Reactive expression returning a data frame #' @param columns Character vector of filterable column names #' @return Reactive expression returning the filtered data frame #' @export dataFilterServer <- function(id, data, columns) { moduleServer(id, function(input, output, session) { ns <- session$ns # データが変わったときに列の選択肢を更新 observeEvent(data(), { available <- intersect(columns, names(data())) updateSelectInput(session, "column", choices = available) }) # 選択した列に基づくダイナミックフィルターコントロール output$filter_control <- renderUI({ req(input$column) col_data <- data()[[input$column]] if (is.numeric(col_data)) { sliderInput( ns("value_range"), "Range", min = min(col_data, na.rm = TRUE), max = max(col_data, na.rm = TRUE), value = range(col_data, na.rm = TRUE) ) } else { selectInput( ns("value_select"), "Values", choices = unique(col_data), multiple = TRUE, selected = unique(col_data) ) } }) # フィルタリングされたデータをリアクティブとして返す filtered <- eventReactive(input$apply, { req(input$column) col <- input$column df <- data() if (is.numeric(df[[col]])) { req(input$value_range) df[df[[col]] >= input$value_range[1] & df[[col]] <= input$value_range[2], ] } else { req(input$value_select) df[df[[col]] %in% input$value_select, ] } }, ignoreNULL = FALSE) return(filtered) }) }
主要なルール:
- 関数名は
規則に従う<name>Server - 最初の引数は常に
id - 追加の引数はリアクティブ式または静的な値
を使用するmoduleServer(id, function(input, output, session) { ... })- サーバー内で作成するダイナミックUIには
を使用するsession$ns - リアクティブな値を明示的に返す
期待結果: 入力を処理しリアクティブな出力を返すサーバー関数。
失敗時: リアクティブな値が更新されない場合は、ダイナミックUIからの入力が(外側の
nsではなく)session$nsを使っているか確認してください。モジュールがNULLを返す場合は、return()がmoduleServer()の内側の最後の式であることを確認してください。
ステップ4: 親アプリへのモジュールの接続
# app_ui.RまたはuiにてM ui <- page_sidebar( title = "Analysis App", sidebar = sidebar( dataFilterUI("filter1") ), card( DT::dataTableOutput("table") ) ) # app_server.RまたはserverにてMJ server <- function(input, output, session) { # 生データソース raw_data <- reactive({ mtcars }) # モジュールを呼び出す — 戻り値をキャプチャ filtered_data <- dataFilterServer( "filter1", data = raw_data, columns = c("cyl", "mpg", "hp", "wt") ) # モジュールが返したリアクティブを使用 output$table <- DT::renderDataTable({ filtered_data() }) }
期待結果: モジュールがUIに表示され、返されたリアクティブが下流の出力に流れます。
失敗時: モジュールUIがレンダリングされない場合は、UI呼び出しとサーバー呼び出しで
id文字列が一致しているか確認してください。返されたリアクティブがNULLの場合は、サーバー関数が実際に値を返しているか確認してください。
ステップ5: ネストしたモジュールの構成(オプション)
他のモジュールを含むモジュールの場合:
analysisUI <- function(id) { ns <- NS(id) tagList( dataFilterUI(ns("filter")), plotOutput(ns("plot")) ) } analysisServer <- function(id, data) { moduleServer(id, function(input, output, session) { # 名前空間化されたIDで内部モジュールを呼び出す filtered <- dataFilterServer("filter", data = data, columns = names(data())) output$plot <- renderPlot({ req(filtered()) plot(filtered()) }) return(filtered) }) }
主要なルール:UIでは
ns("inner_id")でネストする。サーバーでは"inner_id"だけで呼び出す — moduleServerが名前空間のチェーンを処理します。
期待結果: 内部モジュールが外部モジュールの名前空間内で正しくレンダリングされます。
失敗時: 内部モジュールのUIが表示されない場合は、外部UI関数で内部モジュールのIDの周りに
ns()を忘れている可能性があります。サーバーの通信が壊れている場合は、内部モジュールのIDが一致しているか確認してください(サーバー呼び出しにns()はない)。
ステップ6: モジュールを分離してテスト
# モジュールのクイックテストアプリ if (interactive()) { shiny::shinyApp( ui = fluidPage( dataFilterUI("test"), DT::dataTableOutput("result") ), server = function(input, output, session) { data <- reactive(iris) filtered <- dataFilterServer("test", data, names(iris)) output$result <- DT::renderDataTable(filtered()) } ) }
期待結果: モジュールが最小限のテストアプリで正しく動作します。
失敗時: モジュールが分離されたときに失敗するが、完全なアプリでは動作する場合(またはその逆)、グローバル変数や親セッションの状態への暗黙的な依存関係を確認してください。
バリデーション
- モジュールUI関数が最初の引数として
を受け取りid
を使用するNS(id) - UIのすべてのinput/output IDが
でラップされているns() - モジュールサーバーが
を使用するmoduleServer(id, function(input, output, session) { ... }) - サーバー内のダイナミックUIがIDに
を使用するsession$ns - モジュールをID衝突なしに複数回インスタンス化できる
- リアクティブな戻り値が親アプリからアクセス可能
- モジュールが最小限のスタンドアロンテストアプリで動作する
よくある落とし穴
内でrenderUI()
を忘れる: サーバー内で作成するダイナミックUIはns()
を使用する必要があります — 外側のsession$ns
はns
内では使用できません。moduleServer()- 非リアクティブなデータを渡す: 時間とともに変化するモジュール引数はリアクティブ式でなければなりません。
ではなくdata
を渡してください。reactive(data) - IDの不一致: UI呼び出しの
文字列はサーバー呼び出しのid
と完全に一致する必要があります。id - リアクティブを返さない: モジュールが親に必要なものを計算する場合は、リアクティブを
する必要があります。これを忘れるとサイレントなバグになります。return() - ネストしたモジュールの名前空間: UIでは:
。サーバーでは:ただのns("inner_id")
。これらを混同すると名前空間の二重ラッピングやプレフィックスの欠落が起きます。"inner_id"
関連スキル
— モジュールを追加する前のアプリ構造のセットアップscaffold-shiny-app
— testServer()ユニットテストによるモジュールのテストtest-shiny-app
— モジュールUIのbslibレイアウトとテーマdesign-shiny-ui
— モジュール内のキャッシュと非同期パターンoptimize-shiny-performance