From f443c205b4f15e6fc4d8ac80ee7f5cee1142b89c Mon Sep 17 00:00:00 2001 From: Gordon Shotwell Date: Mon, 17 Jul 2023 09:33:53 -0300 Subject: [PATCH 01/14] Add todo list example (#603) Co-authored-by: Joe Cheng --- shiny/examples/todo_list/app.py | 113 ++++++++++++++++++++++ shiny/examples/todo_list/requirements.txt | 1 + 2 files changed, 114 insertions(+) create mode 100644 shiny/examples/todo_list/app.py create mode 100644 shiny/examples/todo_list/requirements.txt diff --git a/shiny/examples/todo_list/app.py b/shiny/examples/todo_list/app.py new file mode 100644 index 000000000..4640a06fa --- /dev/null +++ b/shiny/examples/todo_list/app.py @@ -0,0 +1,113 @@ +import shinyswatch +from htmltools import css + +from shiny import App, module, reactive, render, ui + +app_ui = ui.page_fixed( + {"class": "my-5"}, + shinyswatch.theme.minty(), + ui.panel_title("Shiny TodoMVC"), + ui.layout_sidebar( + ui.panel_sidebar( + ui.input_text("todo_input_text", "", placeholder="Todo text"), + ui.input_action_button("add", "Add to-do"), + ), + ui.panel_main( + ui.output_text("cleared_tasks"), + ui.div(id="tasks", style="margin-top: 0.5em"), + ), + ), +) + + +def server(input, output, session): + finished_tasks = reactive.Value(0) + task_counter = reactive.Value(0) + + @output + @render.text + def cleared_tasks(): + return f"Finished tasks: {finished_tasks()}" + + @reactive.Effect + @reactive.event(input.add) + def add(): + counter = task_counter.get() + 1 + task_counter.set(counter) + id = "task_" + str(counter) + ui.insert_ui( + selector="#tasks", + where="beforeEnd", + ui=task_ui(id), + ) + + finish = task_server(id, text=input.todo_input_text()) + + # Defining a nested reactive effect like this might feel a bit funny but it's the + # correct pattern in this case. We are reacting to the `finish` + # event within the `add` closure, so nesting the reactive effects + # means that we don't have to worry about conflicting with + # finish events from other task elements. + @reactive.Effect + @reactive.event(finish) + def iterate_counter(): + finished_tasks.set(finished_tasks.get() + 1) + + ui.update_text("todo_input_text", value="") + + +# Modules to define the rows + + +@module.ui +def task_ui(): + return ui.output_ui("button_row") + + +@module.server +def task_server(input, output, session, text): + finished = reactive.Value(False) + + @output + @render.ui + def button_row(): + button = None + if finished(): + button = ui.input_action_button("clear", "Clear", class_="btn-warning") + else: + button = ui.input_action_button("finish", "Finish", class_="btn-default") + + return ui.row( + ui.column(4, button), + ui.column(8, text), + class_="mt-3 p-3 border align-items-center", + style=css(text_decoration="line-through" if finished() else None), + ) + + @reactive.Effect + @reactive.event(input.finish) + def finish_task(): + finished.set(True) + + @reactive.Effect + @reactive.event(input.clear) + def clear_task(): + ui.remove_ui(selector=f"div#{session.ns('button_row')}") + + # Since remove_ui only removes the HTML the reactive effects will be held + # in memory unless they're explicitly destroyed. This isn't a big + # deal because they're very small, but it's good to clean them up. + finish_task.destroy() + clear_task.destroy() + + # Returning the input.finish button to the parent scope allows us + # to react to it in the parent context to keep track of the number of + # completed tasks. + # + # This is a good pattern because it makes the module more general. + # The same module can be used by different applications which may + # do different things when the task is completed. + return input.finish + + +app = App(app_ui, server) diff --git a/shiny/examples/todo_list/requirements.txt b/shiny/examples/todo_list/requirements.txt new file mode 100644 index 000000000..69650978b --- /dev/null +++ b/shiny/examples/todo_list/requirements.txt @@ -0,0 +1 @@ +shinyswatch From f2e0535bd21af0770b4c519fa91d8dce791ae60a Mon Sep 17 00:00:00 2001 From: Gordon Shotwell Date: Mon, 17 Jul 2023 14:31:06 -0300 Subject: [PATCH 02/14] Annotation export example (#584) --- examples/annotation-export/app.py | 120 +++++++ examples/annotation-export/boulder_temp.csv | 366 ++++++++++++++++++++ examples/annotation-export/requirements.txt | 2 + 3 files changed, 488 insertions(+) create mode 100644 examples/annotation-export/app.py create mode 100644 examples/annotation-export/boulder_temp.csv create mode 100644 examples/annotation-export/requirements.txt diff --git a/examples/annotation-export/app.py b/examples/annotation-export/app.py new file mode 100644 index 000000000..6f06f4d89 --- /dev/null +++ b/examples/annotation-export/app.py @@ -0,0 +1,120 @@ +from pathlib import Path + +import matplotlib.dates as mdates +import matplotlib.pyplot as plt +import pandas as pd +import seaborn as sns + +from shiny import App, Inputs, Outputs, Session, reactive, render, ui +from shiny.plotutils import brushed_points + +path = Path(__file__).parent / "boulder_temp.csv" +weather_df = pd.read_csv(path) +weather_df["date"] = pd.to_datetime(weather_df["date"]) +weather_df["annotation"] = "" + +app_ui = ui.page_fluid( + ui.panel_title("Plot annotation example"), + ui.p( + """ + Select points to annotate them. + The plot is rendered with seaborn and all interaction is handled by Shiny. + """, + {"style": "font-size: larger"}, + ), + ui.row( + ui.column( + 6, + ui.output_plot("time_series", brush=ui.brush_opts(direction="x")), + ui.output_ui("annotator"), + ), + ui.column( + 4, + ui.h3("Annotated points"), + ui.output_data_frame("annotations"), + ), + ui.column(2, ui.download_button("download", "Download CSV")), + ), +) + + +def server(input: Inputs, output: Outputs, session: Session): + annotated_data = reactive.Value(weather_df) + + @reactive.Calc + def selected_data(): + out = brushed_points(annotated_data(), input.time_series_brush(), xvar="date") + return out + + @reactive.Effect + @reactive.event(input.annotate_button) + def _(): + selected = selected_data() + selected["annotation_new"] = input.annotation() + selected = selected.loc[:, ["date", "annotation_new"]] + + df = annotated_data().copy() + + df = df.merge(selected, on="date", how="left") + df["annotation_new"] = df["annotation_new"].fillna("") + updated_rows = df["annotation_new"] != "" + df.loc[updated_rows, "annotation"] = df.loc[updated_rows, "annotation_new"] + df = df.loc[:, ["date", "temp_c", "annotation"]] + annotated_data.set(df) + + @output + @render.plot + def time_series(): + fig, ax = plt.subplots() + ax.xaxis.set_major_locator(mdates.MonthLocator()) + ax.xaxis.set_major_formatter(mdates.DateFormatter("%Y-%m")) + ax.set_title("Temperature readings, Boulder Colorado") + out = sns.scatterplot( + data=annotated_data(), x="date", y="temp_c", hue="annotation", ax=ax + ) + + out.tick_params(axis="x", rotation=30) + return out.get_figure() + + @output + @render.ui + def annotator(): + if input.time_series_brush() is not None: + selected = selected_data() + + min = str(selected["date"].min()) + max = str(selected["date"].max()) + + min = min.replace(" 00:00:00+00:00", "") + max = max.replace(" 00:00:00+00:00", "") + + out = ui.TagList( + ui.row( + {"style": "padding-top: 20px;"}, + ui.column( + 4, + ui.p(f"{min} to", ui.br(), f"{max}"), + ), + ui.column( + 4, + ui.input_text("annotation", "", placeholder="Enter annotation"), + ), + ui.column(4, ui.input_action_button("annotate_button", "Submit")), + ) + ) + return out + + @output + @render.data_frame + def annotations(): + df = annotated_data().copy() + df["date"] = df["date"].dt.strftime("%Y-%m-%d") + df = df.loc[df["annotation"] != ""] + return df + + @session.download(filename="data.csv") + def download(): + yield annotated_data().to_csv() + + +app = App(app_ui, server) diff --git a/examples/annotation-export/boulder_temp.csv b/examples/annotation-export/boulder_temp.csv new file mode 100644 index 000000000..b09620afb --- /dev/null +++ b/examples/annotation-export/boulder_temp.csv @@ -0,0 +1,366 @@ +date,temp_c +2021-01-01T00:00:00Z,6.1 +2021-01-02T00:00:00Z,11.1 +2021-01-03T00:00:00Z,11.7 +2021-01-04T00:00:00Z,15 +2021-01-05T00:00:00Z,12.8 +2021-01-06T00:00:00Z,7.2 +2021-01-07T00:00:00Z,7.8 +2021-01-08T00:00:00Z,7.2 +2021-01-09T00:00:00Z,2.2 +2021-01-10T00:00:00Z,1.7 +2021-01-11T00:00:00Z,6.7 +2021-01-12T00:00:00Z,12.2 +2021-01-13T00:00:00Z,17.8 +2021-01-14T00:00:00Z,13.3 +2021-01-15T00:00:00Z,7.8 +2021-01-16T00:00:00Z,11.1 +2021-01-17T00:00:00Z,11.1 +2021-01-18T00:00:00Z,7.8 +2021-01-19T00:00:00Z,3.9 +2021-01-20T00:00:00Z,15 +2021-01-21T00:00:00Z,8.3 +2021-01-22T00:00:00Z,7.2 +2021-01-23T00:00:00Z,6.7 +2021-01-24T00:00:00Z,3.3 +2021-01-25T00:00:00Z,3.3 +2021-01-26T00:00:00Z,-2.2 +2021-01-27T00:00:00Z,-2.2 +2021-01-28T00:00:00Z,7.2 +2021-01-29T00:00:00Z,14.4 +2021-01-30T00:00:00Z,12.2 +2021-01-31T00:00:00Z,11.1 +2021-02-01T00:00:00Z,10 +2021-02-02T00:00:00Z,18.3 +2021-02-03T00:00:00Z,18.3 +2021-02-04T00:00:00Z,7.8 +2021-02-05T00:00:00Z,7.2 +2021-02-06T00:00:00Z,7.2 +2021-02-07T00:00:00Z,12.2 +2021-02-08T00:00:00Z,8.9 +2021-02-09T00:00:00Z,-0.6 +2021-02-10T00:00:00Z,-0.6 +2021-02-11T00:00:00Z,-2.8 +2021-02-12T00:00:00Z,-9.4 +2021-02-13T00:00:00Z,-13.9 +2021-02-14T00:00:00Z,-15.6 +2021-02-15T00:00:00Z,-6.1 +2021-02-16T00:00:00Z,5.6 +2021-02-17T00:00:00Z,1.7 +2021-02-18T00:00:00Z,1.1 +2021-02-19T00:00:00Z,7.2 +2021-02-20T00:00:00Z,10 +2021-02-21T00:00:00Z,6.1 +2021-02-22T00:00:00Z,13.3 +2021-02-23T00:00:00Z,15.6 +2021-02-24T00:00:00Z,12.2 +2021-02-25T00:00:00Z,1.7 +2021-02-26T00:00:00Z,7.2 +2021-02-27T00:00:00Z,5 +2021-02-28T00:00:00Z,3.9 +2021-03-01T00:00:00Z,9.4 +2021-03-02T00:00:00Z,16.7 +2021-03-03T00:00:00Z,16.7 +2021-03-04T00:00:00Z,8.9 +2021-03-05T00:00:00Z,15.6 +2021-03-06T00:00:00Z,19.4 +2021-03-07T00:00:00Z,20.6 +2021-03-08T00:00:00Z,20 +2021-03-09T00:00:00Z,20.6 +2021-03-10T00:00:00Z,15.6 +2021-03-11T00:00:00Z,6.7 +2021-03-12T00:00:00Z,5.6 +2021-03-13T00:00:00Z,4.4 +2021-03-14T00:00:00Z,0 +2021-03-15T00:00:00Z,4.4 +2021-03-16T00:00:00Z,4.4 +2021-03-17T00:00:00Z,7.8 +2021-03-18T00:00:00Z,7.8 +2021-03-19T00:00:00Z,14.4 +2021-03-20T00:00:00Z,17.8 +2021-03-21T00:00:00Z,11.1 +2021-03-22T00:00:00Z,4.4 +2021-03-23T00:00:00Z,5 +2021-03-24T00:00:00Z,4.4 +2021-03-25T00:00:00Z,10 +2021-03-26T00:00:00Z,9.4 +2021-03-27T00:00:00Z,12.2 +2021-03-28T00:00:00Z,17.2 +2021-03-29T00:00:00Z,21.7 +2021-03-30T00:00:00Z,16.7 +2021-03-31T00:00:00Z,12.2 +2021-04-01T00:00:00Z,21.7 +2021-04-02T00:00:00Z,23.3 +2021-04-03T00:00:00Z,25.6 +2021-04-04T00:00:00Z,26.7 +2021-04-05T00:00:00Z,25 +2021-04-06T00:00:00Z,24.4 +2021-04-07T00:00:00Z,17.2 +2021-04-08T00:00:00Z,21.1 +2021-04-09T00:00:00Z,19.4 +2021-04-10T00:00:00Z,21.1 +2021-04-11T00:00:00Z,18.3 +2021-04-12T00:00:00Z,11.7 +2021-04-13T00:00:00Z,3.9 +2021-04-14T00:00:00Z,3.9 +2021-04-15T00:00:00Z,3.9 +2021-04-16T00:00:00Z,3.3 +2021-04-17T00:00:00Z,6.7 +2021-04-18T00:00:00Z,14.4 +2021-04-19T00:00:00Z,12.2 +2021-04-20T00:00:00Z,2.8 +2021-04-21T00:00:00Z,1.7 +2021-04-22T00:00:00Z,10 +2021-04-23T00:00:00Z,13.9 +2021-04-24T00:00:00Z,17.8 +2021-04-25T00:00:00Z,26.1 +2021-04-26T00:00:00Z,24.4 +2021-04-27T00:00:00Z,20.6 +2021-04-28T00:00:00Z,15 +2021-04-29T00:00:00Z,20.6 +2021-04-30T00:00:00Z,26.1 +2021-05-01T00:00:00Z,28.9 +2021-05-02T00:00:00Z,26.1 +2021-05-03T00:00:00Z,12.2 +2021-05-04T00:00:00Z,15.6 +2021-05-05T00:00:00Z,15.6 +2021-05-06T00:00:00Z,20.6 +2021-05-07T00:00:00Z,27.8 +2021-05-08T00:00:00Z,24.4 +2021-05-09T00:00:00Z,16.1 +2021-05-10T00:00:00Z,7.2 +2021-05-11T00:00:00Z,5.6 +2021-05-12T00:00:00Z,16.7 +2021-05-13T00:00:00Z,23.9 +2021-05-14T00:00:00Z,22.2 +2021-05-15T00:00:00Z,19.4 +2021-05-16T00:00:00Z,15 +2021-05-17T00:00:00Z,17.2 +2021-05-18T00:00:00Z,17.8 +2021-05-19T00:00:00Z,21.7 +2021-05-20T00:00:00Z,25.6 +2021-05-21T00:00:00Z,24.4 +2021-05-22T00:00:00Z,22.8 +2021-05-23T00:00:00Z,23.9 +2021-05-24T00:00:00Z,22.8 +2021-05-25T00:00:00Z,24.4 +2021-05-26T00:00:00Z,25.6 +2021-05-27T00:00:00Z,25 +2021-05-28T00:00:00Z,26.1 +2021-05-29T00:00:00Z,21.7 +2021-05-30T00:00:00Z,17.8 +2021-05-31T00:00:00Z,13.9 +2021-06-01T00:00:00Z,20 +2021-06-02T00:00:00Z,23.3 +2021-06-03T00:00:00Z,27.8 +2021-06-04T00:00:00Z,30.6 +2021-06-05T00:00:00Z,32.8 +2021-06-06T00:00:00Z,31.1 +2021-06-07T00:00:00Z,29.4 +2021-06-08T00:00:00Z,31.1 +2021-06-09T00:00:00Z,31.7 +2021-06-10T00:00:00Z,32.8 +2021-06-11T00:00:00Z,32.2 +2021-06-12T00:00:00Z,30 +2021-06-13T00:00:00Z,33.3 +2021-06-14T00:00:00Z,34.4 +2021-06-15T00:00:00Z,35.6 +2021-06-16T00:00:00Z,37.2 +2021-06-17T00:00:00Z,37.2 +2021-06-18T00:00:00Z,32.8 +2021-06-19T00:00:00Z,31.7 +2021-06-20T00:00:00Z,27.2 +2021-06-21T00:00:00Z,23.9 +2021-06-22T00:00:00Z,33.9 +2021-06-23T00:00:00Z,35.6 +2021-06-24T00:00:00Z,29.4 +2021-06-25T00:00:00Z,25 +2021-06-26T00:00:00Z,20.6 +2021-06-27T00:00:00Z,22.2 +2021-06-28T00:00:00Z,22.2 +2021-06-29T00:00:00Z,24.4 +2021-06-30T00:00:00Z,27.2 +2021-07-01T00:00:00Z,25 +2021-07-02T00:00:00Z,28.3 +2021-07-03T00:00:00Z,30.6 +2021-07-04T00:00:00Z,29.4 +2021-07-05T00:00:00Z,30 +2021-07-06T00:00:00Z,28.3 +2021-07-07T00:00:00Z,30.6 +2021-07-08T00:00:00Z,36.7 +2021-07-09T00:00:00Z,34.4 +2021-07-10T00:00:00Z,32.8 +2021-07-11T00:00:00Z,28.9 +2021-07-12T00:00:00Z,31.1 +2021-07-13T00:00:00Z,30.6 +2021-07-14T00:00:00Z,25.6 +2021-07-15T00:00:00Z,27.8 +2021-07-16T00:00:00Z,32.2 +2021-07-17T00:00:00Z,32.8 +2021-07-18T00:00:00Z,32.8 +2021-07-19T00:00:00Z,33.3 +2021-07-20T00:00:00Z,33.9 +2021-07-21T00:00:00Z,31.7 +2021-07-22T00:00:00Z,34.4 +2021-07-23T00:00:00Z,33.9 +2021-07-24T00:00:00Z,30 +2021-07-25T00:00:00Z,32.2 +2021-07-26T00:00:00Z,32.8 +2021-07-27T00:00:00Z,33.9 +2021-07-28T00:00:00Z,35 +2021-07-29T00:00:00Z,31.7 +2021-07-30T00:00:00Z,32.8 +2021-07-31T00:00:00Z,24.4 +2021-08-01T00:00:00Z,26.7 +2021-08-02T00:00:00Z,29.4 +2021-08-03T00:00:00Z,27.8 +2021-08-04T00:00:00Z,27.2 +2021-08-05T00:00:00Z,32.8 +2021-08-06T00:00:00Z,32.2 +2021-08-07T00:00:00Z,27.2 +2021-08-08T00:00:00Z,32.8 +2021-08-09T00:00:00Z,34.4 +2021-08-10T00:00:00Z,34.4 +2021-08-11T00:00:00Z,33.9 +2021-08-12T00:00:00Z,30 +2021-08-13T00:00:00Z,28.9 +2021-08-14T00:00:00Z,33.9 +2021-08-15T00:00:00Z,31.7 +2021-08-16T00:00:00Z,32.8 +2021-08-17T00:00:00Z,35 +2021-08-18T00:00:00Z,33.3 +2021-08-19T00:00:00Z,27.8 +2021-08-20T00:00:00Z,26.1 +2021-08-21T00:00:00Z,27.8 +2021-08-22T00:00:00Z,32.2 +2021-08-23T00:00:00Z,32.2 +2021-08-24T00:00:00Z,33.9 +2021-08-25T00:00:00Z,31.1 +2021-08-26T00:00:00Z,28.3 +2021-08-27T00:00:00Z,32.8 +2021-08-28T00:00:00Z,31.7 +2021-08-29T00:00:00Z,28.3 +2021-08-30T00:00:00Z,32.8 +2021-08-31T00:00:00Z,34.4 +2021-09-01T00:00:00Z,30.6 +2021-09-02T00:00:00Z,27.8 +2021-09-03T00:00:00Z,27.2 +2021-09-04T00:00:00Z,28.9 +2021-09-05T00:00:00Z,30.6 +2021-09-06T00:00:00Z,35 +2021-09-07T00:00:00Z,30 +2021-09-08T00:00:00Z,30.6 +2021-09-09T00:00:00Z,35 +2021-09-10T00:00:00Z,37.2 +2021-09-11T00:00:00Z,35 +2021-09-12T00:00:00Z,29.4 +2021-09-13T00:00:00Z,28.3 +2021-09-14T00:00:00Z,25.6 +2021-09-15T00:00:00Z,32.2 +2021-09-16T00:00:00Z,32.8 +2021-09-17T00:00:00Z,23.3 +2021-09-18T00:00:00Z,32.8 +2021-09-19T00:00:00Z,31.1 +2021-09-20T00:00:00Z,26.1 +2021-09-21T00:00:00Z,18.9 +2021-09-22T00:00:00Z,26.1 +2021-09-23T00:00:00Z,28.3 +2021-09-24T00:00:00Z,22.8 +2021-09-25T00:00:00Z,30.6 +2021-09-26T00:00:00Z,32.2 +2021-09-27T00:00:00Z,31.1 +2021-09-28T00:00:00Z,27.8 +2021-09-29T00:00:00Z,21.7 +2021-09-30T00:00:00Z,12.2 +2021-10-01T00:00:00Z,19.4 +2021-10-02T00:00:00Z,22.2 +2021-10-03T00:00:00Z,26.1 +2021-10-04T00:00:00Z,27.8 +2021-10-05T00:00:00Z,28.3 +2021-10-06T00:00:00Z,25 +2021-10-07T00:00:00Z,26.1 +2021-10-08T00:00:00Z,24.4 +2021-10-09T00:00:00Z,20 +2021-10-10T00:00:00Z,21.1 +2021-10-11T00:00:00Z,21.1 +2021-10-12T00:00:00Z,12.2 +2021-10-13T00:00:00Z,16.1 +2021-10-14T00:00:00Z,10 +2021-10-15T00:00:00Z,13.3 +2021-10-16T00:00:00Z,21.7 +2021-10-17T00:00:00Z,23.9 +2021-10-18T00:00:00Z,24.4 +2021-10-19T00:00:00Z,18.3 +2021-10-20T00:00:00Z,18.3 +2021-10-21T00:00:00Z,17.2 +2021-10-22T00:00:00Z,23.9 +2021-10-23T00:00:00Z,22.8 +2021-10-24T00:00:00Z,21.1 +2021-10-25T00:00:00Z,22.8 +2021-10-26T00:00:00Z,21.7 +2021-10-27T00:00:00Z,15 +2021-10-28T00:00:00Z,15 +2021-10-29T00:00:00Z,24.4 +2021-10-30T00:00:00Z,24.4 +2021-10-31T00:00:00Z,12.8 +2021-11-01T00:00:00Z,3.9 +2021-11-02T00:00:00Z,6.7 +2021-11-03T00:00:00Z,16.1 +2021-11-04T00:00:00Z,21.1 +2021-11-05T00:00:00Z,23.9 +2021-11-06T00:00:00Z,23.9 +2021-11-07T00:00:00Z,25.6 +2021-11-08T00:00:00Z,12.8 +2021-11-09T00:00:00Z,12.8 +2021-11-10T00:00:00Z,13.9 +2021-11-11T00:00:00Z,12.8 +2021-11-12T00:00:00Z,13.3 +2021-11-13T00:00:00Z,19.4 +2021-11-14T00:00:00Z,18.9 +2021-11-15T00:00:00Z,22.8 +2021-11-16T00:00:00Z,22.2 +2021-11-17T00:00:00Z,15 +2021-11-18T00:00:00Z,10.6 +2021-11-19T00:00:00Z,18.9 +2021-11-20T00:00:00Z,14.4 +2021-11-21T00:00:00Z,14.4 +2021-11-22T00:00:00Z,21.1 +2021-11-23T00:00:00Z,21.1 +2021-11-24T00:00:00Z,13.9 +2021-11-25T00:00:00Z,16.1 +2021-11-26T00:00:00Z,22.2 +2021-11-27T00:00:00Z,14.4 +2021-11-28T00:00:00Z,19.4 +2021-11-29T00:00:00Z,23.3 +2021-11-30T00:00:00Z,15 +2021-12-01T00:00:00Z,22.2 +2021-12-02T00:00:00Z,22.2 +2021-12-03T00:00:00Z,16.1 +2021-12-04T00:00:00Z,20.6 +2021-12-05T00:00:00Z,17.8 +2021-12-06T00:00:00Z,-0.6 +2021-12-07T00:00:00Z,13.3 +2021-12-08T00:00:00Z,13.3 +2021-12-09T00:00:00Z,10.6 +2021-12-10T00:00:00Z,5.6 +2021-12-11T00:00:00Z,9.4 +2021-12-12T00:00:00Z,18.3 +2021-12-13T00:00:00Z,13.3 +2021-12-14T00:00:00Z,16.7 +2021-12-15T00:00:00Z,13.3 +2021-12-16T00:00:00Z,7.8 +2021-12-17T00:00:00Z,7.2 +2021-12-18T00:00:00Z,3.9 +2021-12-19T00:00:00Z,16.7 +2021-12-20T00:00:00Z,16.7 +2021-12-21T00:00:00Z,17.2 +2021-12-22T00:00:00Z,17.2 +2021-12-23T00:00:00Z,17.2 +2021-12-24T00:00:00Z,12.8 +2021-12-25T00:00:00Z,12.2 +2021-12-26T00:00:00Z,11.1 +2021-12-27T00:00:00Z,6.7 +2021-12-28T00:00:00Z,2.8 +2021-12-29T00:00:00Z,0.6 +2021-12-30T00:00:00Z,6.7 +2021-12-31T00:00:00Z,6.7 diff --git a/examples/annotation-export/requirements.txt b/examples/annotation-export/requirements.txt new file mode 100644 index 000000000..4695c7070 --- /dev/null +++ b/examples/annotation-export/requirements.txt @@ -0,0 +1,2 @@ +seaborn +pandas From ccca241a23315c081c8bab5f3c534b736ac9da86 Mon Sep 17 00:00:00 2001 From: Winston Chang Date: Mon, 17 Jul 2023 15:20:17 -0500 Subject: [PATCH 03/14] Fix example code blocks (#626) --- shiny/_app.py | 15 +++++++------ shiny/_main.py | 22 +++++++++---------- shiny/input_handler.py | 25 +++++++++++----------- shiny/ui/_include_helpers.py | 41 +++++++++++++++++++----------------- 4 files changed, 54 insertions(+), 49 deletions(-) diff --git a/shiny/_app.py b/shiny/_app.py index 0a45d4592..4715a2a93 100644 --- a/shiny/_app.py +++ b/shiny/_app.py @@ -53,16 +53,17 @@ class App: Example ------- - .. code-block:: python + ```{python} + #| eval: false + from shiny import App, Inputs, Outputs, Session, ui - from shiny import App, Inputs, Outputs, Session, ui + app_ui = ui.page_fluid("Hello Shiny!") - app_ui = ui.page_fluid("Hello Shiny!") - - def server(input: Inputs, output: Outputs, session: Session): - pass + def server(input: Inputs, output: Outputs, session: Session): + pass - app = App(app_ui, server) + app = App(app_ui, server) + ``` """ lib_prefix: str = "lib/" diff --git a/shiny/_main.py b/shiny/_main.py index 66aa0fa34..1b14dcc62 100644 --- a/shiny/_main.py +++ b/shiny/_main.py @@ -207,21 +207,21 @@ def run_app( Examples -------- - .. code-block:: python + ```{python} + from shiny import run_app - from shiny import run_app + # Run ``app`` inside ``./app.py`` + run_app() - # Run ``app`` inside ``./app.py`` - run_app() + # Run ``app`` inside ``./myapp.py`` (or ``./myapp/app.py``) + run_app("myapp") - # Run ``app`` inside ``./myapp.py`` (or ``./myapp/app.py``) - run_app("myapp") + # Run ``my_app`` inside ``./myapp.py`` (or ``./myapp/app.py``) + run_app("myapp:my_app") - # Run ``my_app`` inside ``./myapp.py`` (or ``./myapp/app.py``) - run_app("myapp:my_app") - - # Run ``my_app`` inside ``../myapp.py`` (or ``../myapp/app.py``) - run_app("myapp:my_app", app_dir="..") + # Run ``my_app`` inside ``../myapp.py`` (or ``../myapp/app.py``) + run_app("myapp:my_app", app_dir="..") + ``` """ # If port is 0, randomize diff --git a/shiny/input_handler.py b/shiny/input_handler.py index 8639b23a2..92d6b89af 100644 --- a/shiny/input_handler.py +++ b/shiny/input_handler.py @@ -10,7 +10,6 @@ if TYPE_CHECKING: from .session import Session - from .types import ActionButtonValue InputHandlerType = Callable[[Any, str, "Session"], Any] @@ -74,21 +73,23 @@ def _process_value(self, type: str, value: Any, name: str, session: Session) -> Example ------- -.. code-block:: python - - from shiny.input_handler import input_handlers - @input_handlers.add("mypackage.intify") - def _(value, name, session): - return int(value) +```{python} +#| eval: false +from shiny.input_handler import input_handlers +@input_handlers.add("mypackage.intify") +def _(value, name, session): + return int(value) +``` On the Javascript side, the associated input binding must have a corresponding ``getType`` method: -.. code-block:: javascript - - getType: function(el) { - return "mypackage.intify"; - } +```{python} +#| eval: false +getType: function(el) { + return "mypackage.intify"; +} +``` """ diff --git a/shiny/ui/_include_helpers.py b/shiny/ui/_include_helpers.py index 3a100456b..1739bfc40 100644 --- a/shiny/ui/_include_helpers.py +++ b/shiny/ui/_include_helpers.py @@ -63,16 +63,17 @@ def include_js( document, you can wrap it in ``head_content`` (in this case, just make sure you're aware that the DOM probably won't be ready when the script is executed). - .. code-block:: python - - ui.fluidPage( - head_content(ui.include_js("custom.js")), - ) + ```{python} + #| eval: false + ui.page_fluid( + ui.head_content(ui.include_js("custom.js")), + ) - # Alternately you can inline Javscript by changing the method. - ui.fluidPage( - head_content(ui.include_js("custom.js", method = "inline")), - ) + # Alternately you can inline Javscript by changing the method. + ui.page_fluid( + ui.head_content(ui.include_js("custom.js", method = "inline")), + ) + ``` See Also -------- @@ -133,19 +134,21 @@ def include_css( may result in a Flash of Unstyled Content (FOUC). To instead place the CSS in the :func:`~ui.tags.head` of the document, you can wrap it in ``head_content``: - .. code-block:: python - - from htmltools import head_content from shiny import ui + ```{python} + #| eval: false + from htmltools import head_content + from shiny import ui - ui.fluidPage( - head_content(ui.include_css("custom.css")), + ui.page_fluid( + ui.head_content(ui.include_css("custom.css")), - # You can also inline css by passing a dictionary with a `style` element. - ui.div( - {"style": "font-weight: bold;"}, - ui.p("Some text!"), - ) + # You can also inline css by passing a dictionary with a `style` element. + ui.div( + {"style": "font-weight: bold;"}, + ui.p("Some text!"), ) + ) + ``` See Also -------- From 50fd84579bd46b40e2f92c9fbd0e2f2a42610c2b Mon Sep 17 00:00:00 2001 From: Winston Chang Date: Mon, 17 Jul 2023 15:33:10 -0500 Subject: [PATCH 04/14] Don't eval example code block --- shiny/_main.py | 1 + 1 file changed, 1 insertion(+) diff --git a/shiny/_main.py b/shiny/_main.py index 1b14dcc62..ba71a5948 100644 --- a/shiny/_main.py +++ b/shiny/_main.py @@ -208,6 +208,7 @@ def run_app( -------- ```{python} + #|eval: false from shiny import run_app # Run ``app`` inside ``./app.py`` From ef75c0e595988c81349a9e4a9eea364006b4141f Mon Sep 17 00:00:00 2001 From: Joe Cheng Date: Tue, 18 Jul 2023 00:53:28 -0700 Subject: [PATCH 05/14] Make --app-path work with app file argument (#598) --- shiny/_main.py | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/shiny/_main.py b/shiny/_main.py index ba71a5948..21853285f 100644 --- a/shiny/_main.py +++ b/shiny/_main.py @@ -361,18 +361,20 @@ def resolve_app(app: str, app_dir: Optional[str]) -> tuple[str, Optional[str]]: attr = "app" if is_file(module): + # Before checking module path, resolve it relative to app_dir if provided + module_path = module if app_dir is None else os.path.join(app_dir, module) # TODO: We should probably be using some kind of loader # TODO: I don't like that we exit here, if we ever export this it would be bad; # but also printing a massive stack trace for a `shiny run badpath` is way # unfriendly. We should probably throw a custom error that the shiny run # entrypoint knows not to print the stack trace for. - if not os.path.exists(module): - sys.stderr.write(f"Error: {module} not found\n") + if not os.path.exists(module_path): + sys.stderr.write(f"Error: {module_path} not found\n") sys.exit(1) - if not os.path.isfile(module): - sys.stderr.write(f"Error: {module} is not a file\n") + if not os.path.isfile(module_path): + sys.stderr.write(f"Error: {module_path} is not a file\n") sys.exit(1) - dirname, filename = os.path.split(module) + dirname, filename = os.path.split(module_path) module = filename[:-3] if filename.endswith(".py") else filename app_dir = dirname From 3d455d55dc1d64cad99c5475b10aaaa27eecc64d Mon Sep 17 00:00:00 2001 From: Gordon Shotwell Date: Tue, 18 Jul 2023 13:40:11 -0300 Subject: [PATCH 06/14] Rename shiny/examples to shiny/api-examples (and X/examples to X/api-examples) (#627) --- MANIFEST.in | 2 +- e2e/README.md | 2 +- e2e/conftest.py | 4 ++-- e2e/cpuinfo/test_app.py | 2 +- e2e/examples/test_examples.py | 4 +++- examples/penguins/app.py | 9 --------- pyrightconfig.json | 2 +- shiny/_docstring.py | 4 ++-- shiny/_main.py | 2 +- shiny/{examples => api-examples}/Calc/app.py | 0 shiny/{examples => api-examples}/Effect/app.py | 0 shiny/{examples => api-examples}/Module/app.py | 0 shiny/{examples => api-examples}/Progress/app.py | 0 .../{examples => api-examples}/SafeException/app.py | 0 .../SilentCancelOutputException/app.py | 0 .../SilentException/app.py | 0 shiny/{examples => api-examples}/Value/app.py | 0 shiny/{examples => api-examples}/close/app.py | 0 shiny/{examples => api-examples}/data_frame/app.py | 0 shiny/{examples => api-examples}/download/app.py | 0 .../{examples => api-examples}/download/mtcars.csv | 0 .../download_button/app.py | 0 .../download_button/mtcars.csv | 0 .../{examples => api-examples}/download_link/app.py | 0 .../download_link/mtcars.csv | 0 .../{examples => api-examples}/dynamic_route/app.py | 0 shiny/{examples => api-examples}/event/app.py | 0 shiny/{examples => api-examples}/file_reader/app.py | 0 .../file_reader/mtcars.csv | 0 shiny/{examples => api-examples}/include_css/app.py | 0 .../include_css/css/styles.css | 0 .../include_javascript/app.py | 0 .../include_javascript/js/app.js | 0 .../input_action_button/app.py | 0 .../input_action_link/app.py | 0 .../input_checkbox/app.py | 0 .../input_checkbox_group/app.py | 0 shiny/{examples => api-examples}/input_date/app.py | 0 .../input_date_range/app.py | 0 shiny/{examples => api-examples}/input_file/app.py | 0 .../{examples => api-examples}/input_numeric/app.py | 0 .../input_password/app.py | 0 .../input_radio_buttons/app.py | 0 .../{examples => api-examples}/input_select/app.py | 0 .../input_selectize/app.py | 0 .../{examples => api-examples}/input_slider/app.py | 0 .../{examples => api-examples}/input_switch/app.py | 0 shiny/{examples => api-examples}/input_text/app.py | 0 .../input_text_area/app.py | 0 shiny/{examples => api-examples}/insert_ui/app.py | 0 .../invalidate_later/app.py | 0 shiny/{examples => api-examples}/isolate/app.py | 0 .../layout_sidebar/app.py | 0 shiny/{examples => api-examples}/markdown/app.py | 0 shiny/{examples => api-examples}/modal/app.py | 0 shiny/{examples => api-examples}/nav/app.py | 0 .../{examples => api-examples}/navset_hidden/app.py | 0 .../notification_show/app.py | 0 shiny/{examples => api-examples}/on_ended/app.py | 0 shiny/{examples => api-examples}/on_flush/app.py | 0 shiny/{examples => api-examples}/on_flushed/app.py | 0 .../{examples => api-examples}/output_image/app.py | 0 .../output_image/posit-logo.png | Bin shiny/{examples => api-examples}/output_plot/app.py | 0 .../{examples => api-examples}/output_table/app.py | 0 .../output_table/mtcars.csv | 0 shiny/{examples => api-examples}/output_text/app.py | 0 shiny/{examples => api-examples}/output_ui/app.py | 0 shiny/{examples => api-examples}/page_fixed/app.py | 0 shiny/{examples => api-examples}/page_fluid/app.py | 0 .../panel_absolute/app.py | 0 .../panel_conditional/app.py | 0 shiny/{examples => api-examples}/panel_title/app.py | 0 shiny/{examples => api-examples}/poll/app.py | 0 shiny/{examples => api-examples}/remove_ui/app.py | 0 .../{examples => api-examples}/render_image/app.py | 0 shiny/{examples => api-examples}/req/app.py | 0 shiny/{examples => api-examples}/row/app.py | 0 .../send_custom_message/app.py | 0 shiny/{examples => api-examples}/template/app.py | 0 shiny/{examples => api-examples}/todo_list/app.py | 0 .../todo_list/requirements.txt | 0 .../update_action_button/app.py | 0 .../update_checkbox/app.py | 0 .../update_checkbox_group/app.py | 0 shiny/{examples => api-examples}/update_date/app.py | 0 .../update_date_range/app.py | 0 shiny/{examples => api-examples}/update_navs/app.py | 0 .../update_numeric/app.py | 0 .../update_radio_buttons/app.py | 0 .../{examples => api-examples}/update_select/app.py | 0 .../update_selectize/app.py | 0 .../{examples => api-examples}/update_slider/app.py | 0 shiny/{examples => api-examples}/update_text/app.py | 0 shiny/{examples => api-examples}/www_dir/app.py | 0 .../www_dir/www/css/more-styles.css | 0 .../www_dir/www/css/styles.css | 0 .../www_dir/www/js/changetext.js | 0 .../{examples => api-examples}/accordion/app.py | 0 .../accordion_panel/app.py | 0 .../accordion_panel_close/app.py | 0 .../accordion_panel_insert/app.py | 0 .../accordion_panel_open/app.py | 0 .../accordion_panel_remove/app.py | 0 .../accordion_panel_set/app.py | 0 .../accordion_panel_update/app.py | 0 .../as_fill_carrier/app.py | 0 .../{examples => api-examples}/as_fill_item/app.py | 0 .../as_fillable_container/app.py | 0 .../{examples => api-examples}/card/app.py | 0 .../{examples => api-examples}/card_body/app.py | 0 .../{examples => api-examples}/card_footer/app.py | 0 .../{examples => api-examples}/card_header/app.py | 0 .../{examples => api-examples}/card_image/app.py | 0 .../{examples => api-examples}/card_title/app.py | 0 .../input_text_area/app.py | 0 .../layout_column_wrap/app.py | 0 .../layout_sidebar/app.py | 0 .../{examples => api-examples}/page_sidebar/app.py | 0 .../showcase_left_center/app.py | 0 .../showcase_top_right/app.py | 0 .../{examples => api-examples}/sidebar/app.py | 0 .../sidebar_toggle/app.py | 0 .../{examples => api-examples}/value_box/app.py | 0 124 files changed, 12 insertions(+), 19 deletions(-) rename shiny/{examples => api-examples}/Calc/app.py (100%) rename shiny/{examples => api-examples}/Effect/app.py (100%) rename shiny/{examples => api-examples}/Module/app.py (100%) rename shiny/{examples => api-examples}/Progress/app.py (100%) rename shiny/{examples => api-examples}/SafeException/app.py (100%) rename shiny/{examples => api-examples}/SilentCancelOutputException/app.py (100%) rename shiny/{examples => api-examples}/SilentException/app.py (100%) rename shiny/{examples => api-examples}/Value/app.py (100%) rename shiny/{examples => api-examples}/close/app.py (100%) rename shiny/{examples => api-examples}/data_frame/app.py (100%) rename shiny/{examples => api-examples}/download/app.py (100%) rename shiny/{examples => api-examples}/download/mtcars.csv (100%) rename shiny/{examples => api-examples}/download_button/app.py (100%) rename shiny/{examples => api-examples}/download_button/mtcars.csv (100%) rename shiny/{examples => api-examples}/download_link/app.py (100%) rename shiny/{examples => api-examples}/download_link/mtcars.csv (100%) rename shiny/{examples => api-examples}/dynamic_route/app.py (100%) rename shiny/{examples => api-examples}/event/app.py (100%) rename shiny/{examples => api-examples}/file_reader/app.py (100%) rename shiny/{examples => api-examples}/file_reader/mtcars.csv (100%) rename shiny/{examples => api-examples}/include_css/app.py (100%) rename shiny/{examples => api-examples}/include_css/css/styles.css (100%) rename shiny/{examples => api-examples}/include_javascript/app.py (100%) rename shiny/{examples => api-examples}/include_javascript/js/app.js (100%) rename shiny/{examples => api-examples}/input_action_button/app.py (100%) rename shiny/{examples => api-examples}/input_action_link/app.py (100%) rename shiny/{examples => api-examples}/input_checkbox/app.py (100%) rename shiny/{examples => api-examples}/input_checkbox_group/app.py (100%) rename shiny/{examples => api-examples}/input_date/app.py (100%) rename shiny/{examples => api-examples}/input_date_range/app.py (100%) rename shiny/{examples => api-examples}/input_file/app.py (100%) rename shiny/{examples => api-examples}/input_numeric/app.py (100%) rename shiny/{examples => api-examples}/input_password/app.py (100%) rename shiny/{examples => api-examples}/input_radio_buttons/app.py (100%) rename shiny/{examples => api-examples}/input_select/app.py (100%) rename shiny/{examples => api-examples}/input_selectize/app.py (100%) rename shiny/{examples => api-examples}/input_slider/app.py (100%) rename shiny/{examples => api-examples}/input_switch/app.py (100%) rename shiny/{examples => api-examples}/input_text/app.py (100%) rename shiny/{examples => api-examples}/input_text_area/app.py (100%) rename shiny/{examples => api-examples}/insert_ui/app.py (100%) rename shiny/{examples => api-examples}/invalidate_later/app.py (100%) rename shiny/{examples => api-examples}/isolate/app.py (100%) rename shiny/{examples => api-examples}/layout_sidebar/app.py (100%) rename shiny/{examples => api-examples}/markdown/app.py (100%) rename shiny/{examples => api-examples}/modal/app.py (100%) rename shiny/{examples => api-examples}/nav/app.py (100%) rename shiny/{examples => api-examples}/navset_hidden/app.py (100%) rename shiny/{examples => api-examples}/notification_show/app.py (100%) rename shiny/{examples => api-examples}/on_ended/app.py (100%) rename shiny/{examples => api-examples}/on_flush/app.py (100%) rename shiny/{examples => api-examples}/on_flushed/app.py (100%) rename shiny/{examples => api-examples}/output_image/app.py (100%) rename shiny/{examples => api-examples}/output_image/posit-logo.png (100%) rename shiny/{examples => api-examples}/output_plot/app.py (100%) rename shiny/{examples => api-examples}/output_table/app.py (100%) rename shiny/{examples => api-examples}/output_table/mtcars.csv (100%) rename shiny/{examples => api-examples}/output_text/app.py (100%) rename shiny/{examples => api-examples}/output_ui/app.py (100%) rename shiny/{examples => api-examples}/page_fixed/app.py (100%) rename shiny/{examples => api-examples}/page_fluid/app.py (100%) rename shiny/{examples => api-examples}/panel_absolute/app.py (100%) rename shiny/{examples => api-examples}/panel_conditional/app.py (100%) rename shiny/{examples => api-examples}/panel_title/app.py (100%) rename shiny/{examples => api-examples}/poll/app.py (100%) rename shiny/{examples => api-examples}/remove_ui/app.py (100%) rename shiny/{examples => api-examples}/render_image/app.py (100%) rename shiny/{examples => api-examples}/req/app.py (100%) rename shiny/{examples => api-examples}/row/app.py (100%) rename shiny/{examples => api-examples}/send_custom_message/app.py (100%) rename shiny/{examples => api-examples}/template/app.py (100%) rename shiny/{examples => api-examples}/todo_list/app.py (100%) rename shiny/{examples => api-examples}/todo_list/requirements.txt (100%) rename shiny/{examples => api-examples}/update_action_button/app.py (100%) rename shiny/{examples => api-examples}/update_checkbox/app.py (100%) rename shiny/{examples => api-examples}/update_checkbox_group/app.py (100%) rename shiny/{examples => api-examples}/update_date/app.py (100%) rename shiny/{examples => api-examples}/update_date_range/app.py (100%) rename shiny/{examples => api-examples}/update_navs/app.py (100%) rename shiny/{examples => api-examples}/update_numeric/app.py (100%) rename shiny/{examples => api-examples}/update_radio_buttons/app.py (100%) rename shiny/{examples => api-examples}/update_select/app.py (100%) rename shiny/{examples => api-examples}/update_selectize/app.py (100%) rename shiny/{examples => api-examples}/update_slider/app.py (100%) rename shiny/{examples => api-examples}/update_text/app.py (100%) rename shiny/{examples => api-examples}/www_dir/app.py (100%) rename shiny/{examples => api-examples}/www_dir/www/css/more-styles.css (100%) rename shiny/{examples => api-examples}/www_dir/www/css/styles.css (100%) rename shiny/{examples => api-examples}/www_dir/www/js/changetext.js (100%) rename shiny/experimental/{examples => api-examples}/accordion/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_close/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_insert/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_open/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_remove/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_set/app.py (100%) rename shiny/experimental/{examples => api-examples}/accordion_panel_update/app.py (100%) rename shiny/experimental/{examples => api-examples}/as_fill_carrier/app.py (100%) rename shiny/experimental/{examples => api-examples}/as_fill_item/app.py (100%) rename shiny/experimental/{examples => api-examples}/as_fillable_container/app.py (100%) rename shiny/experimental/{examples => api-examples}/card/app.py (100%) rename shiny/experimental/{examples => api-examples}/card_body/app.py (100%) rename shiny/experimental/{examples => api-examples}/card_footer/app.py (100%) rename shiny/experimental/{examples => api-examples}/card_header/app.py (100%) rename shiny/experimental/{examples => api-examples}/card_image/app.py (100%) rename shiny/experimental/{examples => api-examples}/card_title/app.py (100%) rename shiny/experimental/{examples => api-examples}/input_text_area/app.py (100%) rename shiny/experimental/{examples => api-examples}/layout_column_wrap/app.py (100%) rename shiny/experimental/{examples => api-examples}/layout_sidebar/app.py (100%) rename shiny/experimental/{examples => api-examples}/page_sidebar/app.py (100%) rename shiny/experimental/{examples => api-examples}/showcase_left_center/app.py (100%) rename shiny/experimental/{examples => api-examples}/showcase_top_right/app.py (100%) rename shiny/experimental/{examples => api-examples}/sidebar/app.py (100%) rename shiny/experimental/{examples => api-examples}/sidebar_toggle/app.py (100%) rename shiny/experimental/{examples => api-examples}/value_box/app.py (100%) diff --git a/MANIFEST.in b/MANIFEST.in index a5d757e91..3bedfd7ec 100644 --- a/MANIFEST.in +++ b/MANIFEST.in @@ -8,5 +8,5 @@ recursive-include docs *.rst conf.py Makefile make.bat *.jpg *.png *.gif recursive-include shiny/www * recursive-include shiny/experimental/www * -recursive-include shiny/examples * +recursive-include shiny/api-examples * recursive-include shiny/ui/dataframe/js/dist * diff --git a/e2e/README.md b/e2e/README.md index bb3a34d57..18c936a89 100644 --- a/e2e/README.md +++ b/e2e/README.md @@ -10,7 +10,7 @@ tests against their apps.) The actual tests are in subdirectories. Each subdirectory contains one or more Pytest files (`test_*.py`) containing [Playwright](https://playwright.dev/python/) assertions, and optionally, a single app (`app.py`) that the assertions test against. (The app is -optional, because the tests may also be for apps in the `../examples` or `../shiny/examples` directory.) +optional, because the tests may also be for apps in the `../examples` or `../shiny/api-examples` directory.) ## Running tests diff --git a/e2e/conftest.py b/e2e/conftest.py index f8f99e1cd..12c2f99b2 100644 --- a/e2e/conftest.py +++ b/e2e/conftest.py @@ -194,9 +194,9 @@ def create_example_fixture(example_name: str, scope: str = "module"): def create_doc_example_fixture(example_name: str, scope: str = "module"): - """Used to create app fixtures from apps in py-shiny/shiny/examples""" + """Used to create app fixtures from apps in py-shiny/shiny/api-examples""" return create_app_fixture( - here / "../shiny/examples" / example_name / "app.py", scope + here / "../shiny/api-examples" / example_name / "app.py", scope ) diff --git a/e2e/cpuinfo/test_app.py b/e2e/cpuinfo/test_app.py index 141a1add6..d07976b70 100644 --- a/e2e/cpuinfo/test_app.py +++ b/e2e/cpuinfo/test_app.py @@ -1,6 +1,6 @@ # pyright: reportUnknownMemberType=false -# TODO-barret; Convert test into loop that tests all examples to make sure they load +# TODO-karan; Convert test into loop that tests all examples to make sure they load import re diff --git a/e2e/examples/test_examples.py b/e2e/examples/test_examples.py index 5534039d8..cd512bf0c 100644 --- a/e2e/examples/test_examples.py +++ b/e2e/examples/test_examples.py @@ -24,7 +24,7 @@ def get_apps(path: str) -> typing.List[str]: example_apps: typing.List[str] = [ *get_apps("../../examples"), - *get_apps("../../shiny/examples"), + *get_apps("../../shiny/api-examples"), ] app_idle_wait = {"duration": 300, "timeout": 5 * 1000} @@ -36,6 +36,8 @@ def get_apps(path: str) -> typing.List[str]: "SafeException": True, "global_pyplot": True, "static_plots": ["PlotnineWarning", "RuntimeWarning"], + # https://github.com/rstudio/py-shiny/issues/611#issuecomment-1632866419 + "penguins": ["UserWarning", "plt.tight_layout"], } app_allow_js_errors: typing.Dict[str, typing.List[str]] = { "brownian": ["Failed to acquire camera feed:"], diff --git a/examples/penguins/app.py b/examples/penguins/app.py index 863aee746..e019a50cc 100644 --- a/examples/penguins/app.py +++ b/examples/penguins/app.py @@ -1,11 +1,9 @@ # TODO-future: Add filter of X varaible to reduce the data? (Here we would show "Gentoo" has count 0, rather than remove if no data exists) # TODO-future: Add brushing to zoom into the plot. The counts should represent the data in the zoomed area. (Single click would zoom out) -import warnings from pathlib import Path from typing import List -import matplotlib import pandas as pd import seaborn as sns import shinyswatch @@ -14,13 +12,6 @@ import shiny.experimental as x from shiny import App, Inputs, Outputs, Session, reactive, render, req, ui -# There is a matplotlib bug which causes CI failures -# see https://github.com/rstudio/py-shiny/issues/611#issuecomment-1632866419 -if matplotlib.__version__ == "3.7.2": - warnings.filterwarnings( - "ignore", category=UserWarning, message="The figure layout has changed to tight" - ) - sns.set_theme() www_dir = Path(__file__).parent.resolve() / "www" diff --git a/pyrightconfig.json b/pyrightconfig.json index baf6b8057..6f6e3dcc4 100644 --- a/pyrightconfig.json +++ b/pyrightconfig.json @@ -1,6 +1,6 @@ { "ignore": [ - "shiny/examples", + "shiny/api-examples", "examples", "build", "dist", diff --git a/shiny/_docstring.py b/shiny/_docstring.py index 2abd578b4..699fcc450 100644 --- a/shiny/_docstring.py +++ b/shiny/_docstring.py @@ -4,7 +4,7 @@ import os from typing import Any, Callable, Literal, TypeVar -ex_dir: str = os.path.join(os.path.dirname(os.path.abspath(__file__)), "examples") +ex_dir: str = os.path.join(os.path.dirname(os.path.abspath(__file__)), "api-examples") FuncType = Callable[..., Any] F = TypeVar("F", bound=FuncType) @@ -32,7 +32,7 @@ def add_example( Add an example to the docstring of a function, method, or class. This decorator must, at the moment, be used on a function, method, or class whose - ``__name__`` matches the name of directory under ``shiny/examples/``, and must + ``__name__`` matches the name of directory under ``shiny/api-examples/``, and must also contain a ``app.py`` file in that directory. Parameters diff --git a/shiny/_main.py b/shiny/_main.py index 21853285f..f53e51392 100644 --- a/shiny/_main.py +++ b/shiny/_main.py @@ -422,7 +422,7 @@ def create(appdir: str) -> None: app_dir.mkdir() shutil.copyfile( - Path(__file__).parent / "examples" / "template" / "app.py", app_path + Path(__file__).parent / "api-examples" / "template" / "app.py", app_path ) print(f"Created Shiny app at {app_dir / 'app.py'}") diff --git a/shiny/examples/Calc/app.py b/shiny/api-examples/Calc/app.py similarity index 100% rename from shiny/examples/Calc/app.py rename to shiny/api-examples/Calc/app.py diff --git a/shiny/examples/Effect/app.py b/shiny/api-examples/Effect/app.py similarity index 100% rename from shiny/examples/Effect/app.py rename to shiny/api-examples/Effect/app.py diff --git a/shiny/examples/Module/app.py b/shiny/api-examples/Module/app.py similarity index 100% rename from shiny/examples/Module/app.py rename to shiny/api-examples/Module/app.py diff --git a/shiny/examples/Progress/app.py b/shiny/api-examples/Progress/app.py similarity index 100% rename from shiny/examples/Progress/app.py rename to shiny/api-examples/Progress/app.py diff --git a/shiny/examples/SafeException/app.py b/shiny/api-examples/SafeException/app.py similarity index 100% rename from shiny/examples/SafeException/app.py rename to shiny/api-examples/SafeException/app.py diff --git a/shiny/examples/SilentCancelOutputException/app.py b/shiny/api-examples/SilentCancelOutputException/app.py similarity index 100% rename from shiny/examples/SilentCancelOutputException/app.py rename to shiny/api-examples/SilentCancelOutputException/app.py diff --git a/shiny/examples/SilentException/app.py b/shiny/api-examples/SilentException/app.py similarity index 100% rename from shiny/examples/SilentException/app.py rename to shiny/api-examples/SilentException/app.py diff --git a/shiny/examples/Value/app.py b/shiny/api-examples/Value/app.py similarity index 100% rename from shiny/examples/Value/app.py rename to shiny/api-examples/Value/app.py diff --git a/shiny/examples/close/app.py b/shiny/api-examples/close/app.py similarity index 100% rename from shiny/examples/close/app.py rename to shiny/api-examples/close/app.py diff --git a/shiny/examples/data_frame/app.py b/shiny/api-examples/data_frame/app.py similarity index 100% rename from shiny/examples/data_frame/app.py rename to shiny/api-examples/data_frame/app.py diff --git a/shiny/examples/download/app.py b/shiny/api-examples/download/app.py similarity index 100% rename from shiny/examples/download/app.py rename to shiny/api-examples/download/app.py diff --git a/shiny/examples/download/mtcars.csv b/shiny/api-examples/download/mtcars.csv similarity index 100% rename from shiny/examples/download/mtcars.csv rename to shiny/api-examples/download/mtcars.csv diff --git a/shiny/examples/download_button/app.py b/shiny/api-examples/download_button/app.py similarity index 100% rename from shiny/examples/download_button/app.py rename to shiny/api-examples/download_button/app.py diff --git a/shiny/examples/download_button/mtcars.csv b/shiny/api-examples/download_button/mtcars.csv similarity index 100% rename from shiny/examples/download_button/mtcars.csv rename to shiny/api-examples/download_button/mtcars.csv diff --git a/shiny/examples/download_link/app.py b/shiny/api-examples/download_link/app.py similarity index 100% rename from shiny/examples/download_link/app.py rename to shiny/api-examples/download_link/app.py diff --git a/shiny/examples/download_link/mtcars.csv b/shiny/api-examples/download_link/mtcars.csv similarity index 100% rename from shiny/examples/download_link/mtcars.csv rename to shiny/api-examples/download_link/mtcars.csv diff --git a/shiny/examples/dynamic_route/app.py b/shiny/api-examples/dynamic_route/app.py similarity index 100% rename from shiny/examples/dynamic_route/app.py rename to shiny/api-examples/dynamic_route/app.py diff --git a/shiny/examples/event/app.py b/shiny/api-examples/event/app.py similarity index 100% rename from shiny/examples/event/app.py rename to shiny/api-examples/event/app.py diff --git a/shiny/examples/file_reader/app.py b/shiny/api-examples/file_reader/app.py similarity index 100% rename from shiny/examples/file_reader/app.py rename to shiny/api-examples/file_reader/app.py diff --git a/shiny/examples/file_reader/mtcars.csv b/shiny/api-examples/file_reader/mtcars.csv similarity index 100% rename from shiny/examples/file_reader/mtcars.csv rename to shiny/api-examples/file_reader/mtcars.csv diff --git a/shiny/examples/include_css/app.py b/shiny/api-examples/include_css/app.py similarity index 100% rename from shiny/examples/include_css/app.py rename to shiny/api-examples/include_css/app.py diff --git a/shiny/examples/include_css/css/styles.css b/shiny/api-examples/include_css/css/styles.css similarity index 100% rename from shiny/examples/include_css/css/styles.css rename to shiny/api-examples/include_css/css/styles.css diff --git a/shiny/examples/include_javascript/app.py b/shiny/api-examples/include_javascript/app.py similarity index 100% rename from shiny/examples/include_javascript/app.py rename to shiny/api-examples/include_javascript/app.py diff --git a/shiny/examples/include_javascript/js/app.js b/shiny/api-examples/include_javascript/js/app.js similarity index 100% rename from shiny/examples/include_javascript/js/app.js rename to shiny/api-examples/include_javascript/js/app.js diff --git a/shiny/examples/input_action_button/app.py b/shiny/api-examples/input_action_button/app.py similarity index 100% rename from shiny/examples/input_action_button/app.py rename to shiny/api-examples/input_action_button/app.py diff --git a/shiny/examples/input_action_link/app.py b/shiny/api-examples/input_action_link/app.py similarity index 100% rename from shiny/examples/input_action_link/app.py rename to shiny/api-examples/input_action_link/app.py diff --git a/shiny/examples/input_checkbox/app.py b/shiny/api-examples/input_checkbox/app.py similarity index 100% rename from shiny/examples/input_checkbox/app.py rename to shiny/api-examples/input_checkbox/app.py diff --git a/shiny/examples/input_checkbox_group/app.py b/shiny/api-examples/input_checkbox_group/app.py similarity index 100% rename from shiny/examples/input_checkbox_group/app.py rename to shiny/api-examples/input_checkbox_group/app.py diff --git a/shiny/examples/input_date/app.py b/shiny/api-examples/input_date/app.py similarity index 100% rename from shiny/examples/input_date/app.py rename to shiny/api-examples/input_date/app.py diff --git a/shiny/examples/input_date_range/app.py b/shiny/api-examples/input_date_range/app.py similarity index 100% rename from shiny/examples/input_date_range/app.py rename to shiny/api-examples/input_date_range/app.py diff --git a/shiny/examples/input_file/app.py b/shiny/api-examples/input_file/app.py similarity index 100% rename from shiny/examples/input_file/app.py rename to shiny/api-examples/input_file/app.py diff --git a/shiny/examples/input_numeric/app.py b/shiny/api-examples/input_numeric/app.py similarity index 100% rename from shiny/examples/input_numeric/app.py rename to shiny/api-examples/input_numeric/app.py diff --git a/shiny/examples/input_password/app.py b/shiny/api-examples/input_password/app.py similarity index 100% rename from shiny/examples/input_password/app.py rename to shiny/api-examples/input_password/app.py diff --git a/shiny/examples/input_radio_buttons/app.py b/shiny/api-examples/input_radio_buttons/app.py similarity index 100% rename from shiny/examples/input_radio_buttons/app.py rename to shiny/api-examples/input_radio_buttons/app.py diff --git a/shiny/examples/input_select/app.py b/shiny/api-examples/input_select/app.py similarity index 100% rename from shiny/examples/input_select/app.py rename to shiny/api-examples/input_select/app.py diff --git a/shiny/examples/input_selectize/app.py b/shiny/api-examples/input_selectize/app.py similarity index 100% rename from shiny/examples/input_selectize/app.py rename to shiny/api-examples/input_selectize/app.py diff --git a/shiny/examples/input_slider/app.py b/shiny/api-examples/input_slider/app.py similarity index 100% rename from shiny/examples/input_slider/app.py rename to shiny/api-examples/input_slider/app.py diff --git a/shiny/examples/input_switch/app.py b/shiny/api-examples/input_switch/app.py similarity index 100% rename from shiny/examples/input_switch/app.py rename to shiny/api-examples/input_switch/app.py diff --git a/shiny/examples/input_text/app.py b/shiny/api-examples/input_text/app.py similarity index 100% rename from shiny/examples/input_text/app.py rename to shiny/api-examples/input_text/app.py diff --git a/shiny/examples/input_text_area/app.py b/shiny/api-examples/input_text_area/app.py similarity index 100% rename from shiny/examples/input_text_area/app.py rename to shiny/api-examples/input_text_area/app.py diff --git a/shiny/examples/insert_ui/app.py b/shiny/api-examples/insert_ui/app.py similarity index 100% rename from shiny/examples/insert_ui/app.py rename to shiny/api-examples/insert_ui/app.py diff --git a/shiny/examples/invalidate_later/app.py b/shiny/api-examples/invalidate_later/app.py similarity index 100% rename from shiny/examples/invalidate_later/app.py rename to shiny/api-examples/invalidate_later/app.py diff --git a/shiny/examples/isolate/app.py b/shiny/api-examples/isolate/app.py similarity index 100% rename from shiny/examples/isolate/app.py rename to shiny/api-examples/isolate/app.py diff --git a/shiny/examples/layout_sidebar/app.py b/shiny/api-examples/layout_sidebar/app.py similarity index 100% rename from shiny/examples/layout_sidebar/app.py rename to shiny/api-examples/layout_sidebar/app.py diff --git a/shiny/examples/markdown/app.py b/shiny/api-examples/markdown/app.py similarity index 100% rename from shiny/examples/markdown/app.py rename to shiny/api-examples/markdown/app.py diff --git a/shiny/examples/modal/app.py b/shiny/api-examples/modal/app.py similarity index 100% rename from shiny/examples/modal/app.py rename to shiny/api-examples/modal/app.py diff --git a/shiny/examples/nav/app.py b/shiny/api-examples/nav/app.py similarity index 100% rename from shiny/examples/nav/app.py rename to shiny/api-examples/nav/app.py diff --git a/shiny/examples/navset_hidden/app.py b/shiny/api-examples/navset_hidden/app.py similarity index 100% rename from shiny/examples/navset_hidden/app.py rename to shiny/api-examples/navset_hidden/app.py diff --git a/shiny/examples/notification_show/app.py b/shiny/api-examples/notification_show/app.py similarity index 100% rename from shiny/examples/notification_show/app.py rename to shiny/api-examples/notification_show/app.py diff --git a/shiny/examples/on_ended/app.py b/shiny/api-examples/on_ended/app.py similarity index 100% rename from shiny/examples/on_ended/app.py rename to shiny/api-examples/on_ended/app.py diff --git a/shiny/examples/on_flush/app.py b/shiny/api-examples/on_flush/app.py similarity index 100% rename from shiny/examples/on_flush/app.py rename to shiny/api-examples/on_flush/app.py diff --git a/shiny/examples/on_flushed/app.py b/shiny/api-examples/on_flushed/app.py similarity index 100% rename from shiny/examples/on_flushed/app.py rename to shiny/api-examples/on_flushed/app.py diff --git a/shiny/examples/output_image/app.py b/shiny/api-examples/output_image/app.py similarity index 100% rename from shiny/examples/output_image/app.py rename to shiny/api-examples/output_image/app.py diff --git a/shiny/examples/output_image/posit-logo.png b/shiny/api-examples/output_image/posit-logo.png similarity index 100% rename from shiny/examples/output_image/posit-logo.png rename to shiny/api-examples/output_image/posit-logo.png diff --git a/shiny/examples/output_plot/app.py b/shiny/api-examples/output_plot/app.py similarity index 100% rename from shiny/examples/output_plot/app.py rename to shiny/api-examples/output_plot/app.py diff --git a/shiny/examples/output_table/app.py b/shiny/api-examples/output_table/app.py similarity index 100% rename from shiny/examples/output_table/app.py rename to shiny/api-examples/output_table/app.py diff --git a/shiny/examples/output_table/mtcars.csv b/shiny/api-examples/output_table/mtcars.csv similarity index 100% rename from shiny/examples/output_table/mtcars.csv rename to shiny/api-examples/output_table/mtcars.csv diff --git a/shiny/examples/output_text/app.py b/shiny/api-examples/output_text/app.py similarity index 100% rename from shiny/examples/output_text/app.py rename to shiny/api-examples/output_text/app.py diff --git a/shiny/examples/output_ui/app.py b/shiny/api-examples/output_ui/app.py similarity index 100% rename from shiny/examples/output_ui/app.py rename to shiny/api-examples/output_ui/app.py diff --git a/shiny/examples/page_fixed/app.py b/shiny/api-examples/page_fixed/app.py similarity index 100% rename from shiny/examples/page_fixed/app.py rename to shiny/api-examples/page_fixed/app.py diff --git a/shiny/examples/page_fluid/app.py b/shiny/api-examples/page_fluid/app.py similarity index 100% rename from shiny/examples/page_fluid/app.py rename to shiny/api-examples/page_fluid/app.py diff --git a/shiny/examples/panel_absolute/app.py b/shiny/api-examples/panel_absolute/app.py similarity index 100% rename from shiny/examples/panel_absolute/app.py rename to shiny/api-examples/panel_absolute/app.py diff --git a/shiny/examples/panel_conditional/app.py b/shiny/api-examples/panel_conditional/app.py similarity index 100% rename from shiny/examples/panel_conditional/app.py rename to shiny/api-examples/panel_conditional/app.py diff --git a/shiny/examples/panel_title/app.py b/shiny/api-examples/panel_title/app.py similarity index 100% rename from shiny/examples/panel_title/app.py rename to shiny/api-examples/panel_title/app.py diff --git a/shiny/examples/poll/app.py b/shiny/api-examples/poll/app.py similarity index 100% rename from shiny/examples/poll/app.py rename to shiny/api-examples/poll/app.py diff --git a/shiny/examples/remove_ui/app.py b/shiny/api-examples/remove_ui/app.py similarity index 100% rename from shiny/examples/remove_ui/app.py rename to shiny/api-examples/remove_ui/app.py diff --git a/shiny/examples/render_image/app.py b/shiny/api-examples/render_image/app.py similarity index 100% rename from shiny/examples/render_image/app.py rename to shiny/api-examples/render_image/app.py diff --git a/shiny/examples/req/app.py b/shiny/api-examples/req/app.py similarity index 100% rename from shiny/examples/req/app.py rename to shiny/api-examples/req/app.py diff --git a/shiny/examples/row/app.py b/shiny/api-examples/row/app.py similarity index 100% rename from shiny/examples/row/app.py rename to shiny/api-examples/row/app.py diff --git a/shiny/examples/send_custom_message/app.py b/shiny/api-examples/send_custom_message/app.py similarity index 100% rename from shiny/examples/send_custom_message/app.py rename to shiny/api-examples/send_custom_message/app.py diff --git a/shiny/examples/template/app.py b/shiny/api-examples/template/app.py similarity index 100% rename from shiny/examples/template/app.py rename to shiny/api-examples/template/app.py diff --git a/shiny/examples/todo_list/app.py b/shiny/api-examples/todo_list/app.py similarity index 100% rename from shiny/examples/todo_list/app.py rename to shiny/api-examples/todo_list/app.py diff --git a/shiny/examples/todo_list/requirements.txt b/shiny/api-examples/todo_list/requirements.txt similarity index 100% rename from shiny/examples/todo_list/requirements.txt rename to shiny/api-examples/todo_list/requirements.txt diff --git a/shiny/examples/update_action_button/app.py b/shiny/api-examples/update_action_button/app.py similarity index 100% rename from shiny/examples/update_action_button/app.py rename to shiny/api-examples/update_action_button/app.py diff --git a/shiny/examples/update_checkbox/app.py b/shiny/api-examples/update_checkbox/app.py similarity index 100% rename from shiny/examples/update_checkbox/app.py rename to shiny/api-examples/update_checkbox/app.py diff --git a/shiny/examples/update_checkbox_group/app.py b/shiny/api-examples/update_checkbox_group/app.py similarity index 100% rename from shiny/examples/update_checkbox_group/app.py rename to shiny/api-examples/update_checkbox_group/app.py diff --git a/shiny/examples/update_date/app.py b/shiny/api-examples/update_date/app.py similarity index 100% rename from shiny/examples/update_date/app.py rename to shiny/api-examples/update_date/app.py diff --git a/shiny/examples/update_date_range/app.py b/shiny/api-examples/update_date_range/app.py similarity index 100% rename from shiny/examples/update_date_range/app.py rename to shiny/api-examples/update_date_range/app.py diff --git a/shiny/examples/update_navs/app.py b/shiny/api-examples/update_navs/app.py similarity index 100% rename from shiny/examples/update_navs/app.py rename to shiny/api-examples/update_navs/app.py diff --git a/shiny/examples/update_numeric/app.py b/shiny/api-examples/update_numeric/app.py similarity index 100% rename from shiny/examples/update_numeric/app.py rename to shiny/api-examples/update_numeric/app.py diff --git a/shiny/examples/update_radio_buttons/app.py b/shiny/api-examples/update_radio_buttons/app.py similarity index 100% rename from shiny/examples/update_radio_buttons/app.py rename to shiny/api-examples/update_radio_buttons/app.py diff --git a/shiny/examples/update_select/app.py b/shiny/api-examples/update_select/app.py similarity index 100% rename from shiny/examples/update_select/app.py rename to shiny/api-examples/update_select/app.py diff --git a/shiny/examples/update_selectize/app.py b/shiny/api-examples/update_selectize/app.py similarity index 100% rename from shiny/examples/update_selectize/app.py rename to shiny/api-examples/update_selectize/app.py diff --git a/shiny/examples/update_slider/app.py b/shiny/api-examples/update_slider/app.py similarity index 100% rename from shiny/examples/update_slider/app.py rename to shiny/api-examples/update_slider/app.py diff --git a/shiny/examples/update_text/app.py b/shiny/api-examples/update_text/app.py similarity index 100% rename from shiny/examples/update_text/app.py rename to shiny/api-examples/update_text/app.py diff --git a/shiny/examples/www_dir/app.py b/shiny/api-examples/www_dir/app.py similarity index 100% rename from shiny/examples/www_dir/app.py rename to shiny/api-examples/www_dir/app.py diff --git a/shiny/examples/www_dir/www/css/more-styles.css b/shiny/api-examples/www_dir/www/css/more-styles.css similarity index 100% rename from shiny/examples/www_dir/www/css/more-styles.css rename to shiny/api-examples/www_dir/www/css/more-styles.css diff --git a/shiny/examples/www_dir/www/css/styles.css b/shiny/api-examples/www_dir/www/css/styles.css similarity index 100% rename from shiny/examples/www_dir/www/css/styles.css rename to shiny/api-examples/www_dir/www/css/styles.css diff --git a/shiny/examples/www_dir/www/js/changetext.js b/shiny/api-examples/www_dir/www/js/changetext.js similarity index 100% rename from shiny/examples/www_dir/www/js/changetext.js rename to shiny/api-examples/www_dir/www/js/changetext.js diff --git a/shiny/experimental/examples/accordion/app.py b/shiny/experimental/api-examples/accordion/app.py similarity index 100% rename from shiny/experimental/examples/accordion/app.py rename to shiny/experimental/api-examples/accordion/app.py diff --git a/shiny/experimental/examples/accordion_panel/app.py b/shiny/experimental/api-examples/accordion_panel/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel/app.py rename to shiny/experimental/api-examples/accordion_panel/app.py diff --git a/shiny/experimental/examples/accordion_panel_close/app.py b/shiny/experimental/api-examples/accordion_panel_close/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_close/app.py rename to shiny/experimental/api-examples/accordion_panel_close/app.py diff --git a/shiny/experimental/examples/accordion_panel_insert/app.py b/shiny/experimental/api-examples/accordion_panel_insert/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_insert/app.py rename to shiny/experimental/api-examples/accordion_panel_insert/app.py diff --git a/shiny/experimental/examples/accordion_panel_open/app.py b/shiny/experimental/api-examples/accordion_panel_open/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_open/app.py rename to shiny/experimental/api-examples/accordion_panel_open/app.py diff --git a/shiny/experimental/examples/accordion_panel_remove/app.py b/shiny/experimental/api-examples/accordion_panel_remove/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_remove/app.py rename to shiny/experimental/api-examples/accordion_panel_remove/app.py diff --git a/shiny/experimental/examples/accordion_panel_set/app.py b/shiny/experimental/api-examples/accordion_panel_set/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_set/app.py rename to shiny/experimental/api-examples/accordion_panel_set/app.py diff --git a/shiny/experimental/examples/accordion_panel_update/app.py b/shiny/experimental/api-examples/accordion_panel_update/app.py similarity index 100% rename from shiny/experimental/examples/accordion_panel_update/app.py rename to shiny/experimental/api-examples/accordion_panel_update/app.py diff --git a/shiny/experimental/examples/as_fill_carrier/app.py b/shiny/experimental/api-examples/as_fill_carrier/app.py similarity index 100% rename from shiny/experimental/examples/as_fill_carrier/app.py rename to shiny/experimental/api-examples/as_fill_carrier/app.py diff --git a/shiny/experimental/examples/as_fill_item/app.py b/shiny/experimental/api-examples/as_fill_item/app.py similarity index 100% rename from shiny/experimental/examples/as_fill_item/app.py rename to shiny/experimental/api-examples/as_fill_item/app.py diff --git a/shiny/experimental/examples/as_fillable_container/app.py b/shiny/experimental/api-examples/as_fillable_container/app.py similarity index 100% rename from shiny/experimental/examples/as_fillable_container/app.py rename to shiny/experimental/api-examples/as_fillable_container/app.py diff --git a/shiny/experimental/examples/card/app.py b/shiny/experimental/api-examples/card/app.py similarity index 100% rename from shiny/experimental/examples/card/app.py rename to shiny/experimental/api-examples/card/app.py diff --git a/shiny/experimental/examples/card_body/app.py b/shiny/experimental/api-examples/card_body/app.py similarity index 100% rename from shiny/experimental/examples/card_body/app.py rename to shiny/experimental/api-examples/card_body/app.py diff --git a/shiny/experimental/examples/card_footer/app.py b/shiny/experimental/api-examples/card_footer/app.py similarity index 100% rename from shiny/experimental/examples/card_footer/app.py rename to shiny/experimental/api-examples/card_footer/app.py diff --git a/shiny/experimental/examples/card_header/app.py b/shiny/experimental/api-examples/card_header/app.py similarity index 100% rename from shiny/experimental/examples/card_header/app.py rename to shiny/experimental/api-examples/card_header/app.py diff --git a/shiny/experimental/examples/card_image/app.py b/shiny/experimental/api-examples/card_image/app.py similarity index 100% rename from shiny/experimental/examples/card_image/app.py rename to shiny/experimental/api-examples/card_image/app.py diff --git a/shiny/experimental/examples/card_title/app.py b/shiny/experimental/api-examples/card_title/app.py similarity index 100% rename from shiny/experimental/examples/card_title/app.py rename to shiny/experimental/api-examples/card_title/app.py diff --git a/shiny/experimental/examples/input_text_area/app.py b/shiny/experimental/api-examples/input_text_area/app.py similarity index 100% rename from shiny/experimental/examples/input_text_area/app.py rename to shiny/experimental/api-examples/input_text_area/app.py diff --git a/shiny/experimental/examples/layout_column_wrap/app.py b/shiny/experimental/api-examples/layout_column_wrap/app.py similarity index 100% rename from shiny/experimental/examples/layout_column_wrap/app.py rename to shiny/experimental/api-examples/layout_column_wrap/app.py diff --git a/shiny/experimental/examples/layout_sidebar/app.py b/shiny/experimental/api-examples/layout_sidebar/app.py similarity index 100% rename from shiny/experimental/examples/layout_sidebar/app.py rename to shiny/experimental/api-examples/layout_sidebar/app.py diff --git a/shiny/experimental/examples/page_sidebar/app.py b/shiny/experimental/api-examples/page_sidebar/app.py similarity index 100% rename from shiny/experimental/examples/page_sidebar/app.py rename to shiny/experimental/api-examples/page_sidebar/app.py diff --git a/shiny/experimental/examples/showcase_left_center/app.py b/shiny/experimental/api-examples/showcase_left_center/app.py similarity index 100% rename from shiny/experimental/examples/showcase_left_center/app.py rename to shiny/experimental/api-examples/showcase_left_center/app.py diff --git a/shiny/experimental/examples/showcase_top_right/app.py b/shiny/experimental/api-examples/showcase_top_right/app.py similarity index 100% rename from shiny/experimental/examples/showcase_top_right/app.py rename to shiny/experimental/api-examples/showcase_top_right/app.py diff --git a/shiny/experimental/examples/sidebar/app.py b/shiny/experimental/api-examples/sidebar/app.py similarity index 100% rename from shiny/experimental/examples/sidebar/app.py rename to shiny/experimental/api-examples/sidebar/app.py diff --git a/shiny/experimental/examples/sidebar_toggle/app.py b/shiny/experimental/api-examples/sidebar_toggle/app.py similarity index 100% rename from shiny/experimental/examples/sidebar_toggle/app.py rename to shiny/experimental/api-examples/sidebar_toggle/app.py diff --git a/shiny/experimental/examples/value_box/app.py b/shiny/experimental/api-examples/value_box/app.py similarity index 100% rename from shiny/experimental/examples/value_box/app.py rename to shiny/experimental/api-examples/value_box/app.py From 4d0e5ee25606fa40b033022c85659a3767e5f0e2 Mon Sep 17 00:00:00 2001 From: Barret Schloerke Date: Tue, 18 Jul 2023 17:03:37 -0400 Subject: [PATCH 07/14] Use `blib::bs_theme(5,"shiny")` for py-shiny theme (#624) --- CHANGELOG.md | 2 + e2e/controls.py | 10 +- e2e/inputs/input_slider/app.py | 1 + .../input_slider/test_input_slider_app.py | 12 +- e2e/inputs/test_input_checkbox.py | 4 +- e2e/inputs/test_input_file.py | 2 +- e2e/inputs/test_input_slider.py | 6 +- e2e/outputs/test_output_text.py | 2 + examples/duckdb/.gitignore | 3 + examples/penguins/app.py | 2 - js/package-lock.json | 1 - js/package.json | 6 +- scripts/htmlDependencies.R | 110 +++++++++++++--- shiny/__init__.py | 2 +- shiny/api-examples/output_text/app.py | 16 ++- shiny/experimental/e2e/navbar/app.py | 1 - shiny/experimental/ui/_fill.py | 2 +- shiny/experimental/ui/_htmldeps.py | 103 ++++++++++----- shiny/experimental/ui/_layout.py | 2 + shiny/experimental/ui/_navs.py | 8 +- shiny/experimental/ui/_page.py | 15 ++- shiny/experimental/ui/_valuebox.py | 4 + shiny/experimental/www/bslib/_version.json | 2 +- .../bslib/components/accordion/accordion.css | 1 + .../{ => accordion}/accordion.min.js | 0 .../{ => accordion}/accordion.min.js.map | 2 +- .../www/bslib/components/card.min.js | 3 - .../www/bslib/components/card.min.js.map | 7 - .../www/bslib/components/card/card.css | 1 + .../www/bslib/components/card/card.min.js | 3 + .../www/bslib/components/card/card.min.js.map | 7 + .../www/bslib/components/grid/grid.css | 1 + .../components/nav_spacer/nav_spacer.css | 1 + .../page_fillable/page_fillable.css | 1 + .../components/page_navbar/page_navbar.css | 1 + .../components/page_sidebar/page_sidebar.css | 1 + .../www/bslib/components/sidebar.min.js | 3 - .../www/bslib/components/sidebar.min.js.map | 7 - .../www/bslib/components/sidebar/sidebar.css | 1 + .../bslib/components/sidebar/sidebar.min.js | 3 + .../components/sidebar/sidebar.min.js.map | 7 + .../bslib/components/value_box/value_box.css | 1 + .../webComponents/webComponents.min.js | 112 ++++++++++++++++ .../webComponents/webComponents.min.js.map | 7 + .../experimental/www/htmltools/_version.json | 2 +- shiny/ui/_html_dependencies.py | 11 ++ shiny/ui/_input_slider.py | 2 +- shiny/ui/_navs.py | 8 +- shiny/ui/_page.py | 4 +- shiny/ui/_x/_htmldeps.py | 18 ++- shiny/www/shared/_version.json | 2 +- .../components/nav_spacer/nav_spacer.css | 1 + .../components/page_sidebar/page_sidebar.css | 1 + .../shared/_x/bslib/components/sidebar.min.js | 3 - .../_x/bslib/components/sidebar.min.js.map | 7 - .../_x/bslib/components/sidebar/sidebar.css | 1 + .../bslib/components/sidebar/sidebar.min.js | 3 + .../components/sidebar/sidebar.min.js.map | 7 + shiny/www/shared/bootstrap/_version.json | 6 +- shiny/www/shared/bootstrap/bootstrap.min.css | 4 +- shiny/www/shared/bootstrap/font.css | 124 ++++++++++++++++++ ...xRpg3hIP6sJ7fM7PqPMcMnZFqUwX28DBKXhM0.woff | Bin 0 -> 56036 bytes ...xRpg3hIP6sJ7fM7PqPMcMnZFqUwX28DMyQhM0.woff | Bin 0 -> 56044 bytes ...g3hIP6sJ7fM7PqlOPHYvDP_W9O7GQTTbI1rSg.woff | Bin 0 -> 47776 bytes ...g3hIP6sJ7fM7PqlOPHYvDP_W9O7GQTTsoprSg.woff | Bin 0 -> 47984 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk0ZjaVQ.woff | Bin 0 -> 72136 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk5hkaVQ.woff | Bin 0 -> 74700 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk8ZkaVQ.woff | Bin 0 -> 74564 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk_RkaVQ.woff | Bin 0 -> 74940 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0RkxhjaVQ.woff | Bin 0 -> 74644 bytes ...cVXSCEkx2cmqvXlWq8tWZ0Pw86hd0RkyFjaVQ.woff | Bin 0 -> 71660 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgsg-1y4k.woff | Bin 0 -> 68664 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgsgH1y4k.woff | Bin 0 -> 70652 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgshZ1y4k.woff | Bin 0 -> 69392 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgsiH0C4k.woff | Bin 0 -> 70524 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgsjZ0C4k.woff | Bin 0 -> 70792 bytes ...vWbX2vVnXBbObj2OVZyOOSr4dVJWUgsjr0C4k.woff | Bin 0 -> 71144 bytes .../ionrangeslider/css/ion.rangeSlider.css | 52 ++++---- shiny/www/shared/shiny.js | 16 ++- shiny/www/shared/shiny.js.map | 4 +- shiny/www/shared/shiny.min.js | 2 +- shiny/www/shared/shiny.min.js.map | 6 +- 82 files changed, 599 insertions(+), 171 deletions(-) create mode 100644 examples/duckdb/.gitignore create mode 100644 shiny/experimental/www/bslib/components/accordion/accordion.css rename shiny/experimental/www/bslib/components/{ => accordion}/accordion.min.js (100%) rename shiny/experimental/www/bslib/components/{ => accordion}/accordion.min.js.map (99%) delete mode 100644 shiny/experimental/www/bslib/components/card.min.js delete mode 100644 shiny/experimental/www/bslib/components/card.min.js.map create mode 100644 shiny/experimental/www/bslib/components/card/card.css create mode 100644 shiny/experimental/www/bslib/components/card/card.min.js create mode 100644 shiny/experimental/www/bslib/components/card/card.min.js.map create mode 100644 shiny/experimental/www/bslib/components/grid/grid.css create mode 100644 shiny/experimental/www/bslib/components/nav_spacer/nav_spacer.css create mode 100644 shiny/experimental/www/bslib/components/page_fillable/page_fillable.css create mode 100644 shiny/experimental/www/bslib/components/page_navbar/page_navbar.css create mode 100644 shiny/experimental/www/bslib/components/page_sidebar/page_sidebar.css delete mode 100644 shiny/experimental/www/bslib/components/sidebar.min.js delete mode 100644 shiny/experimental/www/bslib/components/sidebar.min.js.map create mode 100644 shiny/experimental/www/bslib/components/sidebar/sidebar.css create mode 100644 shiny/experimental/www/bslib/components/sidebar/sidebar.min.js create mode 100644 shiny/experimental/www/bslib/components/sidebar/sidebar.min.js.map create mode 100644 shiny/experimental/www/bslib/components/value_box/value_box.css create mode 100644 shiny/experimental/www/bslib/components/webComponents/webComponents.min.js create mode 100644 shiny/experimental/www/bslib/components/webComponents/webComponents.min.js.map create mode 100644 shiny/www/shared/_x/bslib/components/nav_spacer/nav_spacer.css create mode 100644 shiny/www/shared/_x/bslib/components/page_sidebar/page_sidebar.css delete mode 100644 shiny/www/shared/_x/bslib/components/sidebar.min.js delete mode 100644 shiny/www/shared/_x/bslib/components/sidebar.min.js.map create mode 100644 shiny/www/shared/_x/bslib/components/sidebar/sidebar.css create mode 100644 shiny/www/shared/_x/bslib/components/sidebar/sidebar.min.js create mode 100644 shiny/www/shared/_x/bslib/components/sidebar/sidebar.min.js.map create mode 100644 shiny/www/shared/bootstrap/font.css create mode 100644 shiny/www/shared/bootstrap/fonts/HI_diYsKILxRpg3hIP6sJ7fM7PqPMcMnZFqUwX28DBKXhM0.woff create mode 100644 shiny/www/shared/bootstrap/fonts/HI_diYsKILxRpg3hIP6sJ7fM7PqPMcMnZFqUwX28DMyQhM0.woff create mode 100644 shiny/www/shared/bootstrap/fonts/HI_jiYsKILxRpg3hIP6sJ7fM7PqlOPHYvDP_W9O7GQTTbI1rSg.woff create mode 100644 shiny/www/shared/bootstrap/fonts/HI_jiYsKILxRpg3hIP6sJ7fM7PqlOPHYvDP_W9O7GQTTsoprSg.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk0ZjaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk5hkaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk8ZkaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0Rk_RkaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0RkxhjaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memQYaGs126MiZpBA-UFUIcVXSCEkx2cmqvXlWq8tWZ0Pw86hd0RkyFjaVQ.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgsg-1y4k.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgsgH1y4k.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgshZ1y4k.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgsiH0C4k.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgsjZ0C4k.woff create mode 100644 shiny/www/shared/bootstrap/fonts/memSYaGs126MiZpBA-UvWbX2vVnXBbObj2OVZyOOSr4dVJWUgsjr0C4k.woff diff --git a/CHANGELOG.md b/CHANGELOG.md index 0299552e1..a64c40473 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### New features * `shiny run` now takes a `--reload-dir ` argument that indicates a directory `--reload` should (recursively) monitor for changes, in addition to the app's parent directory. Can be used more than once. (#353) +* The default theme has been updated to use Bootstrap 5 with custom Shiny style enhancements. (#624) + ### Bug fixes diff --git a/e2e/controls.py b/e2e/controls.py index 8eb1e78c9..4091bf4bf 100644 --- a/e2e/controls.py +++ b/e2e/controls.py @@ -1374,10 +1374,14 @@ def __init__( def expect_tick_labels( self, - value: ListPatternOrStr, + value: ListPatternOrStr | None, *, timeout: Timeout = None, ) -> None: + if value is None: + playwright_expect(self.loc_irs_ticks).to_have_count(0) + return + playwright_expect(self.loc_irs_ticks).to_have_text(value, timeout=timeout) def expect_animate(self, exists: bool, *, timeout: Timeout = None) -> None: @@ -1553,10 +1557,10 @@ def slow_move(x: float, y: float, delay: float = sleep_time) -> None: ) def _grid_bb(self, *, timeout: Timeout = None) -> FloatRect: - grid = self.loc_container.locator(".irs-grid") + grid = self.loc_irs.locator("> .irs > .irs-line") grid_bb = grid.bounding_box(timeout=timeout) if grid_bb is None: - raise RuntimeError("Couldn't find bounding box for .irs-grid") + raise RuntimeError("Couldn't find bounding box for .irs-line") return grid_bb def _handle_center( diff --git a/e2e/inputs/input_slider/app.py b/e2e/inputs/input_slider/app.py index 5294da152..5f3626604 100644 --- a/e2e/inputs/input_slider/app.py +++ b/e2e/inputs/input_slider/app.py @@ -51,6 +51,7 @@ def slider_row( pre="$", sep=",", animate=True, + ticks=True, ), slider_row( "Looping Animation", diff --git a/e2e/inputs/input_slider/test_input_slider_app.py b/e2e/inputs/input_slider/test_input_slider_app.py index ae6c5750c..bb109e7f1 100644 --- a/e2e/inputs/input_slider/test_input_slider_app.py +++ b/e2e/inputs/input_slider/test_input_slider_app.py @@ -17,7 +17,7 @@ def test_slider_regular(page: Page, local_app: ShinyAppProc) -> None: s0.expect_min("0") s0.expect_max("1000") s0.expect_step("1") - s0.expect_ticks("true") + s0.expect_ticks("false") s0.expect_sep(",") s0.expect_pre(None) s0.expect_post(None) @@ -27,7 +27,7 @@ def test_slider_regular(page: Page, local_app: ShinyAppProc) -> None: s0.expect_animate(exists=False) OutputTextVerbatim(page, "txt0").expect_value("500") - new_val = "36" + new_val = "20" s0.set(new_val) s0.expect_value(new_val) OutputTextVerbatim(page, "txt0").expect_value(new_val) @@ -42,7 +42,7 @@ def test_slider_range(page: Page, local_app: ShinyAppProc) -> None: s1.expect_min("1") s1.expect_max("1000") s1.expect_step("1") - s1.expect_ticks("true") + s1.expect_ticks("false") s1.expect_sep(",") s1.expect_pre(None) s1.expect_post(None) @@ -52,7 +52,7 @@ def test_slider_range(page: Page, local_app: ShinyAppProc) -> None: s1.expect_animate(exists=False) OutputTextVerbatim(page, "txt1").expect_value("(200, 500)") - new_val = ("605", "885") + new_val = ("605", "840") s1.set(new_val, max_err_values=1000) try: s1.expect_value((MISSING, MISSING)) # type: ignore @@ -97,7 +97,7 @@ def test_slider_loop(page: Page, local_app: ShinyAppProc) -> None: s3.expect_min("1") s3.expect_max("2000") s3.expect_step("10") - s3.expect_ticks("true") + s3.expect_ticks("false") s3.expect_sep(",") s3.expect_pre(None) s3.expect_post(None) @@ -131,7 +131,7 @@ def test_slider_play(page: Page, local_app: ShinyAppProc) -> None: s4.expect_min("0") s4.expect_max("5") s4.expect_step("1") - s4.expect_ticks("true") + s4.expect_ticks("false") s4.expect_sep(",") s4.expect_pre(None) s4.expect_post(None) diff --git a/e2e/inputs/test_input_checkbox.py b/e2e/inputs/test_input_checkbox.py index 9561136a0..ff7efba4b 100644 --- a/e2e/inputs/test_input_checkbox.py +++ b/e2e/inputs/test_input_checkbox.py @@ -16,7 +16,7 @@ def test_input_checkbox_kitchen(page: Page, app: ShinyAppProc) -> None: somevalue.expect_checked(False) somevalue.expect_width(None) - # TODO-barret test output value + # TODO-karan test output value somevalue.set(True) @@ -28,4 +28,4 @@ def test_input_checkbox_kitchen(page: Page, app: ShinyAppProc) -> None: somevalue.toggle() somevalue.expect_checked(True) - # TODO-barret test output value + # TODO-karan test output value diff --git a/e2e/inputs/test_input_file.py b/e2e/inputs/test_input_file.py index 7e100dfbb..85b993848 100644 --- a/e2e/inputs/test_input_file.py +++ b/e2e/inputs/test_input_file.py @@ -40,4 +40,4 @@ def test_input_file_kitchen(page: Page, app: ShinyAppProc) -> None: file1.expect_complete() - # TODO-barret; Test UI output to not be empty + # TODO-karan; Test UI output to not be empty diff --git a/e2e/inputs/test_input_slider.py b/e2e/inputs/test_input_slider.py index 6180a0ee3..fdf965889 100644 --- a/e2e/inputs/test_input_slider.py +++ b/e2e/inputs/test_input_slider.py @@ -15,9 +15,7 @@ def test_input_slider_kitchen(page: Page, slider_app: ShinyAppProc) -> None: expect(obs.loc_label).to_have_text("Number of bins:") - obs.expect_tick_labels( - ["10", "19", "28", "37", "46", "55", "64", "73", "82", "91", "100"] - ) + obs.expect_tick_labels(None) obs.expect_value("30") obs.expect_animate(False) @@ -27,7 +25,7 @@ def test_input_slider_kitchen(page: Page, slider_app: ShinyAppProc) -> None: obs.expect_max("100") # obs.expect_from() obs.expect_step("1") - obs.expect_ticks("true") + obs.expect_ticks("false") obs.expect_sep(",") obs.expect_pre(None) obs.expect_post(None) diff --git a/e2e/outputs/test_output_text.py b/e2e/outputs/test_output_text.py index 09f3f63f5..72a16c434 100644 --- a/e2e/outputs/test_output_text.py +++ b/e2e/outputs/test_output_text.py @@ -13,6 +13,8 @@ def test_output_text_kitchen(page: Page, app: ShinyAppProc) -> None: verb = OutputTextVerbatim(page, "verb") verb_no_placeholder = OutputTextVerbatim(page, "verb_no_placeholder") + txt.set("") # Reset text + text.expect_value("") text.expect_inline(False) diff --git a/examples/duckdb/.gitignore b/examples/duckdb/.gitignore new file mode 100644 index 000000000..fda687dc3 --- /dev/null +++ b/examples/duckdb/.gitignore @@ -0,0 +1,3 @@ +cities.csv +weather_forecasts.csv +weather.db diff --git a/examples/penguins/app.py b/examples/penguins/app.py index e019a50cc..e7bbe5919 100644 --- a/examples/penguins/app.py +++ b/examples/penguins/app.py @@ -6,7 +6,6 @@ import pandas as pd import seaborn as sns -import shinyswatch from colors import bg_palette, palette import shiny.experimental as x @@ -44,7 +43,6 @@ ui.input_switch("by_species", "Show species", value=True), ui.input_switch("show_margins", "Show marginal plots", value=True), ), - shinyswatch.theme.pulse(), ui.output_ui("value_boxes"), x.ui.output_plot("scatter", fill=True), ) diff --git a/js/package-lock.json b/js/package-lock.json index b097bd913..9615ae8a8 100644 --- a/js/package-lock.json +++ b/js/package-lock.json @@ -6,7 +6,6 @@ "packages": { "": { "name": "shiny-dataframe-binding", - "version": "1.0.0", "license": "MIT", "dependencies": { "@tanstack/react-table": "^8.9.1" diff --git a/js/package.json b/js/package.json index 13cd817a0..6fe3cfb25 100644 --- a/js/package.json +++ b/js/package.json @@ -1,14 +1,12 @@ { "name": "shiny-dataframe-binding", - "version": "1.0.0", - "description": "", + "private": true, + "license": "MIT", "main": "index.js", "scripts": { "build": "tsc -noEmit && eslint . && tsx build.ts", "watch": "npx nodemon --exec 'npm run build' --ext '*' --ignore dist/ --ignore esbuild-metadata.json" }, - "author": "", - "license": "MIT", "devDependencies": { "@preact/compat": "^17.1.2", "@tanstack/react-virtual": "^3.0.0-beta.54", diff --git a/scripts/htmlDependencies.R b/scripts/htmlDependencies.R index 78af77030..512a659ef 100755 --- a/scripts/htmlDependencies.R +++ b/scripts/htmlDependencies.R @@ -1,10 +1,18 @@ #!/usr/bin/env Rscript +message("Checking for node / npm") +if (Sys.which("npm")[["npm"]] == "") { + stop("Please install node / npm before running script") +} + versions <- list() # Use local lib path for installing packages so we don't pollute the user's library +message("Installing GitHub packages: bslib, shiny, htmltools") withr::local_temp_libpaths() -pak::pkg_install(c("rstudio/bslib@main", "rstudio/shiny@main", "rstudio/htmltools@main")) +ignore <- capture.output({ + pak::pkg_install(c("rstudio/bslib@main", "rstudio/shiny@main", "rstudio/htmltools@main")) +}) # pak::pkg_install(c("cran::bslib", "cran::shiny", "cran::htmltools")) versions["shiny_html_deps"] <- as.character(packageVersion("shiny")) @@ -33,8 +41,8 @@ bslib_version <- pkg_source_version("bslib") shiny_version <- pkg_source_version("shiny") htmltools_version <- pkg_source_version("htmltools") -library(htmltools) -library(bslib) +library(htmltools, quietly = TRUE, warn.conflicts = FALSE) +library(bslib, quietly = TRUE, warn.conflicts = FALSE) shiny_path <- fs::path(getwd(), "shiny") www <- fs::path(shiny_path, "www") @@ -77,24 +85,36 @@ copy_from_pkg <- function(pkg_name, pkg_dir, local_dir, version_dir = fs::path_d } +# ------------------------------------------------------------------------------ +message("Copy bslib components") # Copy over bslib's components directory -copy_from_pkg("bslib", "components", x_www_bslib_components) -# Remove unused Sass files +copy_from_pkg("bslib", "components/dist", x_www_bslib_components) +# Remove non-minified files fs::file_delete( - fs::dir_ls(x_www_bslib_components, type = "file", regexp = "\\.scss$") + fs::dir_ls( + x_www_bslib_components, + type = "file", + recurse = TRUE, + regexp = "\\.(min\\.|css)", + invert = TRUE + ) ) -# Remove unused tag require -fs::file_delete(fs::path(x_www_bslib_components, "tag-require.js")) + +# ------------------------------------------------------------------------------ +message("Copy htmltools - fill") # Copy over htmltools's fill directory copy_from_pkg("htmltools", "fill", fs::path(x_www, "htmltools", "fill")) - - +# ------------------------------------------------------------------------------ +message("Copy shiny www/shared") # Copy over shiny's www/shared directory copy_from_pkg("shiny", "www/shared", www_shared, www_shared) + +# ------------------------------------------------------------------------------ +message("Cleanup shiny www/shared") # Don't need legacy (hopefully) fs::dir_delete(fs::path(www_shared, "legacy")) # Don't need dataTables (hopefully) @@ -105,8 +125,40 @@ fs::file_delete( fs::dir_ls(www_shared, type = "file", regexp = "jquery") ) + +# ------------------------------------------------------------------------------ +message("Save ionRangeSlider dep") + # Upgrade to Bootstrap 5 by default -deps <- bs_theme_dependencies(bs_theme(version = 5)) +shiny_theme <- bslib::bs_theme(version = 5, preset = "shiny") +# Save iorange slider dep +# Get _dynamic_ ionrangeslider dep +ion_dep <- shiny:::ionRangeSliderDependencyCSS(shiny_theme) +if (inherits(ion_dep, "html_dependency")) { + ion_dep <- list(ion_dep) +} +# Save to temp folder +temp_ion_dep_dir <- fs::path_temp("shiny-ion-range-slider") +fs::dir_create(temp_ion_dep_dir) +withr::with_options( + list(htmltools.dir.version = FALSE), + ignore <- lapply(ion_dep, htmltools::copyDependencyToDir, temp_ion_dep_dir) +) +# Overwrite css file +ion_dep_dir <- fs::path(www_shared, "ionrangeslider") +fs::file_move( + fs::path(temp_ion_dep_dir, "ionRangeSlider", "ionRangeSlider.css"), + fs::path(ion_dep_dir, "css", "ion.rangeSlider.css") +) +# Cleanup +fs::dir_delete(temp_ion_dep_dir) + +# TODO - make a UI object and save its dependencies. Loop through the different UI elements of bslib and save them accordingly. Similar to shiny::input_slider? instead of reaching into the `:::` + + +message("Save bootstrap bundle") +# Save htmldeps +deps <- bslib::bs_theme_dependencies(shiny_theme) withr::with_options( list(htmltools.dir.version = FALSE), ignore <- lapply(deps, copyDependencyToDir, www_shared) @@ -123,6 +175,19 @@ write_json( ) ) +message("Reduce font files") +font_txt <- unlist(strsplit(readLines("shiny/www/shared/bootstrap/font.css"), ";")) +woff_files <- list.files("shiny/www/shared/bootstrap/fonts", pattern = "\\.woff", full.names = TRUE) +ignored <- lapply(woff_files, function(woff_file) { + file_name <- basename(woff_file) + if (!any(grepl(file_name, font_txt, fixed = TRUE))) { + unlink(woff_file) + } +}) + + +# ------------------------------------------------------------------------------ +message("Cleanup bootstrap bundle") # This additional bs3compat HTMLDependency() only holds # the JS shim for tab panel logic, which we don't need # since we're generating BS5+ tab markup. Note, however, @@ -130,13 +195,17 @@ write_json( # comes in via the bootstrap HTMLDependency() fs::dir_delete(fs::path(www_shared, "bs3compat")) + +# ------------------------------------------------------------------------------ +message("Save requirejs") requirejs_version <- "2.3.6" versions["requirejs"] <- requirejs_version requirejs <- fs::path(www_shared, "requirejs") fs::dir_create(requirejs) download.file( paste0("https://cdnjs.cloudflare.com/ajax/libs/require.js/", requirejs_version, "/require.min.js"), - fs::path(requirejs, "require.min.js") + fs::path(requirejs, "require.min.js"), + quiet = TRUE ) shims <- fs::path(getwd(), "scripts", "define-shims.js") @@ -149,6 +218,8 @@ cat( ) +# ------------------------------------------------------------------------------ +message("Save _versions.py") version_vars <- paste0(names(versions), " = ", "\"", versions, "\"\n", collapse = "") version_all <- paste0( collapse = "", @@ -165,9 +236,8 @@ cat( ) - # ------------------------------------------------------------------------------ -# Copy x assets to shiny main_x assets +message("Copy www/shared/_x assets") if (fs::dir_exists(main_x_www)) fs::dir_delete(main_x_www) @@ -183,7 +253,17 @@ fs::dir_copy(x_www_htmltools_fill, main_x_htmltools_fill) fs::file_delete( fs::dir_ls( fs::path(main_x_bslib_components, "components"), - regexp="(_version|sidebar)", + regexp="(_version|sidebar|nav_spacer)", invert = TRUE ) ) + + +# ------------------------------------------------------------------------------ +message("Create dataframe.js via npm") + +js_path <- fs::path_abs(fs::path(shiny_path, "..", "js")) +ignore <- system( + paste0("cd ", js_path, " && npm install && npm run build"), + intern = TRUE +) diff --git a/shiny/__init__.py b/shiny/__init__.py index 48747ab82..d5698b83b 100644 --- a/shiny/__init__.py +++ b/shiny/__init__.py @@ -1,6 +1,6 @@ """A package for building reactive web applications.""" -__version__ = "0.4.0.9000" +__version__ = "0.4.0.9001" from ._shinyenv import is_pyodide as _is_pyodide diff --git a/shiny/api-examples/output_text/app.py b/shiny/api-examples/output_text/app.py index 9dba76c20..f41594267 100644 --- a/shiny/api-examples/output_text/app.py +++ b/shiny/api-examples/output_text/app.py @@ -1,14 +1,22 @@ from shiny import App, Inputs, Outputs, Session, render, ui app_ui = ui.page_fluid( - ui.input_text("txt", "Enter the text to display below:"), + ui.input_text("txt", "Enter the text to display below:", "delete me"), ui.row( - ui.column(6, ui.output_text("text")), - ui.column(6, ui.output_text_verbatim("verb", placeholder=True)), + ui.column(6, ui.code("ui.output_text()"), ui.output_text("text")), + ui.column( + 6, + ui.code("ui.output_text_verbatim(placeholder=True)"), + ui.output_text_verbatim("verb", placeholder=True), + ), ), ui.row( ui.column(6), - ui.column(6, ui.output_text_verbatim("verb_no_placeholder", placeholder=False)), + ui.column( + 6, + ui.code("ui.output_text_verbatim(placeholder=False)"), + ui.output_text_verbatim("verb_no_placeholder", placeholder=False), + ), ), ) diff --git a/shiny/experimental/e2e/navbar/app.py b/shiny/experimental/e2e/navbar/app.py index 856998b88..a9cf2d9e1 100644 --- a/shiny/experimental/e2e/navbar/app.py +++ b/shiny/experimental/e2e/navbar/app.py @@ -46,7 +46,6 @@ def nav_items(prefix: str) -> list[NavSetArg]: sidebar=my_sidebar, title="page_navbar()", bg="#0062cc", - inverse=True, header=ui.markdown( "Testing app for `bslib::nav_spacer()` and `bslib::nav_item()` [#319](https://github.com/rstudio/bslib/pull/319)." ), diff --git a/shiny/experimental/ui/_fill.py b/shiny/experimental/ui/_fill.py index 8792c6e02..6f13b92b0 100644 --- a/shiny/experimental/ui/_fill.py +++ b/shiny/experimental/ui/_fill.py @@ -476,7 +476,7 @@ def _is_fill_layout( # tag: Tagifiable and not (Tag or FillingLayout) raise TypeError( - f"`_is_fill_layout(tag=)` must be a `Tag` or implement the `FillingLayout` protocol methods TODO-barret expand on method names. Received object of type: `{type(tag).__name__}`" + f"`_is_fill_layout(tag=)` must be a `Tag` or implement the `FillingLayout` protocol methods. Received object of type: `{type(tag).__name__}`" ) diff --git a/shiny/experimental/ui/_htmldeps.py b/shiny/experimental/ui/_htmldeps.py index c5b309118..da4541fed 100644 --- a/shiny/experimental/ui/_htmldeps.py +++ b/shiny/experimental/ui/_htmldeps.py @@ -10,56 +10,97 @@ _x_www = PurePath(__file__).parent.parent / "www" _x_www_path = str(_x_www) -_x_components_path = str(_x_www / "bslib" / "components") -_x_fill_path = str(_x_www / "htmltools" / "fill") +_x_htmltools_path = _x_www / "htmltools" +_x_components_path = _x_www / "bslib" / "components" -def card_dependency() -> HTMLDependency: +def _htmltools_dep( + name: str, + script: bool = False, + stylesheet: bool = False, + all_files: bool = True, +) -> HTMLDependency: return HTMLDependency( - name="bslib-card", - version=bslib_version, + name=f"htmltools-{name}", + version=htmltools_version, source={ "package": "shiny", - "subdir": _x_components_path, + "subdir": str(_x_htmltools_path / "fill"), }, - script={"src": "card.min.js"}, + script={"src": f"{name}.min.js"} if script else None, + stylesheet={"href": f"{name}.css"} if stylesheet else None, + all_files=all_files, ) -def fill_dependency() -> HTMLDependency: +def _bslib_component_dep( + name: str, + script: bool = False, + stylesheet: bool = False, + all_files: bool = True, +) -> HTMLDependency: return HTMLDependency( - "htmltools-fill", - htmltools_version, + name=f"bslib-{name}", + version=bslib_version, source={ "package": "shiny", - "subdir": _x_fill_path, + "subdir": str(_x_components_path / name), }, - stylesheet={"href": "fill.css"}, + script={"src": f"{name}.min.js"} if script else None, + stylesheet={"href": f"{name}.css"} if stylesheet else None, + all_files=all_files, ) -def sidebar_dependency() -> HTMLDependency: - return HTMLDependency( - "bslib-sidebar", - bslib_version, - source={ - "package": "shiny", - "subdir": _x_components_path, - }, - script={"src": "sidebar.min.js"}, - ) +# -- htmltools --------------------- + + +def fill_dependency() -> HTMLDependency: + return _htmltools_dep("fill", stylesheet=True) + + +# -- bslib ------------------------- def accordion_dependency() -> HTMLDependency: - return HTMLDependency( - "bslib-accordion", - version=bslib_version, - source={ - "package": "shiny", - "subdir": _x_components_path, - }, - script={"src": "accordion.min.js"}, - ) + return _bslib_component_dep("accordion", script=True, stylesheet=True) + + +def card_dependency() -> HTMLDependency: + return _bslib_component_dep("card", script=True, stylesheet=True) + + +def grid_dependency() -> HTMLDependency: + return _bslib_component_dep("grid", stylesheet=True) + + +# Coped to `ui._x._htmldeps.py` +# def nav_spacer_dependency() -> HTMLDependency: +# return _bslib_component_dep("nav_spacer", stylesheet=True) + + +def page_fillable_dependency() -> HTMLDependency: + return _bslib_component_dep("page_fillable", stylesheet=True) + + +# # Not used! +# def page_navbar_dependency() -> HTMLDependency: +# return _bslib_component_dep("page_navbar", stylesheet=True) + + +def page_sidebar_dependency() -> HTMLDependency: + return _bslib_component_dep("page_sidebar", stylesheet=True) + + +def sidebar_dependency() -> HTMLDependency: + return _bslib_component_dep("sidebar", script=True, stylesheet=True) + + +def value_box_dependency() -> HTMLDependency: + return _bslib_component_dep("value_box", stylesheet=True) + + +# -- Experimental ------------------ def autoresize_dependency(): diff --git a/shiny/experimental/ui/_layout.py b/shiny/experimental/ui/_layout.py index 7c5c2dfb7..9f878ba01 100644 --- a/shiny/experimental/ui/_layout.py +++ b/shiny/experimental/ui/_layout.py @@ -6,6 +6,7 @@ from ._css_unit import CssUnit, as_css_unit from ._fill import as_fill_item, as_fillable_container +from ._htmldeps import grid_dependency from ._utils import consolidate_attrs, is_01_scalar @@ -119,6 +120,7 @@ def layout_column_wrap( }, attrs, *upgraded_children, + grid_dependency(), ) if fill: tag = as_fill_item(tag) diff --git a/shiny/experimental/ui/_navs.py b/shiny/experimental/ui/_navs.py index 38c487b68..5f9f169cd 100644 --- a/shiny/experimental/ui/_navs.py +++ b/shiny/experimental/ui/_navs.py @@ -160,7 +160,7 @@ def layout(self, nav: Tag, content: Tag) -> Tag: content_val = navset_card_body(content, sidebar=self.sidebar) if self.placement == "below": - # TODO-barret; have carson double check this change + # TODO-carson; have carson double check this change return card( card_header(self.header) if self.header else None, content_val, @@ -170,7 +170,7 @@ def layout(self, nav: Tag, content: Tag) -> Tag: card_footer(nav), ) else: - # TODO-barret; have carson double check this change + # TODO-carson; have carson double check this change return card( card_header(nav), card_body(self.header, fill=False, fillable=False) @@ -520,9 +520,7 @@ def navset_bar( inverse Either ``True`` for a light text color or ``False`` for a dark text color. collapsible - ``True`` to automatically collapse the navigation elements into a menu when the - width of the browser is less than 940 pixels (useful for viewing on smaller - touchscreen device) + ``True`` to automatically collapse the navigation elements into an expandable menu on mobile devices or narrow window widths. fluid ``True`` to use fluid layout; ``False`` to use fixed layout. diff --git a/shiny/experimental/ui/_page.py b/shiny/experimental/ui/_page.py index d125f1619..c786896d8 100644 --- a/shiny/experimental/ui/_page.py +++ b/shiny/experimental/ui/_page.py @@ -18,6 +18,7 @@ from ...ui._utils import get_window_title from ._css_unit import CssUnit, as_css_padding, as_css_unit from ._fill import as_fillable_container +from ._htmldeps import page_fillable_dependency, page_sidebar_dependency from ._navs import navset_bar from ._sidebar import Sidebar, layout_sidebar from ._utils import consolidate_attrs @@ -68,17 +69,20 @@ def page_sidebar( if isinstance(title, str): title = tags.h1(title, class_="bslib-page-title") + attrs, children = consolidate_attrs(*args, **kwargs) + return page_fillable( title, layout_sidebar( sidebar, - *args, + *children, + attrs, fillable=fillable, border=False, border_radius=False, - **kwargs, ), get_window_title(title, window_title=window_title), + page_sidebar_dependency(), padding=0, gap=0, lang=lang, @@ -102,7 +106,7 @@ def page_navbar( header: Optional[TagChild] = None, footer: Optional[TagChild] = None, bg: Optional[str] = None, - inverse: bool = False, + inverse: bool = True, collapsible: bool = True, fluid: bool = True, window_title: str | MISSING_TYPE = MISSING, @@ -147,9 +151,7 @@ def page_navbar( inverse Either ``True`` for a light text color or ``False`` for a dark text color. collapsible - ``True`` to automatically collapse the navigation elements into a menu when the - width of the browser is less than 940 pixels (useful for viewing on smaller - touchscreen device) + ``True`` to automatically collapse the elements into an expandable menu on mobile devices or narrow window widths. fluid ``True`` to use fluid layout; ``False`` to use fixed layout. window_title @@ -284,6 +286,7 @@ def page_fillable( *children, ) ), + page_fillable_dependency(), title=title, lang=lang, ) diff --git a/shiny/experimental/ui/_valuebox.py b/shiny/experimental/ui/_valuebox.py index 7a32d9593..b5abeb1d8 100644 --- a/shiny/experimental/ui/_valuebox.py +++ b/shiny/experimental/ui/_valuebox.py @@ -7,6 +7,7 @@ from ._card import CardItem, card, card_body from ._css_unit import CssUnit, as_css_unit, as_width_unit from ._fill import as_fill_carrier +from ._htmldeps import value_box_dependency from ._layout import layout_column_wrap from ._utils import consolidate_attrs, is_01_scalar @@ -112,6 +113,7 @@ def value_box( return card( contents, attrs, + value_box_dependency(), full_screen=full_screen, height=height, max_height=max_height, @@ -229,6 +231,8 @@ def _layout(showcase: TagChild | TagAttrs, contents: Tag) -> CardItem: items = reversed(items) width_fs = reversed(width_fs) + width_fs = " ".join(width_fs) + layout_css_args = { "--bslib-value-box-widths": width_css_unit, "--bslib-value-box-widths-full-screen": width_fs, diff --git a/shiny/experimental/www/bslib/_version.json b/shiny/experimental/www/bslib/_version.json index 8b406d77f..b5cdc163e 100644 --- a/shiny/experimental/www/bslib/_version.json +++ b/shiny/experimental/www/bslib/_version.json @@ -1,5 +1,5 @@ { "note!": "This file is auto-generated by scripts/htmlDependencies.R", "package": "bslib", - "version": "Github (rstudio/bslib@70e602be54ab9850de9db43ee1a34376a99bbc59)" + "version": "Github (rstudio/bslib@175ad4624b9a1ede4130ebb92042f8da12eb7d70)" } diff --git a/shiny/experimental/www/bslib/components/accordion/accordion.css b/shiny/experimental/www/bslib/components/accordion/accordion.css new file mode 100644 index 000000000..37be35021 --- /dev/null +++ b/shiny/experimental/www/bslib/components/accordion/accordion.css @@ -0,0 +1 @@ +.accordion .accordion-header{font-size:calc(1.325rem + .9vw);margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2;color:var(--bs-heading-color);margin-bottom:0}@media (min-width: 1200px){.accordion .accordion-header{font-size:2rem}}.accordion .accordion-icon:not(:empty){margin-right:0.25rem;display:flex}.accordion .accordion-button:not(.collapsed){box-shadow:none}.accordion .accordion-button:not(.collapsed):focus{box-shadow:var(--bs-accordion-btn-focus-box-shadow)} diff --git a/shiny/experimental/www/bslib/components/accordion.min.js b/shiny/experimental/www/bslib/components/accordion/accordion.min.js similarity index 100% rename from shiny/experimental/www/bslib/components/accordion.min.js rename to shiny/experimental/www/bslib/components/accordion/accordion.min.js diff --git a/shiny/experimental/www/bslib/components/accordion.min.js.map b/shiny/experimental/www/bslib/components/accordion/accordion.min.js.map similarity index 99% rename from shiny/experimental/www/bslib/components/accordion.min.js.map rename to shiny/experimental/www/bslib/components/accordion/accordion.min.js.map index 2e2b17409..a6a79225c 100644 --- a/shiny/experimental/www/bslib/components/accordion.min.js.map +++ b/shiny/experimental/www/bslib/components/accordion/accordion.min.js.map @@ -1,6 +1,6 @@ { "version": 3, - "sources": ["../../srcts/src/components/_utils.ts", "../../srcts/src/components/accordion.ts"], + "sources": ["../../../../srcts/src/components/_utils.ts", "../../../../srcts/src/components/accordion.ts"], "sourcesContent": ["import type { HtmlDep } from \"rstudio-shiny/srcts/types/src/shiny/render\";\n\nimport type { InputBinding as InputBindingType } from \"rstudio-shiny/srcts/types/src/bindings/input\";\n\n// Exclude undefined from T\ntype NotUndefined = T extends undefined ? never : T;\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst InputBinding = (\n window.Shiny ? Shiny.InputBinding : class {}\n) as typeof InputBindingType;\n\nfunction registerBinding(\n inputBindingClass: new () => InputBindingType,\n name: string\n): void {\n if (window.Shiny) {\n Shiny.inputBindings.register(new inputBindingClass(), \"bslib.\" + name);\n }\n}\n\n// Return true if the key exists on the object and the value is not undefined.\n//\n// This method is mainly used in input bindings' `receiveMessage` method.\n// Since we know that the values are sent by Shiny via `{jsonlite}`,\n// then we know that there are no `undefined` values. `null` is possible, but not `undefined`.\nfunction hasDefinedProperty<\n Prop extends keyof X,\n X extends { [key: string]: any }\n>(\n obj: X,\n prop: Prop\n): obj is X & { [key in NonNullable]: NotUndefined } {\n return (\n Object.prototype.hasOwnProperty.call(obj, prop) && obj[prop] !== undefined\n );\n}\n\n// TODO: Shiny should trigger resize events when the output\n// https://github.com/rstudio/shiny/pull/3682\nfunction doWindowResizeOnElementResize(el: HTMLElement): void {\n if ($(el).data(\"window-resize-observer\")) {\n return;\n }\n const resizeEvent = new Event(\"resize\");\n const ro = new ResizeObserver(() => {\n window.dispatchEvent(resizeEvent);\n });\n ro.observe(el);\n $(el).data(\"window-resize-observer\", ro);\n}\n\nfunction getAllFocusableChildren(el: HTMLElement): HTMLElement[] {\n // Cross-referenced with https://allyjs.io/data-tables/focusable.html\n const base = [\n \"a[href]\",\n \"area[href]\",\n \"button\",\n \"details summary\",\n \"input\",\n \"iframe\",\n \"select\",\n \"textarea\",\n '[contentEditable=\"\"]',\n '[contentEditable=\"true\"]',\n '[contentEditable=\"TRUE\"]',\n \"[tabindex]\",\n ];\n const modifiers = [':not([tabindex=\"-1\"])', \":not([disabled])\"];\n const selectors = base.map((b) => b + modifiers.join(\"\"));\n const focusable = el.querySelectorAll(selectors.join(\", \"));\n return Array.from(focusable) as HTMLElement[];\n}\n\nexport {\n InputBinding,\n registerBinding,\n hasDefinedProperty,\n doWindowResizeOnElementResize,\n getAllFocusableChildren,\n};\nexport type { HtmlDep };\n", "import type { HtmlDep } from \"./_utils\";\nimport { InputBinding, registerBinding, hasDefinedProperty } from \"./_utils\";\n\ntype AccordionItem = {\n item: HTMLElement;\n value: string;\n isOpen: () => boolean;\n show: () => void;\n hide: () => void;\n};\n\ntype HTMLContent = {\n html: string;\n deps?: HtmlDep[];\n};\n\ntype SetMessage = {\n method: \"set\";\n values: string[];\n};\n\ntype OpenMessage = {\n method: \"open\";\n values: string[] | true;\n};\n\ntype CloseMessage = {\n method: \"close\";\n values: string[] | true;\n};\n\ntype InsertMessage = {\n method: \"insert\";\n panel: HTMLContent;\n target: string;\n position: \"after\" | \"before\";\n};\n\ntype RemoveMessage = {\n method: \"remove\";\n target: string[];\n};\n\ntype UpdateMessage = {\n method: \"update\";\n target: string;\n value: string;\n body: HTMLContent;\n title: HTMLContent;\n icon: HTMLContent;\n};\n\ntype MessageData =\n | CloseMessage\n | InsertMessage\n | OpenMessage\n | RemoveMessage\n | SetMessage\n | UpdateMessage;\n\nclass AccordionInputBinding extends InputBinding {\n find(scope: HTMLElement) {\n return $(scope).find(\".accordion.bslib-accordion-input\");\n }\n\n getValue(el: HTMLElement): string[] | null {\n const items = this._getItemInfo(el);\n const selected = items.filter((x) => x.isOpen()).map((x) => x.value);\n return selected.length === 0 ? null : selected;\n }\n\n subscribe(el: HTMLElement, callback: (x: boolean) => void) {\n $(el).on(\n \"shown.bs.collapse.accordionInputBinding hidden.bs.collapse.accordionInputBinding\",\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n function (event) {\n callback(true);\n }\n );\n }\n\n unsubscribe(el: HTMLElement) {\n $(el).off(\".accordionInputBinding\");\n }\n\n receiveMessage(el: HTMLElement, data: MessageData) {\n const method = data.method;\n if (method === \"set\") {\n this._setItems(el, data);\n } else if (method === \"open\") {\n this._openItems(el, data);\n } else if (method === \"close\") {\n this._closeItems(el, data);\n } else if (method === \"remove\") {\n this._removeItem(el, data);\n } else if (method === \"insert\") {\n this._insertItem(el, data);\n } else if (method === \"update\") {\n this._updateItem(el, data);\n } else {\n throw new Error(`Method not yet implemented: ${method}`);\n }\n }\n\n protected _setItems(el: HTMLElement, data: SetMessage) {\n const items = this._getItemInfo(el);\n const vals = this._getValues(el, items, data.values);\n items.forEach((x) => {\n vals.indexOf(x.value) > -1 ? x.show() : x.hide();\n });\n }\n\n protected _openItems(el: HTMLElement, data: OpenMessage) {\n const items = this._getItemInfo(el);\n const vals = this._getValues(el, items, data.values);\n items.forEach((x) => {\n if (vals.indexOf(x.value) > -1) x.show();\n });\n }\n\n protected _closeItems(el: HTMLElement, data: CloseMessage) {\n const items = this._getItemInfo(el);\n const vals = this._getValues(el, items, data.values);\n items.forEach((x) => {\n if (vals.indexOf(x.value) > -1) x.hide();\n });\n }\n\n protected _insertItem(el: HTMLElement, data: InsertMessage) {\n let targetItem = this._findItem(el, data.target);\n\n // If no target was specified, or the target was not found, then default\n // to the first or last item, depending on the position\n if (!targetItem) {\n targetItem = (\n data.position === \"before\" ? el.firstElementChild : el.lastElementChild\n ) as HTMLElement;\n }\n\n const panel = data.panel;\n\n // If there is still no targetItem, then there are no items in the accordion\n if (targetItem) {\n Shiny.renderContent(\n targetItem,\n panel,\n data.position === \"before\" ? \"beforeBegin\" : \"afterEnd\"\n );\n } else {\n Shiny.renderContent(el, panel);\n }\n\n // Need to add a reference to the parent id that makes autoclose to work\n if (this._isAutoClosing(el)) {\n const val = $(panel.html).attr(\"data-value\");\n $(el)\n .find(`[data-value=\"${val}\"] .accordion-collapse`)\n .attr(\"data-bs-parent\", \"#\" + el.id);\n }\n }\n\n protected _removeItem(el: HTMLElement, data: RemoveMessage) {\n const targetItems = this._getItemInfo(el).filter(\n (x) => data.target.indexOf(x.value) > -1\n );\n\n const unbindAll = Shiny?.unbindAll;\n\n targetItems.forEach((x) => {\n if (unbindAll) unbindAll(x.item);\n x.item.remove();\n });\n }\n\n protected _updateItem(el: HTMLElement, data: UpdateMessage) {\n const target = this._findItem(el, data.target);\n\n if (!target) {\n throw new Error(\n `Unable to find an accordion_panel() with a value of ${data.target}`\n );\n }\n\n if (hasDefinedProperty(data, \"value\")) {\n target.dataset.value = data.value;\n }\n\n if (hasDefinedProperty(data, \"body\")) {\n const body = target.querySelector(\".accordion-body\") as HTMLElement; // always exists\n Shiny.renderContent(body, data.body);\n }\n\n const header = target.querySelector(\".accordion-header\") as HTMLElement; // always exists\n\n if (hasDefinedProperty(data, \"title\")) {\n const title = header.querySelector(\".accordion-title\") as HTMLElement; // always exists\n Shiny.renderContent(title, data.title);\n }\n\n if (hasDefinedProperty(data, \"icon\")) {\n const icon = header.querySelector(\n \".accordion-button > .accordion-icon\"\n ) as HTMLElement; // always exists\n Shiny.renderContent(icon, data.icon);\n }\n }\n\n protected _getItemInfo(el: HTMLElement): AccordionItem[] {\n const items = Array.from(\n el.querySelectorAll(\":scope > .accordion-item\")\n ) as HTMLElement[];\n return items.map((x) => this._getSingleItemInfo(x));\n }\n\n protected _getSingleItemInfo(x: HTMLElement): AccordionItem {\n const collapse = x.querySelector(\".accordion-collapse\") as HTMLElement;\n const isOpen = () => $(collapse).hasClass(\"show\");\n return {\n item: x,\n value: x.dataset.value as string,\n isOpen: isOpen,\n show: () => {\n if (!isOpen()) $(collapse).collapse(\"show\");\n },\n hide: () => {\n if (isOpen()) $(collapse).collapse(\"hide\");\n },\n };\n }\n\n protected _getValues(\n el: HTMLElement,\n items: AccordionItem[],\n values: string[] | true\n ): string[] {\n let vals = values !== true ? values : items.map((x) => x.value);\n const autoclose = this._isAutoClosing(el);\n if (autoclose) {\n vals = vals.slice(vals.length - 1, vals.length);\n }\n return vals;\n }\n\n protected _findItem(el: HTMLElement, value: string): HTMLElement | null {\n return el.querySelector(`[data-value=\"${value}\"]`);\n }\n\n protected _isAutoClosing(el: HTMLElement): boolean {\n return el.classList.contains(\"autoclose\");\n }\n}\n\nregisterBinding(AccordionInputBinding, \"accordion\");\n"], "mappings": "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", "names": ["InputBinding", "registerBinding", "inputBindingClass", "name", "hasDefinedProperty", "obj", "prop", "AccordionInputBinding", "InputBinding", "scope", "el", "selected", "x", "callback", "event", "data", "method", "items", "vals", "targetItem", "panel", "val", "targetItems", "unbindAll", "target", "hasDefinedProperty", "body", "header", "title", "icon", "collapse", "isOpen", "values", "value", "registerBinding"] diff --git a/shiny/experimental/www/bslib/components/card.min.js b/shiny/experimental/www/bslib/components/card.min.js deleted file mode 100644 index 28f26fa3e..000000000 --- a/shiny/experimental/www/bslib/components/card.min.js +++ /dev/null @@ -1,3 +0,0 @@ -/*! bslib 0.5.0.9000 | (c) 2012-2023 RStudio, PBC. | License: MIT + file LICENSE */ -"use strict";(()=>{var b=window.Shiny?Shiny.InputBinding:class{};function p(u){let e=["a[href]","area[href]","button","details summary","input","iframe","select","textarea",'[contentEditable=""]','[contentEditable="true"]','[contentEditable="TRUE"]',"[tabindex]"],t=[':not([tabindex="-1"])',":not([disabled])"],r=e.map(n=>n+t.join("")),s=u.querySelectorAll(r.join(", "));return Array.from(s)}var d=class{constructor(){this.resizeObserverEntries=[],this.resizeObserver=new ResizeObserver(e=>{let t=new Event("resize");if(window.dispatchEvent(t),!window.Shiny)return;let r=[];for(let s of e)s.target instanceof HTMLElement&&s.target.querySelector(".shiny-bound-output")&&s.target.querySelectorAll(".shiny-bound-output").forEach(n=>{if(r.includes(n))return;let{binding:a,onResize:h}=$(n).data("shinyOutputBinding");if(!a||!a.resize)return;let c=n.shinyResizeObserver;if(c&&c!==this||(c||(n.shinyResizeObserver=this),h(n),r.push(n),!n.classList.contains("shiny-plot-output")))return;let l=n.querySelector('img:not([width="100%"])');l&&l.setAttribute("width","100%")})})}observe(e){this.resizeObserver.observe(e),this.resizeObserverEntries.push(e)}unobserve(e){let t=this.resizeObserverEntries.indexOf(e);t<0||(this.resizeObserver.unobserve(e),this.resizeObserverEntries.splice(t,1))}flush(){this.resizeObserverEntries.forEach(e=>{document.body.contains(e)||this.unobserve(e)})}};var E=window.bootstrap?window.bootstrap.Tooltip:class{},i=class{constructor(e){var t;e.removeAttribute(i.attr.ATTR_INIT),(t=e.querySelector(`script[${i.attr.ATTR_INIT}]`))==null||t.remove(),this.card=e,i.instanceMap.set(e,this),i.shinyResizeObserver.observe(this.card),this._addEventListeners(),this._enableTooltips(),this.overlay=this._createOverlay(),this._exitFullScreenOnEscape=this._exitFullScreenOnEscape.bind(this),this._trapFocusExit=this._trapFocusExit.bind(this)}enterFullScreen(e){e&&e.preventDefault(),document.addEventListener("keydown",this._exitFullScreenOnEscape,!1),document.addEventListener("keydown",this._trapFocusExit,!0),this.card.contains(document.activeElement)||(this.card.setAttribute("tabindex","-1"),this.card.focus()),this.card.classList.add(i.attr.CLASS_FULL_SCREEN),document.body.classList.add(i.attr.CLASS_HAS_FULL_SCREEN),this.card.insertAdjacentElement("beforebegin",this.overlay.container)}exitFullScreen(){document.removeEventListener("keydown",this._exitFullScreenOnEscape,!1),document.removeEventListener("keydown",this._trapFocusExit,!0),this.overlay.container.remove(),this.card.classList.remove(i.attr.CLASS_FULL_SCREEN),this.card.removeAttribute("tabindex"),document.body.classList.remove(i.attr.CLASS_HAS_FULL_SCREEN)}_addEventListeners(){let e=this.card.querySelector(`:scope > .${i.attr.CLASS_FULL_SCREEN_ENTER}`);e&&e.addEventListener("click",t=>this.enterFullScreen(t))}_enableTooltips(){let e=`.${i.attr.CLASS_FULL_SCREEN_ENTER}[data-bs-toggle='tooltip']`;if(!this.card.querySelector(e))return;this.card.querySelectorAll(e).forEach(r=>new E(r))}_exitFullScreenOnEscape(e){if(!(e.target instanceof HTMLElement))return;let t=["select[open]","input[aria-expanded='true']"];e.target.matches(t.join(", "))||e.key==="Escape"&&this.exitFullScreen()}_trapFocusExit(e){if(!(e instanceof KeyboardEvent)||e.key!=="Tab")return;let t=e.target===this.card,r=e.target===this.overlay.anchor,s=this.card.contains(e.target),n=()=>{e.preventDefault(),e.stopImmediatePropagation()};if(!(s||t||r)){n(),this.card.focus();return}let a=p(this.card);if(!(a.length>0)){n(),this.overlay.anchor.focus();return}if(t)return;let c=a[a.length-1],l=e.target===c;if(r&&e.shiftKey){n(),c.focus();return}if(l&&!e.shiftKey){n(),this.overlay.anchor.focus();return}}_createOverlay(){let e=document.createElement("div");e.id=i.attr.ID_FULL_SCREEN_OVERLAY,e.onclick=this.exitFullScreen.bind(this);let t=this._createOverlayCloseAnchor();return e.appendChild(t),{container:e,anchor:t}}_createOverlayCloseAnchor(){let e=document.createElement("a");return e.classList.add(i.attr.CLASS_FULL_SCREEN_EXIT),e.tabIndex=0,e.onclick=()=>this.exitFullScreen(),e.onkeydown=t=>{(t.key==="Enter"||t.key===" ")&&this.exitFullScreen()},e.innerHTML=this._overlayCloseHtml(),e}_overlayCloseHtml(){return"Close "}static getInstance(e){return i.instanceMap.get(e)}static initializeAllCards(e=!0){if(document.readyState==="loading"){i.onReadyScheduled||(i.onReadyScheduled=!0,document.addEventListener("DOMContentLoaded",()=>{i.initializeAllCards(!1)}));return}e&&i.shinyResizeObserver.flush();let t=`.${i.attr.CLASS_CARD}[${i.attr.ATTR_INIT}]`;if(!document.querySelector(t))return;document.querySelectorAll(t).forEach(s=>new i(s))}},o=i;o.attr={ATTR_INIT:"data-bslib-card-init",CLASS_CARD:"bslib-card",CLASS_FULL_SCREEN:"bslib-full-screen",CLASS_HAS_FULL_SCREEN:"bslib-has-full-screen",CLASS_FULL_SCREEN_ENTER:"bslib-full-screen-enter",CLASS_FULL_SCREEN_EXIT:"bslib-full-screen-exit",ID_FULL_SCREEN_OVERLAY:"bslib-full-screen-overlay"},o.shinyResizeObserver=new d,o.instanceMap=new WeakMap,o.onReadyScheduled=!1;window.bslib=window.bslib||{};window.bslib.Card=o;})(); -//# sourceMappingURL=card.min.js.map diff --git a/shiny/experimental/www/bslib/components/card.min.js.map b/shiny/experimental/www/bslib/components/card.min.js.map deleted file mode 100644 index 1a7e9c50c..000000000 --- a/shiny/experimental/www/bslib/components/card.min.js.map +++ /dev/null @@ -1,7 +0,0 @@ -{ - "version": 3, - "sources": ["../../srcts/src/components/_utils.ts", "../../srcts/src/components/_shinyResizeObserver.ts", "../../srcts/src/components/card.ts"], - "sourcesContent": ["import type { HtmlDep } from \"rstudio-shiny/srcts/types/src/shiny/render\";\n\nimport type { InputBinding as InputBindingType } from \"rstudio-shiny/srcts/types/src/bindings/input\";\n\n// Exclude undefined from T\ntype NotUndefined = T extends undefined ? never : T;\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst InputBinding = (\n window.Shiny ? Shiny.InputBinding : class {}\n) as typeof InputBindingType;\n\nfunction registerBinding(\n inputBindingClass: new () => InputBindingType,\n name: string\n): void {\n if (window.Shiny) {\n Shiny.inputBindings.register(new inputBindingClass(), \"bslib.\" + name);\n }\n}\n\n// Return true if the key exists on the object and the value is not undefined.\n//\n// This method is mainly used in input bindings' `receiveMessage` method.\n// Since we know that the values are sent by Shiny via `{jsonlite}`,\n// then we know that there are no `undefined` values. `null` is possible, but not `undefined`.\nfunction hasDefinedProperty<\n Prop extends keyof X,\n X extends { [key: string]: any }\n>(\n obj: X,\n prop: Prop\n): obj is X & { [key in NonNullable]: NotUndefined } {\n return (\n Object.prototype.hasOwnProperty.call(obj, prop) && obj[prop] !== undefined\n );\n}\n\n// TODO: Shiny should trigger resize events when the output\n// https://github.com/rstudio/shiny/pull/3682\nfunction doWindowResizeOnElementResize(el: HTMLElement): void {\n if ($(el).data(\"window-resize-observer\")) {\n return;\n }\n const resizeEvent = new Event(\"resize\");\n const ro = new ResizeObserver(() => {\n window.dispatchEvent(resizeEvent);\n });\n ro.observe(el);\n $(el).data(\"window-resize-observer\", ro);\n}\n\nfunction getAllFocusableChildren(el: HTMLElement): HTMLElement[] {\n // Cross-referenced with https://allyjs.io/data-tables/focusable.html\n const base = [\n \"a[href]\",\n \"area[href]\",\n \"button\",\n \"details summary\",\n \"input\",\n \"iframe\",\n \"select\",\n \"textarea\",\n '[contentEditable=\"\"]',\n '[contentEditable=\"true\"]',\n '[contentEditable=\"TRUE\"]',\n \"[tabindex]\",\n ];\n const modifiers = [':not([tabindex=\"-1\"])', \":not([disabled])\"];\n const selectors = base.map((b) => b + modifiers.join(\"\"));\n const focusable = el.querySelectorAll(selectors.join(\", \"));\n return Array.from(focusable) as HTMLElement[];\n}\n\nexport {\n InputBinding,\n registerBinding,\n hasDefinedProperty,\n doWindowResizeOnElementResize,\n getAllFocusableChildren,\n};\nexport type { HtmlDep };\n", "/**\n * A resize observer that ensures Shiny outputs resize during or just after\n * their parent container size changes. Useful, in particular, for sidebar\n * transitions or for full-screen card transitions.\n *\n * @class ShinyResizeObserver\n * @typedef {ShinyResizeObserver}\n */\nclass ShinyResizeObserver {\n /**\n * The actual ResizeObserver instance.\n * @private\n * @type {ResizeObserver}\n */\n private resizeObserver: ResizeObserver;\n /**\n * An array of elements that are currently being watched by the Resize\n * Observer.\n *\n * @details\n * We don't currently have lifecycle hooks that allow us to unobserve elements\n * when they are removed from the DOM. As a result, we need to manually check\n * that the elements we're watching still exist in the DOM. This array keeps\n * track of the elements we're watching so that we can check them later.\n * @private\n * @type {HTMLElement[]}\n */\n private resizeObserverEntries: HTMLElement[];\n\n /**\n * Watch containers for size changes and ensure that Shiny outputs and\n * htmlwidgets within resize appropriately.\n *\n * @details\n * The ShinyResizeObserver is used to watch the containers, such as Sidebars\n * and Cards for size changes, in particular when the sidebar state is toggled\n * or the card body is expanded full screen. It performs two primary tasks:\n *\n * 1. Dispatches a `resize` event on the window object. This is necessary to\n * ensure that Shiny outputs resize appropriately. In general, the window\n * resizing is throttled and the output update occurs when the transition\n * is complete.\n * 2. If an output with a resize method on the output binding is detected, we\n * directly call the `.onResize()` method of the binding. This ensures that\n * htmlwidgets transition smoothly. In static mode, htmlwidgets does this\n * already.\n *\n * @note\n * This resize observer also handles race conditions in some complex\n * fill-based layouts with multiple outputs (e.g., plotly), where shiny\n * initializes with the correct sizing, but in-between the 1st and last\n * renderValue(), the size of the output containers can change, meaning every\n * output but the 1st gets initialized with the wrong size during their\n * renderValue(). Then, after the render phase, shiny won't know to trigger a\n * resize since all the widgets will return to their original size (and thus,\n * Shiny thinks there isn't any resizing to do). The resize observer works\n * around this by ensuring that the output is resized whenever its container\n * size changes.\n * @constructor\n */\n constructor() {\n this.resizeObserverEntries = [];\n this.resizeObserver = new ResizeObserver((entries) => {\n const resizeEvent = new Event(\"resize\");\n window.dispatchEvent(resizeEvent);\n\n // the rest of this callback is only relevant in Shiny apps\n if (!window.Shiny) return;\n\n const resized = [] as HTMLElement[];\n\n for (const entry of entries) {\n if (!(entry.target instanceof HTMLElement)) continue;\n if (!entry.target.querySelector(\".shiny-bound-output\")) continue;\n\n entry.target\n .querySelectorAll(\".shiny-bound-output\")\n .forEach((el) => {\n if (resized.includes(el)) return;\n\n const { binding, onResize } = $(el).data(\"shinyOutputBinding\");\n if (!binding || !binding.resize) return;\n\n // if this output is owned by another observer, skip it\n const owner = (el as any).shinyResizeObserver;\n if (owner && owner !== this) return;\n // mark this output as owned by this shinyResizeObserver instance\n if (!owner) (el as any).shinyResizeObserver = this;\n\n // trigger immediate resizing of outputs with a resize method\n onResize(el);\n // only once per output and resize event\n resized.push(el);\n\n // set plot images to 100% width temporarily during the transition\n if (!el.classList.contains(\"shiny-plot-output\")) return;\n const img = el.querySelector(\n 'img:not([width=\"100%\"])'\n );\n if (img) img.setAttribute(\"width\", \"100%\");\n });\n }\n });\n }\n\n /**\n * Observe an element for size changes.\n * @param {HTMLElement} el - The element to observe.\n */\n observe(el: HTMLElement): void {\n this.resizeObserver.observe(el);\n this.resizeObserverEntries.push(el);\n }\n\n /**\n * Stop observing an element for size changes.\n * @param {HTMLElement} el - The element to stop observing.\n */\n unobserve(el: HTMLElement): void {\n const idxEl = this.resizeObserverEntries.indexOf(el);\n if (idxEl < 0) return;\n\n this.resizeObserver.unobserve(el);\n this.resizeObserverEntries.splice(idxEl, 1);\n }\n\n /**\n * This method checks that we're not continuing to watch elements that no\n * longer exist in the DOM. If any are found, we stop observing them and\n * remove them from our array of observed elements.\n *\n * @private\n * @static\n */\n flush(): void {\n this.resizeObserverEntries.forEach((el) => {\n if (!document.body.contains(el)) this.unobserve(el);\n });\n }\n}\n\nexport { ShinyResizeObserver };\n", "import type { Tooltip as TooltipType } from \"bootstrap\";\nimport { getAllFocusableChildren } from \"./_utils\";\nimport { ShinyResizeObserver } from \"./_shinyResizeObserver\";\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst Tooltip = (\n window.bootstrap ? window.bootstrap.Tooltip : class {}\n) as typeof TooltipType;\n\n/**\n * The overlay element that is placed behind the card when expanded full screen.\n *\n * @interface CardFullScreenOverlay\n * @typedef {CardFullScreenOverlay}\n */\ninterface CardFullScreenOverlay {\n /**\n * The full screen overlay container.\n * @type {HTMLDivElement}\n */\n container: HTMLDivElement;\n /**\n * The anchor element used to close the full screen overlay.\n * @type {HTMLAnchorElement}\n */\n anchor: HTMLAnchorElement;\n}\n\n/**\n * The bslib card component class.\n *\n * @class Card\n * @typedef {Card}\n */\nclass Card {\n /**\n * The card container element.\n * @private\n * @type {HTMLElement}\n */\n private card: HTMLElement;\n /**\n * The card's full screen overlay element. We create this element once and add\n * and remove it from the DOM as needed (this simplifies focus management\n * while in full screen mode).\n * @private\n * @type {CardFullScreenOverlay}\n */\n private overlay: CardFullScreenOverlay;\n\n /**\n * Key bslib-specific classes and attributes used by the card component.\n * @private\n * @static\n * @type {{ ATTR_INIT: string; CLASS_CARD: string; CLASS_FULL_SCREEN: string; CLASS_HAS_FULL_SCREEN: string; CLASS_FULL_SCREEN_ENTER: string; CLASS_FULL_SCREEN_EXIT: string; ID_FULL_SCREEN_OVERLAY: string; }}\n */\n private static attr = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ATTR_INIT: \"data-bslib-card-init\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_CARD: \"bslib-card\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_FULL_SCREEN: \"bslib-full-screen\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_HAS_FULL_SCREEN: \"bslib-has-full-screen\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_FULL_SCREEN_ENTER: \"bslib-full-screen-enter\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_FULL_SCREEN_EXIT: \"bslib-full-screen-exit\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ID_FULL_SCREEN_OVERLAY: \"bslib-full-screen-overlay\",\n };\n\n /**\n * A Shiny-specific resize observer that ensures Shiny outputs in within the\n * card resize appropriately.\n * @private\n * @type {ShinyResizeObserver}\n * @static\n */\n private static shinyResizeObserver = new ShinyResizeObserver();\n\n /**\n * Creates an instance of a bslib Card component.\n *\n * @constructor\n * @param {HTMLElement} card\n */\n constructor(card: HTMLElement) {\n // remove initialization attribute and script\n card.removeAttribute(Card.attr.ATTR_INIT);\n card\n .querySelector(`script[${Card.attr.ATTR_INIT}]`)\n ?.remove();\n\n this.card = card;\n Card.instanceMap.set(card, this);\n\n // Let Shiny know to trigger resize when the card size changes\n // TODO: shiny could/should do this itself (rstudio/shiny#3682)\n Card.shinyResizeObserver.observe(this.card);\n\n this._addEventListeners();\n this._enableTooltips();\n this.overlay = this._createOverlay();\n\n // bind event handler methods to this card instance\n this._exitFullScreenOnEscape = this._exitFullScreenOnEscape.bind(this);\n this._trapFocusExit = this._trapFocusExit.bind(this);\n }\n\n /**\n * Enter the card's full screen mode, either programmatically or via an event\n * handler. Full screen mode is activated by adding a class to the card that\n * positions it absolutely and expands it to fill the viewport. In addition,\n * we add a full screen overlay element behind the card and we trap focus in\n * the expanded card while in full screen mode.\n *\n * @param {?Event} [event]\n */\n enterFullScreen(event?: Event): void {\n if (event) event.preventDefault();\n\n document.addEventListener(\"keydown\", this._exitFullScreenOnEscape, false);\n\n // trap focus in the fullscreen container, listening for Tab key on the\n // capture phase so we have the best chance of preventing other handlers\n document.addEventListener(\"keydown\", this._trapFocusExit, true);\n\n // Set initial focus on the card, if not already\n if (!this.card.contains(document.activeElement)) {\n this.card.setAttribute(\"tabindex\", \"-1\");\n this.card.focus();\n }\n\n this.card.classList.add(Card.attr.CLASS_FULL_SCREEN);\n document.body.classList.add(Card.attr.CLASS_HAS_FULL_SCREEN);\n this.card.insertAdjacentElement(\"beforebegin\", this.overlay.container);\n }\n\n /**\n * Exit full screen mode. This removes the full screen overlay element,\n * removes the full screen class from the card, and removes the keyboard event\n * listeners that were added when entering full screen mode.\n */\n exitFullScreen(): void {\n document.removeEventListener(\n \"keydown\",\n this._exitFullScreenOnEscape,\n false\n );\n document.removeEventListener(\"keydown\", this._trapFocusExit, true);\n\n // Remove overlay and remove full screen classes from card\n this.overlay.container.remove();\n this.card.classList.remove(Card.attr.CLASS_FULL_SCREEN);\n this.card.removeAttribute(\"tabindex\");\n document.body.classList.remove(Card.attr.CLASS_HAS_FULL_SCREEN);\n }\n\n /**\n * Adds general card-specific event listeners.\n * @private\n */\n private _addEventListeners(): void {\n const btnFullScreen = this.card.querySelector(\n `:scope > .${Card.attr.CLASS_FULL_SCREEN_ENTER}`\n );\n if (!btnFullScreen) return;\n btnFullScreen.addEventListener(\"click\", (ev) => this.enterFullScreen(ev));\n }\n\n /**\n * Enable tooltips used by the card component.\n * @private\n */\n private _enableTooltips(): void {\n const selector = `.${Card.attr.CLASS_FULL_SCREEN_ENTER}[data-bs-toggle='tooltip']`;\n if (!this.card.querySelector(selector)) {\n return;\n }\n const tooltipList = this.card.querySelectorAll(selector);\n tooltipList.forEach((tt) => new Tooltip(tt));\n }\n\n /**\n * An event handler to exit full screen mode when the Escape key is pressed.\n * @private\n * @param {KeyboardEvent} event\n */\n private _exitFullScreenOnEscape(event: KeyboardEvent): void {\n if (!(event.target instanceof HTMLElement)) return;\n // If the user is in the middle of a select input choice, don't exit\n const selOpenSelectInput = [\"select[open]\", \"input[aria-expanded='true']\"];\n if (event.target.matches(selOpenSelectInput.join(\", \"))) return;\n\n if (event.key === \"Escape\") {\n this.exitFullScreen();\n }\n }\n\n /**\n * An event handler to trap focus within the card when in full screen mode.\n *\n * @description\n * This keyboard event handler ensures that tab focus stays within the card\n * when in full screen mode. When the card is first expanded,\n * we move focus to the card element itself. If focus somehow leaves the card,\n * we returns focus to the card container.\n *\n * Within the card, we handle only tabbing from the close anchor or the last\n * focusable element and only when tab focus would have otherwise left the\n * card. In those cases, we cycle focus to the last focusable element or back\n * to the anchor. If the card doesn't have any focusable elements, we move\n * focus to the close anchor.\n *\n * @note\n * Because the card contents may change, we check for focusable elements\n * every time the handler is called.\n *\n * @private\n * @param {KeyboardEvent} event\n */\n private _trapFocusExit(event: KeyboardEvent): void {\n if (!(event instanceof KeyboardEvent)) return;\n if (event.key !== \"Tab\") return;\n\n const isFocusedContainer = event.target === this.card;\n const isFocusedAnchor = event.target === this.overlay.anchor;\n const isFocusedWithin = this.card.contains(event.target as Node);\n\n const stopEvent = () => {\n event.preventDefault();\n event.stopImmediatePropagation();\n };\n\n if (!(isFocusedWithin || isFocusedContainer || isFocusedAnchor)) {\n // If focus is outside the card, return to the card\n stopEvent();\n this.card.focus();\n return;\n }\n\n // Check focusables every time because the card contents may have changed\n const focusableElements = getAllFocusableChildren(this.card);\n const hasFocusableElements = focusableElements.length > 0;\n\n // We need to handle five cases:\n // 1. The card has no focusable elements --> focus the anchor\n // 2. Focus is on the card container (do nothing, natural tab order)\n // 3. Focus is on the anchor and the user pressed Tab + Shift (backwards)\n // -> Move to the last focusable element (end of card)\n // 4. Focus is on the last focusable element and the user pressed Tab\n // (forwards) -> Move to the anchor (top of card)\n // 5. otherwise we don't interfere\n\n if (!hasFocusableElements) {\n // case 1\n stopEvent();\n this.overlay.anchor.focus();\n return;\n }\n\n // case 2\n if (isFocusedContainer) return;\n\n const lastFocusable = focusableElements[focusableElements.length - 1];\n const isFocusedLast = event.target === lastFocusable;\n\n if (isFocusedAnchor && event.shiftKey) {\n stopEvent();\n lastFocusable.focus();\n return;\n }\n\n if (isFocusedLast && !event.shiftKey) {\n stopEvent();\n this.overlay.anchor.focus();\n return;\n }\n }\n\n /**\n * Creates the full screen overlay.\n * @private\n * @returns {CardFullScreenOverlay}\n */\n private _createOverlay(): CardFullScreenOverlay {\n const container = document.createElement(\"div\");\n container.id = Card.attr.ID_FULL_SCREEN_OVERLAY;\n container.onclick = this.exitFullScreen.bind(this);\n\n const anchor = this._createOverlayCloseAnchor();\n container.appendChild(anchor);\n\n return { container, anchor };\n }\n\n /**\n * Creates the anchor element used to exit the full screen mode.\n * @private\n * @returns {HTMLAnchorElement}\n */\n private _createOverlayCloseAnchor(): HTMLAnchorElement {\n const anchor = document.createElement(\"a\");\n anchor.classList.add(Card.attr.CLASS_FULL_SCREEN_EXIT);\n anchor.tabIndex = 0;\n anchor.onclick = () => this.exitFullScreen();\n anchor.onkeydown = (ev) => {\n if (ev.key === \"Enter\" || ev.key === \" \") {\n this.exitFullScreen();\n }\n };\n anchor.innerHTML = this._overlayCloseHtml();\n\n return anchor;\n }\n\n /**\n * Returns the HTML for the close icon.\n * @private\n * @returns {string}\n */\n private _overlayCloseHtml(): string {\n return (\n \"Close \" +\n \"\" +\n \"\"\n );\n }\n\n /**\n * The registry of card instances and their associated DOM elements.\n * @private\n * @static\n * @type {WeakMap}\n */\n private static instanceMap: WeakMap = new WeakMap();\n\n /**\n * Returns the card instance associated with the given element, if any.\n * @public\n * @static\n * @param {HTMLElement} el\n * @returns {(Card | undefined)}\n */\n public static getInstance(el: HTMLElement): Card | undefined {\n return Card.instanceMap.get(el);\n }\n\n /**\n * If cards are initialized before the DOM is ready, we re-schedule the\n * initialization to occur on DOMContentLoaded.\n * @private\n * @static\n * @type {boolean}\n */\n private static onReadyScheduled = false;\n\n /**\n * Initializes all cards that require initialization on the page, or schedules\n * initialization if the DOM is not yet ready.\n * @public\n * @static\n * @param {boolean} [flushResizeObserver=true]\n */\n public static initializeAllCards(flushResizeObserver = true): void {\n if (document.readyState === \"loading\") {\n if (!Card.onReadyScheduled) {\n Card.onReadyScheduled = true;\n document.addEventListener(\"DOMContentLoaded\", () => {\n Card.initializeAllCards(false);\n });\n }\n return;\n }\n\n if (flushResizeObserver) {\n // Trigger a recheck of observed cards to unobserve non-existent cards\n Card.shinyResizeObserver.flush();\n }\n\n const initSelector = `.${Card.attr.CLASS_CARD}[${Card.attr.ATTR_INIT}]`;\n if (!document.querySelector(initSelector)) {\n // no cards to initialize\n return;\n }\n\n const cards = document.querySelectorAll(initSelector);\n cards.forEach((card) => new Card(card as HTMLElement));\n }\n}\n\n// attach Sidebar class to window for global usage\n(window as any).bslib = (window as any).bslib || {};\n(window as any).bslib.Card = Card;\n\nexport { Card };\n"], - "mappings": "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", - "names": ["InputBinding", "getAllFocusableChildren", "el", "base", "modifiers", "selectors", "b", "focusable", "ShinyResizeObserver", "entries", "resizeEvent", "resized", "entry", "el", "binding", "onResize", "owner", "img", "idxEl", "Tooltip", "_Card", "card", "_a", "event", "btnFullScreen", "ev", "selector", "tt", "selOpenSelectInput", "isFocusedContainer", "isFocusedAnchor", "isFocusedWithin", "stopEvent", "focusableElements", "getAllFocusableChildren", "lastFocusable", "isFocusedLast", "container", "anchor", "el", "flushResizeObserver", "initSelector", "Card", "ShinyResizeObserver"] -} diff --git a/shiny/experimental/www/bslib/components/card/card.css b/shiny/experimental/www/bslib/components/card/card.css new file mode 100644 index 000000000..f51fa7439 --- /dev/null +++ b/shiny/experimental/www/bslib/components/card/card.css @@ -0,0 +1 @@ +.bslib-card .card-body+.card-body{padding-top:0}.bslib-card .card-body{overflow:auto}.bslib-card .card-body p{margin-top:0}.bslib-card .card-body p:last-child{margin-bottom:0}.bslib-card .card-body{max-height:var(--bslib-card-body-max-height, none)}.bslib-card[data-full-screen="true"]>.card-body{max-height:var(--bslib-card-body-max-height-full-screen, none)}.bslib-card .card-header .form-group{margin-bottom:0}.bslib-card .card-header .selectize-control{margin-bottom:0}.bslib-card .card-header .selectize-control .item{margin-right:1.15rem}.bslib-card .card-footer{margin-top:auto}.bslib-card .bslib-navs-card-title{display:flex;flex-wrap:wrap;justify-content:space-between;align-items:center}.bslib-card .bslib-navs-card-title .nav{margin-left:auto}.bslib-card .bslib-sidebar-layout:not([data-bslib-sidebar-border="true"]){border:none}.bslib-card .bslib-sidebar-layout:not([data-bslib-sidebar-border-radius="true"]){border-top-left-radius:0;border-top-right-radius:0}[data-full-screen="true"]{position:fixed;inset:3.5rem 1rem 1rem;height:auto !important;max-height:none !important;width:auto !important;z-index:1070}.bslib-full-screen-enter{display:none;position:absolute;bottom:1px;right:3px;margin:0.5rem;padding:0.55rem !important;font-size:.8rem;cursor:pointer;opacity:.6;color:rgba(var(--bs-body-bg-rgb), 1);z-index:1070}.bslib-full-screen-enter:hover{opacity:1}.card[data-full-screen="false"]:hover>*>.bslib-full-screen-enter{display:block}.bslib-has-full-screen .card:hover>*>.bslib-full-screen-enter{display:none}@media (max-width: 575.98px){.bslib-full-screen-enter{display:none !important}}.bslib-full-screen-exit{position:relative;top:1.35rem;font-size:0.9rem;cursor:pointer;text-decoration:none;display:flex;float:right;margin-right:2.15rem;align-items:center;color:rgba(var(--bs-body-bg-rgb), 0.8)}.bslib-full-screen-exit:hover{color:rgba(var(--bs-body-bg-rgb), 1)}.bslib-full-screen-exit svg{margin-left:0.5rem;font-size:1.5rem}#bslib-full-screen-overlay{position:fixed;inset:0;background-color:rgba(var(--bs-body-color-rgb), 0.6);backdrop-filter:blur(2px);-webkit-backdrop-filter:blur(2px);z-index:1069;animation:bslib-full-screen-overlay-enter 400ms cubic-bezier(0.6, 0.02, 0.65, 1) forwards}@keyframes bslib-full-screen-overlay-enter{0%{opacity:0}100%{opacity:1}} diff --git a/shiny/experimental/www/bslib/components/card/card.min.js b/shiny/experimental/www/bslib/components/card/card.min.js new file mode 100644 index 000000000..810b060e3 --- /dev/null +++ b/shiny/experimental/www/bslib/components/card/card.min.js @@ -0,0 +1,3 @@ +/*! bslib 0.5.0.9000 | (c) 2012-2023 RStudio, PBC. | License: MIT + file LICENSE */ +"use strict";(()=>{var b=window.Shiny?Shiny.InputBinding:class{};function E(u){let e=["a[href]","area[href]","button","details summary","input","iframe","select","textarea",'[contentEditable=""]','[contentEditable="true"]','[contentEditable="TRUE"]',"[tabindex]"],t=[':not([tabindex="-1"])',":not([disabled])"],s=e.map(r=>r+t.join("")),i=u.querySelectorAll(s.join(", "));return Array.from(i)}var d=class{constructor(){this.resizeObserverEntries=[],this.resizeObserver=new ResizeObserver(e=>{let t=new Event("resize");if(window.dispatchEvent(t),!window.Shiny)return;let s=[];for(let i of e)i.target instanceof HTMLElement&&i.target.querySelector(".shiny-bound-output")&&i.target.querySelectorAll(".shiny-bound-output").forEach(r=>{if(s.includes(r))return;let{binding:o,onResize:h}=$(r).data("shinyOutputBinding");if(!o||!o.resize)return;let c=r.shinyResizeObserver;if(c&&c!==this||(c||(r.shinyResizeObserver=this),h(r),s.push(r),!r.classList.contains("shiny-plot-output")))return;let l=r.querySelector('img:not([width="100%"])');l&&l.setAttribute("width","100%")})})}observe(e){this.resizeObserver.observe(e),this.resizeObserverEntries.push(e)}unobserve(e){let t=this.resizeObserverEntries.indexOf(e);t<0||(this.resizeObserver.unobserve(e),this.resizeObserverEntries.splice(t,1))}flush(){this.resizeObserverEntries.forEach(e=>{document.body.contains(e)||this.unobserve(e)})}};var n=class{constructor(e){var t;e.removeAttribute(n.attr.ATTR_INIT),(t=e.querySelector(`script[${n.attr.ATTR_INIT}]`))==null||t.remove(),this.card=e,n.instanceMap.set(e,this),n.shinyResizeObserver.observe(this.card),this._addEventListeners(),this.overlay=this._createOverlay(),this._exitFullScreenOnEscape=this._exitFullScreenOnEscape.bind(this),this._trapFocusExit=this._trapFocusExit.bind(this)}enterFullScreen(e){e&&e.preventDefault(),document.addEventListener("keydown",this._exitFullScreenOnEscape,!1),document.addEventListener("keydown",this._trapFocusExit,!0),this.card.contains(document.activeElement)||(this.card.setAttribute("tabindex","-1"),this.card.focus()),this.card.setAttribute(n.attr.ATTR_FULL_SCREEN,"true"),document.body.classList.add(n.attr.CLASS_HAS_FULL_SCREEN),this.card.insertAdjacentElement("beforebegin",this.overlay.container)}exitFullScreen(){document.removeEventListener("keydown",this._exitFullScreenOnEscape,!1),document.removeEventListener("keydown",this._trapFocusExit,!0),this.overlay.container.remove(),this.card.setAttribute(n.attr.ATTR_FULL_SCREEN,"false"),this.card.removeAttribute("tabindex"),document.body.classList.remove(n.attr.CLASS_HAS_FULL_SCREEN)}_addEventListeners(){let e=this.card.querySelector(`:scope > * > .${n.attr.CLASS_FULL_SCREEN_ENTER}`);e&&e.addEventListener("click",t=>this.enterFullScreen(t))}_exitFullScreenOnEscape(e){if(!(e.target instanceof HTMLElement))return;let t=["select[open]","input[aria-expanded='true']"];e.target.matches(t.join(", "))||e.key==="Escape"&&this.exitFullScreen()}_trapFocusExit(e){if(!(e instanceof KeyboardEvent)||e.key!=="Tab")return;let t=e.target===this.card,s=e.target===this.overlay.anchor,i=this.card.contains(e.target),r=()=>{e.preventDefault(),e.stopImmediatePropagation()};if(!(i||t||s)){r(),this.card.focus();return}let o=E(this.card);if(!(o.length>0)){r(),this.overlay.anchor.focus();return}if(t)return;let c=o[o.length-1],l=e.target===c;if(s&&e.shiftKey){r(),c.focus();return}if(l&&!e.shiftKey){r(),this.overlay.anchor.focus();return}}_createOverlay(){let e=document.createElement("div");e.id=n.attr.ID_FULL_SCREEN_OVERLAY,e.onclick=this.exitFullScreen.bind(this);let t=this._createOverlayCloseAnchor();return e.appendChild(t),{container:e,anchor:t}}_createOverlayCloseAnchor(){let e=document.createElement("a");return e.classList.add(n.attr.CLASS_FULL_SCREEN_EXIT),e.tabIndex=0,e.onclick=()=>this.exitFullScreen(),e.onkeydown=t=>{(t.key==="Enter"||t.key===" ")&&this.exitFullScreen()},e.innerHTML=this._overlayCloseHtml(),e}_overlayCloseHtml(){return"Close "}static getInstance(e){return n.instanceMap.get(e)}static initializeAllCards(e=!0){if(document.readyState==="loading"){n.onReadyScheduled||(n.onReadyScheduled=!0,document.addEventListener("DOMContentLoaded",()=>{n.initializeAllCards(!1)}));return}e&&n.shinyResizeObserver.flush();let t=`.${n.attr.CLASS_CARD}[${n.attr.ATTR_INIT}]`;if(!document.querySelector(t))return;document.querySelectorAll(t).forEach(i=>new n(i))}},a=n;a.attr={ATTR_INIT:"data-bslib-card-init",CLASS_CARD:"bslib-card",ATTR_FULL_SCREEN:"data-full-screen",CLASS_HAS_FULL_SCREEN:"bslib-has-full-screen",CLASS_FULL_SCREEN_ENTER:"bslib-full-screen-enter",CLASS_FULL_SCREEN_EXIT:"bslib-full-screen-exit",ID_FULL_SCREEN_OVERLAY:"bslib-full-screen-overlay"},a.shinyResizeObserver=new d,a.instanceMap=new WeakMap,a.onReadyScheduled=!1;window.bslib=window.bslib||{};window.bslib.Card=a;})(); +//# sourceMappingURL=card.min.js.map diff --git a/shiny/experimental/www/bslib/components/card/card.min.js.map b/shiny/experimental/www/bslib/components/card/card.min.js.map new file mode 100644 index 000000000..353c7ded3 --- /dev/null +++ b/shiny/experimental/www/bslib/components/card/card.min.js.map @@ -0,0 +1,7 @@ +{ + "version": 3, + "sources": ["../../../../srcts/src/components/_utils.ts", "../../../../srcts/src/components/_shinyResizeObserver.ts", "../../../../srcts/src/components/card.ts"], + "sourcesContent": ["import type { HtmlDep } from \"rstudio-shiny/srcts/types/src/shiny/render\";\n\nimport type { InputBinding as InputBindingType } from \"rstudio-shiny/srcts/types/src/bindings/input\";\n\n// Exclude undefined from T\ntype NotUndefined = T extends undefined ? never : T;\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst InputBinding = (\n window.Shiny ? Shiny.InputBinding : class {}\n) as typeof InputBindingType;\n\nfunction registerBinding(\n inputBindingClass: new () => InputBindingType,\n name: string\n): void {\n if (window.Shiny) {\n Shiny.inputBindings.register(new inputBindingClass(), \"bslib.\" + name);\n }\n}\n\n// Return true if the key exists on the object and the value is not undefined.\n//\n// This method is mainly used in input bindings' `receiveMessage` method.\n// Since we know that the values are sent by Shiny via `{jsonlite}`,\n// then we know that there are no `undefined` values. `null` is possible, but not `undefined`.\nfunction hasDefinedProperty<\n Prop extends keyof X,\n X extends { [key: string]: any }\n>(\n obj: X,\n prop: Prop\n): obj is X & { [key in NonNullable]: NotUndefined } {\n return (\n Object.prototype.hasOwnProperty.call(obj, prop) && obj[prop] !== undefined\n );\n}\n\n// TODO: Shiny should trigger resize events when the output\n// https://github.com/rstudio/shiny/pull/3682\nfunction doWindowResizeOnElementResize(el: HTMLElement): void {\n if ($(el).data(\"window-resize-observer\")) {\n return;\n }\n const resizeEvent = new Event(\"resize\");\n const ro = new ResizeObserver(() => {\n window.dispatchEvent(resizeEvent);\n });\n ro.observe(el);\n $(el).data(\"window-resize-observer\", ro);\n}\n\nfunction getAllFocusableChildren(el: HTMLElement): HTMLElement[] {\n // Cross-referenced with https://allyjs.io/data-tables/focusable.html\n const base = [\n \"a[href]\",\n \"area[href]\",\n \"button\",\n \"details summary\",\n \"input\",\n \"iframe\",\n \"select\",\n \"textarea\",\n '[contentEditable=\"\"]',\n '[contentEditable=\"true\"]',\n '[contentEditable=\"TRUE\"]',\n \"[tabindex]\",\n ];\n const modifiers = [':not([tabindex=\"-1\"])', \":not([disabled])\"];\n const selectors = base.map((b) => b + modifiers.join(\"\"));\n const focusable = el.querySelectorAll(selectors.join(\", \"));\n return Array.from(focusable) as HTMLElement[];\n}\n\nexport {\n InputBinding,\n registerBinding,\n hasDefinedProperty,\n doWindowResizeOnElementResize,\n getAllFocusableChildren,\n};\nexport type { HtmlDep };\n", "/**\n * A resize observer that ensures Shiny outputs resize during or just after\n * their parent container size changes. Useful, in particular, for sidebar\n * transitions or for full-screen card transitions.\n *\n * @class ShinyResizeObserver\n * @typedef {ShinyResizeObserver}\n */\nclass ShinyResizeObserver {\n /**\n * The actual ResizeObserver instance.\n * @private\n * @type {ResizeObserver}\n */\n private resizeObserver: ResizeObserver;\n /**\n * An array of elements that are currently being watched by the Resize\n * Observer.\n *\n * @details\n * We don't currently have lifecycle hooks that allow us to unobserve elements\n * when they are removed from the DOM. As a result, we need to manually check\n * that the elements we're watching still exist in the DOM. This array keeps\n * track of the elements we're watching so that we can check them later.\n * @private\n * @type {HTMLElement[]}\n */\n private resizeObserverEntries: HTMLElement[];\n\n /**\n * Watch containers for size changes and ensure that Shiny outputs and\n * htmlwidgets within resize appropriately.\n *\n * @details\n * The ShinyResizeObserver is used to watch the containers, such as Sidebars\n * and Cards for size changes, in particular when the sidebar state is toggled\n * or the card body is expanded full screen. It performs two primary tasks:\n *\n * 1. Dispatches a `resize` event on the window object. This is necessary to\n * ensure that Shiny outputs resize appropriately. In general, the window\n * resizing is throttled and the output update occurs when the transition\n * is complete.\n * 2. If an output with a resize method on the output binding is detected, we\n * directly call the `.onResize()` method of the binding. This ensures that\n * htmlwidgets transition smoothly. In static mode, htmlwidgets does this\n * already.\n *\n * @note\n * This resize observer also handles race conditions in some complex\n * fill-based layouts with multiple outputs (e.g., plotly), where shiny\n * initializes with the correct sizing, but in-between the 1st and last\n * renderValue(), the size of the output containers can change, meaning every\n * output but the 1st gets initialized with the wrong size during their\n * renderValue(). Then, after the render phase, shiny won't know to trigger a\n * resize since all the widgets will return to their original size (and thus,\n * Shiny thinks there isn't any resizing to do). The resize observer works\n * around this by ensuring that the output is resized whenever its container\n * size changes.\n * @constructor\n */\n constructor() {\n this.resizeObserverEntries = [];\n this.resizeObserver = new ResizeObserver((entries) => {\n const resizeEvent = new Event(\"resize\");\n window.dispatchEvent(resizeEvent);\n\n // the rest of this callback is only relevant in Shiny apps\n if (!window.Shiny) return;\n\n const resized = [] as HTMLElement[];\n\n for (const entry of entries) {\n if (!(entry.target instanceof HTMLElement)) continue;\n if (!entry.target.querySelector(\".shiny-bound-output\")) continue;\n\n entry.target\n .querySelectorAll(\".shiny-bound-output\")\n .forEach((el) => {\n if (resized.includes(el)) return;\n\n const { binding, onResize } = $(el).data(\"shinyOutputBinding\");\n if (!binding || !binding.resize) return;\n\n // if this output is owned by another observer, skip it\n const owner = (el as any).shinyResizeObserver;\n if (owner && owner !== this) return;\n // mark this output as owned by this shinyResizeObserver instance\n if (!owner) (el as any).shinyResizeObserver = this;\n\n // trigger immediate resizing of outputs with a resize method\n onResize(el);\n // only once per output and resize event\n resized.push(el);\n\n // set plot images to 100% width temporarily during the transition\n if (!el.classList.contains(\"shiny-plot-output\")) return;\n const img = el.querySelector(\n 'img:not([width=\"100%\"])'\n );\n if (img) img.setAttribute(\"width\", \"100%\");\n });\n }\n });\n }\n\n /**\n * Observe an element for size changes.\n * @param {HTMLElement} el - The element to observe.\n */\n observe(el: HTMLElement): void {\n this.resizeObserver.observe(el);\n this.resizeObserverEntries.push(el);\n }\n\n /**\n * Stop observing an element for size changes.\n * @param {HTMLElement} el - The element to stop observing.\n */\n unobserve(el: HTMLElement): void {\n const idxEl = this.resizeObserverEntries.indexOf(el);\n if (idxEl < 0) return;\n\n this.resizeObserver.unobserve(el);\n this.resizeObserverEntries.splice(idxEl, 1);\n }\n\n /**\n * This method checks that we're not continuing to watch elements that no\n * longer exist in the DOM. If any are found, we stop observing them and\n * remove them from our array of observed elements.\n *\n * @private\n * @static\n */\n flush(): void {\n this.resizeObserverEntries.forEach((el) => {\n if (!document.body.contains(el)) this.unobserve(el);\n });\n }\n}\n\nexport { ShinyResizeObserver };\n", "import { getAllFocusableChildren } from \"./_utils\";\nimport { ShinyResizeObserver } from \"./_shinyResizeObserver\";\n\n/**\n * The overlay element that is placed behind the card when expanded full screen.\n *\n * @interface CardFullScreenOverlay\n * @typedef {CardFullScreenOverlay}\n */\ninterface CardFullScreenOverlay {\n /**\n * The full screen overlay container.\n * @type {HTMLDivElement}\n */\n container: HTMLDivElement;\n /**\n * The anchor element used to close the full screen overlay.\n * @type {HTMLAnchorElement}\n */\n anchor: HTMLAnchorElement;\n}\n\n/**\n * The bslib card component class.\n *\n * @class Card\n * @typedef {Card}\n */\nclass Card {\n /**\n * The card container element.\n * @private\n * @type {HTMLElement}\n */\n private card: HTMLElement;\n /**\n * The card's full screen overlay element. We create this element once and add\n * and remove it from the DOM as needed (this simplifies focus management\n * while in full screen mode).\n * @private\n * @type {CardFullScreenOverlay}\n */\n private overlay: CardFullScreenOverlay;\n\n /**\n * Key bslib-specific classes and attributes used by the card component.\n * @private\n * @static\n * @type {{ ATTR_INIT: string; CLASS_CARD: string; CLASS_FULL_SCREEN: string; CLASS_HAS_FULL_SCREEN: string; CLASS_FULL_SCREEN_ENTER: string; CLASS_FULL_SCREEN_EXIT: string; ID_FULL_SCREEN_OVERLAY: string; }}\n */\n private static attr = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ATTR_INIT: \"data-bslib-card-init\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_CARD: \"bslib-card\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ATTR_FULL_SCREEN: \"data-full-screen\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_HAS_FULL_SCREEN: \"bslib-has-full-screen\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_FULL_SCREEN_ENTER: \"bslib-full-screen-enter\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n CLASS_FULL_SCREEN_EXIT: \"bslib-full-screen-exit\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n ID_FULL_SCREEN_OVERLAY: \"bslib-full-screen-overlay\",\n };\n\n /**\n * A Shiny-specific resize observer that ensures Shiny outputs in within the\n * card resize appropriately.\n * @private\n * @type {ShinyResizeObserver}\n * @static\n */\n private static shinyResizeObserver = new ShinyResizeObserver();\n\n /**\n * Creates an instance of a bslib Card component.\n *\n * @constructor\n * @param {HTMLElement} card\n */\n constructor(card: HTMLElement) {\n // remove initialization attribute and script\n card.removeAttribute(Card.attr.ATTR_INIT);\n card\n .querySelector(`script[${Card.attr.ATTR_INIT}]`)\n ?.remove();\n\n this.card = card;\n Card.instanceMap.set(card, this);\n\n // Let Shiny know to trigger resize when the card size changes\n // TODO: shiny could/should do this itself (rstudio/shiny#3682)\n Card.shinyResizeObserver.observe(this.card);\n\n this._addEventListeners();\n this.overlay = this._createOverlay();\n\n // bind event handler methods to this card instance\n this._exitFullScreenOnEscape = this._exitFullScreenOnEscape.bind(this);\n this._trapFocusExit = this._trapFocusExit.bind(this);\n }\n\n /**\n * Enter the card's full screen mode, either programmatically or via an event\n * handler. Full screen mode is activated by adding a class to the card that\n * positions it absolutely and expands it to fill the viewport. In addition,\n * we add a full screen overlay element behind the card and we trap focus in\n * the expanded card while in full screen mode.\n *\n * @param {?Event} [event]\n */\n enterFullScreen(event?: Event): void {\n if (event) event.preventDefault();\n\n document.addEventListener(\"keydown\", this._exitFullScreenOnEscape, false);\n\n // trap focus in the fullscreen container, listening for Tab key on the\n // capture phase so we have the best chance of preventing other handlers\n document.addEventListener(\"keydown\", this._trapFocusExit, true);\n\n // Set initial focus on the card, if not already\n if (!this.card.contains(document.activeElement)) {\n this.card.setAttribute(\"tabindex\", \"-1\");\n this.card.focus();\n }\n\n this.card.setAttribute(Card.attr.ATTR_FULL_SCREEN, \"true\");\n document.body.classList.add(Card.attr.CLASS_HAS_FULL_SCREEN);\n this.card.insertAdjacentElement(\"beforebegin\", this.overlay.container);\n }\n\n /**\n * Exit full screen mode. This removes the full screen overlay element,\n * removes the full screen class from the card, and removes the keyboard event\n * listeners that were added when entering full screen mode.\n */\n exitFullScreen(): void {\n document.removeEventListener(\n \"keydown\",\n this._exitFullScreenOnEscape,\n false\n );\n document.removeEventListener(\"keydown\", this._trapFocusExit, true);\n\n // Remove overlay and remove full screen classes from card\n this.overlay.container.remove();\n this.card.setAttribute(Card.attr.ATTR_FULL_SCREEN, \"false\");\n this.card.removeAttribute(\"tabindex\");\n document.body.classList.remove(Card.attr.CLASS_HAS_FULL_SCREEN);\n }\n\n /**\n * Adds general card-specific event listeners.\n * @private\n */\n private _addEventListeners(): void {\n const btnFullScreen = this.card.querySelector(\n `:scope > * > .${Card.attr.CLASS_FULL_SCREEN_ENTER}`\n );\n if (!btnFullScreen) return;\n btnFullScreen.addEventListener(\"click\", (ev) => this.enterFullScreen(ev));\n }\n\n /**\n * An event handler to exit full screen mode when the Escape key is pressed.\n * @private\n * @param {KeyboardEvent} event\n */\n private _exitFullScreenOnEscape(event: KeyboardEvent): void {\n if (!(event.target instanceof HTMLElement)) return;\n // If the user is in the middle of a select input choice, don't exit\n const selOpenSelectInput = [\"select[open]\", \"input[aria-expanded='true']\"];\n if (event.target.matches(selOpenSelectInput.join(\", \"))) return;\n\n if (event.key === \"Escape\") {\n this.exitFullScreen();\n }\n }\n\n /**\n * An event handler to trap focus within the card when in full screen mode.\n *\n * @description\n * This keyboard event handler ensures that tab focus stays within the card\n * when in full screen mode. When the card is first expanded,\n * we move focus to the card element itself. If focus somehow leaves the card,\n * we returns focus to the card container.\n *\n * Within the card, we handle only tabbing from the close anchor or the last\n * focusable element and only when tab focus would have otherwise left the\n * card. In those cases, we cycle focus to the last focusable element or back\n * to the anchor. If the card doesn't have any focusable elements, we move\n * focus to the close anchor.\n *\n * @note\n * Because the card contents may change, we check for focusable elements\n * every time the handler is called.\n *\n * @private\n * @param {KeyboardEvent} event\n */\n private _trapFocusExit(event: KeyboardEvent): void {\n if (!(event instanceof KeyboardEvent)) return;\n if (event.key !== \"Tab\") return;\n\n const isFocusedContainer = event.target === this.card;\n const isFocusedAnchor = event.target === this.overlay.anchor;\n const isFocusedWithin = this.card.contains(event.target as Node);\n\n const stopEvent = () => {\n event.preventDefault();\n event.stopImmediatePropagation();\n };\n\n if (!(isFocusedWithin || isFocusedContainer || isFocusedAnchor)) {\n // If focus is outside the card, return to the card\n stopEvent();\n this.card.focus();\n return;\n }\n\n // Check focusables every time because the card contents may have changed\n const focusableElements = getAllFocusableChildren(this.card);\n const hasFocusableElements = focusableElements.length > 0;\n\n // We need to handle five cases:\n // 1. The card has no focusable elements --> focus the anchor\n // 2. Focus is on the card container (do nothing, natural tab order)\n // 3. Focus is on the anchor and the user pressed Tab + Shift (backwards)\n // -> Move to the last focusable element (end of card)\n // 4. Focus is on the last focusable element and the user pressed Tab\n // (forwards) -> Move to the anchor (top of card)\n // 5. otherwise we don't interfere\n\n if (!hasFocusableElements) {\n // case 1\n stopEvent();\n this.overlay.anchor.focus();\n return;\n }\n\n // case 2\n if (isFocusedContainer) return;\n\n const lastFocusable = focusableElements[focusableElements.length - 1];\n const isFocusedLast = event.target === lastFocusable;\n\n if (isFocusedAnchor && event.shiftKey) {\n stopEvent();\n lastFocusable.focus();\n return;\n }\n\n if (isFocusedLast && !event.shiftKey) {\n stopEvent();\n this.overlay.anchor.focus();\n return;\n }\n }\n\n /**\n * Creates the full screen overlay.\n * @private\n * @returns {CardFullScreenOverlay}\n */\n private _createOverlay(): CardFullScreenOverlay {\n const container = document.createElement(\"div\");\n container.id = Card.attr.ID_FULL_SCREEN_OVERLAY;\n container.onclick = this.exitFullScreen.bind(this);\n\n const anchor = this._createOverlayCloseAnchor();\n container.appendChild(anchor);\n\n return { container, anchor };\n }\n\n /**\n * Creates the anchor element used to exit the full screen mode.\n * @private\n * @returns {HTMLAnchorElement}\n */\n private _createOverlayCloseAnchor(): HTMLAnchorElement {\n const anchor = document.createElement(\"a\");\n anchor.classList.add(Card.attr.CLASS_FULL_SCREEN_EXIT);\n anchor.tabIndex = 0;\n anchor.onclick = () => this.exitFullScreen();\n anchor.onkeydown = (ev) => {\n if (ev.key === \"Enter\" || ev.key === \" \") {\n this.exitFullScreen();\n }\n };\n anchor.innerHTML = this._overlayCloseHtml();\n\n return anchor;\n }\n\n /**\n * Returns the HTML for the close icon.\n * @private\n * @returns {string}\n */\n private _overlayCloseHtml(): string {\n return (\n \"Close \" +\n \"\" +\n \"\"\n );\n }\n\n /**\n * The registry of card instances and their associated DOM elements.\n * @private\n * @static\n * @type {WeakMap}\n */\n private static instanceMap: WeakMap = new WeakMap();\n\n /**\n * Returns the card instance associated with the given element, if any.\n * @public\n * @static\n * @param {HTMLElement} el\n * @returns {(Card | undefined)}\n */\n public static getInstance(el: HTMLElement): Card | undefined {\n return Card.instanceMap.get(el);\n }\n\n /**\n * If cards are initialized before the DOM is ready, we re-schedule the\n * initialization to occur on DOMContentLoaded.\n * @private\n * @static\n * @type {boolean}\n */\n private static onReadyScheduled = false;\n\n /**\n * Initializes all cards that require initialization on the page, or schedules\n * initialization if the DOM is not yet ready.\n * @public\n * @static\n * @param {boolean} [flushResizeObserver=true]\n */\n public static initializeAllCards(flushResizeObserver = true): void {\n if (document.readyState === \"loading\") {\n if (!Card.onReadyScheduled) {\n Card.onReadyScheduled = true;\n document.addEventListener(\"DOMContentLoaded\", () => {\n Card.initializeAllCards(false);\n });\n }\n return;\n }\n\n if (flushResizeObserver) {\n // Trigger a recheck of observed cards to unobserve non-existent cards\n Card.shinyResizeObserver.flush();\n }\n\n const initSelector = `.${Card.attr.CLASS_CARD}[${Card.attr.ATTR_INIT}]`;\n if (!document.querySelector(initSelector)) {\n // no cards to initialize\n return;\n }\n\n const cards = document.querySelectorAll(initSelector);\n cards.forEach((card) => new Card(card as HTMLElement));\n }\n}\n\n// attach Sidebar class to window for global usage\n(window as any).bslib = (window as any).bslib || {};\n(window as any).bslib.Card = Card;\n\nexport { Card };\n"], + "mappings": "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", + "names": ["InputBinding", "getAllFocusableChildren", "el", "base", "modifiers", "selectors", "b", "focusable", "ShinyResizeObserver", "entries", "resizeEvent", "resized", "entry", "el", "binding", "onResize", "owner", "img", "idxEl", "_Card", "card", "_a", "event", "btnFullScreen", "ev", "selOpenSelectInput", "isFocusedContainer", "isFocusedAnchor", "isFocusedWithin", "stopEvent", "focusableElements", "getAllFocusableChildren", "lastFocusable", "isFocusedLast", "container", "anchor", "el", "flushResizeObserver", "initSelector", "Card", "ShinyResizeObserver"] +} diff --git a/shiny/experimental/www/bslib/components/grid/grid.css b/shiny/experimental/www/bslib/components/grid/grid.css new file mode 100644 index 000000000..be4eb0379 --- /dev/null +++ b/shiny/experimental/www/bslib/components/grid/grid.css @@ -0,0 +1 @@ +.bslib-grid{display:grid !important;gap:var(--bslib-spacer, 1rem);height:var(--bslib-grid-height)}.bslib-grid.grid{grid-template-rows:unset;grid-auto-rows:var(--bslib-grid--row-heights);--bslib-grid--row-heights--xs: unset;--bslib-grid--row-heights--sm: unset;--bslib-grid--row-heights--md: unset;--bslib-grid--row-heights--lg: unset;--bslib-grid--row-heights--xl: unset;--bslib-grid--row-heights--xxl: unset}.bslib-grid.grid.bslib-grid--row-heights--xs{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xs)}@media (min-width: 576px){.bslib-grid.grid.bslib-grid--row-heights--sm{--bslib-grid--row-heights: var(--bslib-grid--row-heights--sm)}}@media (min-width: 768px){.bslib-grid.grid.bslib-grid--row-heights--md{--bslib-grid--row-heights: var(--bslib-grid--row-heights--md)}}@media (min-width: 992px){.bslib-grid.grid.bslib-grid--row-heights--lg{--bslib-grid--row-heights: var(--bslib-grid--row-heights--lg)}}@media (min-width: 1200px){.bslib-grid.grid.bslib-grid--row-heights--xl{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xl)}}@media (min-width: 1400px){.bslib-grid.grid.bslib-grid--row-heights--xxl{--bslib-grid--row-heights: var(--bslib-grid--row-heights--xxl)}}.bslib-grid-item{grid-column:auto/span 1}@media (max-width: 767.98px){.bslib-grid-item{grid-column:1 / -1}}@media (max-width: 575.98px){.bslib-grid{grid-template-columns:1fr !important;height:var(--bslib-grid-height-mobile)}.bslib-grid.grid{height:unset !important;grid-auto-rows:var(--bslib-grid--row-heights--xs, auto)}} diff --git a/shiny/experimental/www/bslib/components/nav_spacer/nav_spacer.css b/shiny/experimental/www/bslib/components/nav_spacer/nav_spacer.css new file mode 100644 index 000000000..be42a4f9e --- /dev/null +++ b/shiny/experimental/www/bslib/components/nav_spacer/nav_spacer.css @@ -0,0 +1 @@ +@media (min-width: 576px){.nav:not(.nav-hidden){display:flex !important;display:-webkit-flex !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column){float:none !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column)>.bslib-nav-spacer{margin-left:auto !important}.nav:not(.nav-hidden):not(.nav-stacked):not(.flex-column)>.form-inline{margin-top:auto;margin-bottom:auto}.nav:not(.nav-hidden).nav-stacked{flex-direction:column;-webkit-flex-direction:column;height:100%}.nav:not(.nav-hidden).nav-stacked>.bslib-nav-spacer{margin-top:auto !important}} diff --git a/shiny/experimental/www/bslib/components/page_fillable/page_fillable.css b/shiny/experimental/www/bslib/components/page_fillable/page_fillable.css new file mode 100644 index 000000000..b4a492c79 --- /dev/null +++ b/shiny/experimental/www/bslib/components/page_fillable/page_fillable.css @@ -0,0 +1 @@ +html{height:100%}.bslib-page-fill{width:100%;height:100%;margin:0;padding:var(--bslib-spacer, 1rem);gap:var(--bslib-spacer, 1rem)}@media (max-width: 575.98px){.bslib-page-fill{height:var(--bslib-page-fill-mobile-height, auto)}} diff --git a/shiny/experimental/www/bslib/components/page_navbar/page_navbar.css b/shiny/experimental/www/bslib/components/page_navbar/page_navbar.css new file mode 100644 index 000000000..fca340be7 --- /dev/null +++ b/shiny/experimental/www/bslib/components/page_navbar/page_navbar.css @@ -0,0 +1 @@ +.navbar+.container-fluid:has(>.tab-content>.tab-pane.active.html-fill-container){padding-left:0;padding-right:0}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container{padding:var(--bslib-spacer, 1rem);gap:var(--bslib-spacer, 1rem)}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container:has(>.bslib-sidebar-layout:only-child){padding:0}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border="true"]){border-left:none;border-right:none;border-bottom:none}.navbar+.container-fluid>.tab-content>.tab-pane.active.html-fill-container>.bslib-sidebar-layout:only-child:not([data-bslib-sidebar-border-radius="true"]){border-radius:0}.navbar+div>.bslib-sidebar-layout{border-top:var(--bslib-sidebar-border)} diff --git a/shiny/experimental/www/bslib/components/page_sidebar/page_sidebar.css b/shiny/experimental/www/bslib/components/page_sidebar/page_sidebar.css new file mode 100644 index 000000000..be45d5768 --- /dev/null +++ b/shiny/experimental/www/bslib/components/page_sidebar/page_sidebar.css @@ -0,0 +1 @@ +.bslib-page-title{background-color:#212529;color:#fff;font-size:1.5rem;font-weight:300;padding:var(--bslib-spacer, 1rem);padding-left:1.5rem;margin-bottom:0} diff --git a/shiny/experimental/www/bslib/components/sidebar.min.js b/shiny/experimental/www/bslib/components/sidebar.min.js deleted file mode 100644 index 7c93659de..000000000 --- a/shiny/experimental/www/bslib/components/sidebar.min.js +++ /dev/null @@ -1,3 +0,0 @@ -/*! bslib 0.5.0.9000 | (c) 2012-2023 RStudio, PBC. | License: MIT + file LICENSE */ -"use strict";(()=>{var g=window.Shiny?Shiny.InputBinding:class{};function f(b,e){window.Shiny&&Shiny.inputBindings.register(new b,"bslib."+e)}var u=class{constructor(){this.resizeObserverEntries=[],this.resizeObserver=new ResizeObserver(e=>{let t=new Event("resize");if(window.dispatchEvent(t),!window.Shiny)return;let i=[];for(let r of e)r.target instanceof HTMLElement&&r.target.querySelector(".shiny-bound-output")&&r.target.querySelectorAll(".shiny-bound-output").forEach(o=>{if(i.includes(o))return;let{binding:l,onResize:d}=$(o).data("shinyOutputBinding");if(!l||!l.resize)return;let s=o.shinyResizeObserver;if(s&&s!==this||(s||(o.shinyResizeObserver=this),d(o),i.push(o),!o.classList.contains("shiny-plot-output")))return;let c=o.querySelector('img:not([width="100%"])');c&&c.setAttribute("width","100%")})})}observe(e){this.resizeObserver.observe(e),this.resizeObserverEntries.push(e)}unobserve(e){let t=this.resizeObserverEntries.indexOf(e);t<0||(this.resizeObserver.unobserve(e),this.resizeObserverEntries.splice(t,1))}flush(){this.resizeObserverEntries.forEach(e=>{document.body.contains(e)||this.unobserve(e)})}};var n=class{constructor(e){if(n.instanceMap.set(e,this),this.layout={container:e,main:e.querySelector(":scope > .main"),sidebar:e.querySelector(":scope > .sidebar"),toggle:e.querySelector(":scope > .collapse-toggle")},!this.layout.toggle)throw new Error("Tried to initialize a non-collapsible sidebar.");this._initEventListeners(),this._initSidebarCounters(),this._initDesktop(),n.shinyResizeObserver.observe(this.layout.main),e.removeAttribute("data-bslib-sidebar-init");let t=e.querySelector(":scope > script[data-bslib-sidebar-init]");t&&e.removeChild(t)}get isClosed(){return this.layout.container.classList.contains(n.classes.COLLAPSE)}static getInstance(e){return n.instanceMap.get(e)}static initCollapsibleAll(e=!0){if(document.readyState==="loading"){n.onReadyScheduled||(n.onReadyScheduled=!0,document.addEventListener("DOMContentLoaded",()=>{n.initCollapsibleAll(!1)}));return}let t=`.${n.classes.LAYOUT}[data-bslib-sidebar-init]`;if(!document.querySelector(t))return;e&&n.shinyResizeObserver.flush(),document.querySelectorAll(t).forEach(r=>new n(r))}_initEventListeners(){var t;let{toggle:e}=this.layout;e.addEventListener("click",i=>{i.preventDefault(),this.toggle("toggle")}),(t=e.querySelector(".collapse-icon"))==null||t.addEventListener("transitionend",()=>this._finalizeState())}_initSidebarCounters(){let{container:e}=this.layout,t=`.${n.classes.LAYOUT}> .main > .${n.classes.LAYOUT}:not([data-bslib-sidebar-open="always"])`;if(!(e.querySelector(t)===null))return;function r(s){return s=s?s.parentElement:null,s&&s.classList.contains("main")&&(s=s.parentElement),s&&s.classList.contains(n.classes.LAYOUT)?s:null}let o=[e],l=r(e);for(;l;)o.unshift(l),l=r(l);let d={left:0,right:0};o.forEach(function(s,c){s.style.setProperty("--bslib-sidebar-counter",c.toString());let h=s.classList.contains("sidebar-right")?d.right++:d.left++;s.style.setProperty("--bslib-sidebar-overlap-counter",h.toString())})}_initDesktop(){var i;let{container:e}=this.layout;if(((i=e.dataset.bslibSidebarOpen)==null?void 0:i.trim())!=="desktop")return;window.getComputedStyle(e).getPropertyValue("--bslib-sidebar-js-init-collapsed").trim()==="true"&&this.toggle("close")}toggle(e){typeof e=="undefined"&&(e="toggle");let{container:t,sidebar:i}=this.layout,r=this.isClosed;if(["open","close","toggle"].indexOf(e)===-1)throw new Error(`Unknown method ${e}`);e==="toggle"&&(e=r?"open":"close"),!(r&&e==="close"||!r&&e==="open")&&(e==="open"&&(i.hidden=!1),t.classList.add(n.classes.TRANSITIONING),t.classList.toggle(n.classes.COLLAPSE))}_finalizeState(){let{container:e,sidebar:t,toggle:i}=this.layout;e.classList.remove(n.classes.TRANSITIONING),t.hidden=this.isClosed,i.ariaExpanded=this.isClosed?"false":"true";let r=new CustomEvent("bslib.sidebar",{bubbles:!0,detail:{open:!this.isClosed}});t.dispatchEvent(r),$(t).trigger("toggleCollapse.sidebarInputBinding"),$(t).trigger(this.isClosed?"hidden":"shown")}},a=n;a.shinyResizeObserver=new u,a.classes={LAYOUT:"bslib-sidebar-layout",COLLAPSE:"sidebar-collapsed",TRANSITIONING:"transitioning"},a.onReadyScheduled=!1,a.instanceMap=new WeakMap;var p=class extends g{find(e){return $(e).find(`.${a.classes.LAYOUT} > .bslib-sidebar-input`)}getValue(e){let t=a.getInstance(e.parentElement);return t?!t.isClosed:!1}setValue(e,t){let i=t?"open":"close";this.receiveMessage(e,{method:i})}subscribe(e,t){$(e).on("toggleCollapse.sidebarInputBinding",function(i){t(!0)})}unsubscribe(e){$(e).off(".sidebarInputBinding")}receiveMessage(e,t){let i=a.getInstance(e.parentElement);i&&i.toggle(t.method)}};f(p,"sidebar");window.bslib=window.bslib||{};window.bslib.Sidebar=a;})(); -//# sourceMappingURL=sidebar.min.js.map diff --git a/shiny/experimental/www/bslib/components/sidebar.min.js.map b/shiny/experimental/www/bslib/components/sidebar.min.js.map deleted file mode 100644 index 0e14b8773..000000000 --- a/shiny/experimental/www/bslib/components/sidebar.min.js.map +++ /dev/null @@ -1,7 +0,0 @@ -{ - "version": 3, - "sources": ["../../srcts/src/components/_utils.ts", "../../srcts/src/components/_shinyResizeObserver.ts", "../../srcts/src/components/sidebar.ts"], - "sourcesContent": ["import type { HtmlDep } from \"rstudio-shiny/srcts/types/src/shiny/render\";\n\nimport type { InputBinding as InputBindingType } from \"rstudio-shiny/srcts/types/src/bindings/input\";\n\n// Exclude undefined from T\ntype NotUndefined = T extends undefined ? never : T;\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst InputBinding = (\n window.Shiny ? Shiny.InputBinding : class {}\n) as typeof InputBindingType;\n\nfunction registerBinding(\n inputBindingClass: new () => InputBindingType,\n name: string\n): void {\n if (window.Shiny) {\n Shiny.inputBindings.register(new inputBindingClass(), \"bslib.\" + name);\n }\n}\n\n// Return true if the key exists on the object and the value is not undefined.\n//\n// This method is mainly used in input bindings' `receiveMessage` method.\n// Since we know that the values are sent by Shiny via `{jsonlite}`,\n// then we know that there are no `undefined` values. `null` is possible, but not `undefined`.\nfunction hasDefinedProperty<\n Prop extends keyof X,\n X extends { [key: string]: any }\n>(\n obj: X,\n prop: Prop\n): obj is X & { [key in NonNullable]: NotUndefined } {\n return (\n Object.prototype.hasOwnProperty.call(obj, prop) && obj[prop] !== undefined\n );\n}\n\n// TODO: Shiny should trigger resize events when the output\n// https://github.com/rstudio/shiny/pull/3682\nfunction doWindowResizeOnElementResize(el: HTMLElement): void {\n if ($(el).data(\"window-resize-observer\")) {\n return;\n }\n const resizeEvent = new Event(\"resize\");\n const ro = new ResizeObserver(() => {\n window.dispatchEvent(resizeEvent);\n });\n ro.observe(el);\n $(el).data(\"window-resize-observer\", ro);\n}\n\nfunction getAllFocusableChildren(el: HTMLElement): HTMLElement[] {\n // Cross-referenced with https://allyjs.io/data-tables/focusable.html\n const base = [\n \"a[href]\",\n \"area[href]\",\n \"button\",\n \"details summary\",\n \"input\",\n \"iframe\",\n \"select\",\n \"textarea\",\n '[contentEditable=\"\"]',\n '[contentEditable=\"true\"]',\n '[contentEditable=\"TRUE\"]',\n \"[tabindex]\",\n ];\n const modifiers = [':not([tabindex=\"-1\"])', \":not([disabled])\"];\n const selectors = base.map((b) => b + modifiers.join(\"\"));\n const focusable = el.querySelectorAll(selectors.join(\", \"));\n return Array.from(focusable) as HTMLElement[];\n}\n\nexport {\n InputBinding,\n registerBinding,\n hasDefinedProperty,\n doWindowResizeOnElementResize,\n getAllFocusableChildren,\n};\nexport type { HtmlDep };\n", "/**\n * A resize observer that ensures Shiny outputs resize during or just after\n * their parent container size changes. Useful, in particular, for sidebar\n * transitions or for full-screen card transitions.\n *\n * @class ShinyResizeObserver\n * @typedef {ShinyResizeObserver}\n */\nclass ShinyResizeObserver {\n /**\n * The actual ResizeObserver instance.\n * @private\n * @type {ResizeObserver}\n */\n private resizeObserver: ResizeObserver;\n /**\n * An array of elements that are currently being watched by the Resize\n * Observer.\n *\n * @details\n * We don't currently have lifecycle hooks that allow us to unobserve elements\n * when they are removed from the DOM. As a result, we need to manually check\n * that the elements we're watching still exist in the DOM. This array keeps\n * track of the elements we're watching so that we can check them later.\n * @private\n * @type {HTMLElement[]}\n */\n private resizeObserverEntries: HTMLElement[];\n\n /**\n * Watch containers for size changes and ensure that Shiny outputs and\n * htmlwidgets within resize appropriately.\n *\n * @details\n * The ShinyResizeObserver is used to watch the containers, such as Sidebars\n * and Cards for size changes, in particular when the sidebar state is toggled\n * or the card body is expanded full screen. It performs two primary tasks:\n *\n * 1. Dispatches a `resize` event on the window object. This is necessary to\n * ensure that Shiny outputs resize appropriately. In general, the window\n * resizing is throttled and the output update occurs when the transition\n * is complete.\n * 2. If an output with a resize method on the output binding is detected, we\n * directly call the `.onResize()` method of the binding. This ensures that\n * htmlwidgets transition smoothly. In static mode, htmlwidgets does this\n * already.\n *\n * @note\n * This resize observer also handles race conditions in some complex\n * fill-based layouts with multiple outputs (e.g., plotly), where shiny\n * initializes with the correct sizing, but in-between the 1st and last\n * renderValue(), the size of the output containers can change, meaning every\n * output but the 1st gets initialized with the wrong size during their\n * renderValue(). Then, after the render phase, shiny won't know to trigger a\n * resize since all the widgets will return to their original size (and thus,\n * Shiny thinks there isn't any resizing to do). The resize observer works\n * around this by ensuring that the output is resized whenever its container\n * size changes.\n * @constructor\n */\n constructor() {\n this.resizeObserverEntries = [];\n this.resizeObserver = new ResizeObserver((entries) => {\n const resizeEvent = new Event(\"resize\");\n window.dispatchEvent(resizeEvent);\n\n // the rest of this callback is only relevant in Shiny apps\n if (!window.Shiny) return;\n\n const resized = [] as HTMLElement[];\n\n for (const entry of entries) {\n if (!(entry.target instanceof HTMLElement)) continue;\n if (!entry.target.querySelector(\".shiny-bound-output\")) continue;\n\n entry.target\n .querySelectorAll(\".shiny-bound-output\")\n .forEach((el) => {\n if (resized.includes(el)) return;\n\n const { binding, onResize } = $(el).data(\"shinyOutputBinding\");\n if (!binding || !binding.resize) return;\n\n // if this output is owned by another observer, skip it\n const owner = (el as any).shinyResizeObserver;\n if (owner && owner !== this) return;\n // mark this output as owned by this shinyResizeObserver instance\n if (!owner) (el as any).shinyResizeObserver = this;\n\n // trigger immediate resizing of outputs with a resize method\n onResize(el);\n // only once per output and resize event\n resized.push(el);\n\n // set plot images to 100% width temporarily during the transition\n if (!el.classList.contains(\"shiny-plot-output\")) return;\n const img = el.querySelector(\n 'img:not([width=\"100%\"])'\n );\n if (img) img.setAttribute(\"width\", \"100%\");\n });\n }\n });\n }\n\n /**\n * Observe an element for size changes.\n * @param {HTMLElement} el - The element to observe.\n */\n observe(el: HTMLElement): void {\n this.resizeObserver.observe(el);\n this.resizeObserverEntries.push(el);\n }\n\n /**\n * Stop observing an element for size changes.\n * @param {HTMLElement} el - The element to stop observing.\n */\n unobserve(el: HTMLElement): void {\n const idxEl = this.resizeObserverEntries.indexOf(el);\n if (idxEl < 0) return;\n\n this.resizeObserver.unobserve(el);\n this.resizeObserverEntries.splice(idxEl, 1);\n }\n\n /**\n * This method checks that we're not continuing to watch elements that no\n * longer exist in the DOM. If any are found, we stop observing them and\n * remove them from our array of observed elements.\n *\n * @private\n * @static\n */\n flush(): void {\n this.resizeObserverEntries.forEach((el) => {\n if (!document.body.contains(el)) this.unobserve(el);\n });\n }\n}\n\nexport { ShinyResizeObserver };\n", "import { InputBinding, registerBinding } from \"./_utils\";\nimport { ShinyResizeObserver } from \"./_shinyResizeObserver\";\n\n/**\n * Methods for programmatically toggling the state of the sidebar. These methods\n * describe the desired state of the sidebar: `\"close\"` and `\"open\"` transition\n * the sidebar to the desired state, unless the sidebar is already in that\n * state. `\"toggle\"` transitions the sidebar to the state opposite of its\n * current state.\n * @typedef {SidebarToggleMethod}\n */\ntype SidebarToggleMethod = \"close\" | \"open\" | \"toggle\";\n\n/**\n * Data received by the input binding's `receiveMessage` method.\n * @typedef {SidebarMessageData}\n */\ntype SidebarMessageData = {\n method: SidebarToggleMethod;\n};\n\n/**\n * The DOM elements that make up the sidebar. `main`, `sidebar`, and `toggle`\n * are all direct children of `container` (in that order).\n * @interface SidebarComponents\n * @typedef {SidebarComponents}\n */\ninterface SidebarComponents {\n /**\n * The `layout_sidebar()` parent container, with class\n * `Sidebar.classes.LAYOUT`.\n * @type {HTMLElement}\n */\n container: HTMLElement;\n /**\n * The main content area of the sidebar layout.\n * @type {HTMLElement}\n */\n main: HTMLElement;\n /**\n * The sidebar container of the sidebar layout.\n * @type {HTMLElement}\n */\n sidebar: HTMLElement;\n /**\n * The toggle button that is used to toggle the sidebar state.\n * @type {HTMLElement}\n */\n toggle: HTMLElement;\n}\n\n/**\n * The bslib sidebar component class. This class is only used for collapsible\n * sidebars.\n *\n * @class Sidebar\n * @typedef {Sidebar}\n */\nclass Sidebar {\n /**\n * The DOM elements that make up the sidebar, see `SidebarComponents`.\n * @private\n * @type {SidebarComponents}\n */\n private layout: SidebarComponents;\n\n /**\n * A Shiny-specific resize observer that ensures Shiny outputs in the main\n * content areas of the sidebar resize appropriately.\n * @private\n * @type {ShinyResizeObserver}\n * @static\n */\n private static shinyResizeObserver = new ShinyResizeObserver();\n\n /**\n * Creates an instance of a collapsible bslib Sidebar.\n * @constructor\n * @param {HTMLElement} container\n */\n constructor(container: HTMLElement) {\n Sidebar.instanceMap.set(container, this);\n this.layout = {\n container,\n main: container.querySelector(\":scope > .main\") as HTMLElement,\n sidebar: container.querySelector(\":scope > .sidebar\") as HTMLElement,\n toggle: container.querySelector(\n \":scope > .collapse-toggle\"\n ) as HTMLElement,\n } as SidebarComponents;\n\n if (!this.layout.toggle) {\n throw new Error(\"Tried to initialize a non-collapsible sidebar.\");\n }\n\n this._initEventListeners();\n this._initSidebarCounters();\n this._initDesktop();\n\n // Start watching the main content area for size changes to ensure Shiny\n // outputs resize appropriately during sidebar transitions.\n Sidebar.shinyResizeObserver.observe(this.layout.main);\n\n container.removeAttribute(\"data-bslib-sidebar-init\");\n const initScript = container.querySelector(\n \":scope > script[data-bslib-sidebar-init]\"\n );\n if (initScript) {\n container.removeChild(initScript);\n }\n }\n\n /**\n * Read the current state of the sidebar. Note that, when calling this method,\n * the sidebar may be transitioning into the state returned by this method.\n *\n * @description\n * The sidebar state works as follows, starting from the open state. When the\n * sidebar is closed:\n * 1. We add both the `COLLAPSE` and `TRANSITIONING` classes to the sidebar.\n * 2. The sidebar collapse begins to animate. On desktop devices, and where it\n * is supported, we transition the `grid-template-columns` property of the\n * sidebar layout. On mobile, the sidebar is hidden immediately. In both\n * cases, the collapse icon rotates and we use this rotation to determine\n * when the transition is complete.\n * 3. If another sidebar state toggle is requested while closing the sidebar,\n * we remove the `COLLAPSE` class and the animation immediately starts to\n * reverse.\n * 4. When the `transition` is complete, we remove the `TRANSITIONING` class.\n * @readonly\n * @type {boolean}\n */\n get isClosed(): boolean {\n return this.layout.container.classList.contains(Sidebar.classes.COLLAPSE);\n }\n\n /**\n * Static classes related to the sidebar layout or state.\n * @public\n * @static\n * @readonly\n * @type {{ LAYOUT: string; COLLAPSE: string; TRANSITIONING: string; }}\n */\n public static readonly classes = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n LAYOUT: \"bslib-sidebar-layout\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n COLLAPSE: \"sidebar-collapsed\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n TRANSITIONING: \"transitioning\",\n };\n\n /**\n * If sidebars are initialized before the DOM is ready, we re-schedule the\n * initialization to occur on DOMContentLoaded.\n * @private\n * @static\n * @type {boolean}\n */\n private static onReadyScheduled = false;\n /**\n * A map of initialized sidebars to their respective Sidebar instances.\n * @private\n * @static\n * @type {WeakMap}\n */\n private static instanceMap: WeakMap = new WeakMap();\n\n /**\n * Given a sidebar container, return the Sidebar instance associated with it.\n * @public\n * @static\n * @param {HTMLElement} el\n * @returns {(Sidebar | undefined)}\n */\n public static getInstance(el: HTMLElement): Sidebar | undefined {\n return Sidebar.instanceMap.get(el);\n }\n\n /**\n * Initialize all collapsible sidebars on the page.\n * @public\n * @static\n * @param {boolean} [flushResizeObserver=true] When `true`, we remove\n * non-existent elements from the ResizeObserver. This is required\n * periodically to prevent memory leaks. To avoid over-checking, we only flush\n * the ResizeObserver when initializing sidebars after page load.\n */\n public static initCollapsibleAll(flushResizeObserver = true): void {\n if (document.readyState === \"loading\") {\n if (!Sidebar.onReadyScheduled) {\n Sidebar.onReadyScheduled = true;\n document.addEventListener(\"DOMContentLoaded\", () => {\n Sidebar.initCollapsibleAll(false);\n });\n }\n return;\n }\n\n const initSelector = `.${Sidebar.classes.LAYOUT}[data-bslib-sidebar-init]`;\n if (!document.querySelector(initSelector)) {\n // no sidebars to initialize\n return;\n }\n\n if (flushResizeObserver) Sidebar.shinyResizeObserver.flush();\n\n const containers = document.querySelectorAll(initSelector);\n containers.forEach((container) => new Sidebar(container as HTMLElement));\n }\n\n /**\n * Initialize event listeners for the sidebar toggle button.\n * @private\n */\n private _initEventListeners(): void {\n const { toggle } = this.layout;\n\n toggle.addEventListener(\"click\", (ev) => {\n ev.preventDefault();\n this.toggle(\"toggle\");\n });\n\n // Remove the transitioning class when the transition ends. We watch the\n // collapse toggle icon because it's guaranteed to transition, whereas the\n // sidebar doesn't animate on mobile (or in browsers where animating\n // grid-template-columns is not supported).\n toggle\n .querySelector(\".collapse-icon\")\n ?.addEventListener(\"transitionend\", () => this._finalizeState());\n }\n\n /**\n * Initialize nested sidebar counters.\n *\n * @description\n * This function walks up the DOM tree, adding CSS variables to each direct\n * parent sidebar layout that count the layout's position in the stack of\n * nested layouts. We use these counters to keep the collapse toggles from\n * overlapping. Note that always-open sidebars that don't have collapse\n * toggles break the chain of nesting.\n * @private\n */\n private _initSidebarCounters(): void {\n const { container } = this.layout;\n\n const selectorChildLayouts =\n `.${Sidebar.classes.LAYOUT}` +\n \"> .main > \" +\n `.${Sidebar.classes.LAYOUT}:not([data-bslib-sidebar-open=\"always\"])`;\n\n const isInnermostLayout =\n container.querySelector(selectorChildLayouts) === null;\n\n if (!isInnermostLayout) {\n // There are sidebar layouts nested within this layout; defer to children\n return;\n }\n\n function nextSidebarParent(el: HTMLElement | null): HTMLElement | null {\n el = el ? el.parentElement : null;\n if (el && el.classList.contains(\"main\")) {\n // .bslib-sidebar-layout > .main > .bslib-sidebar-layout\n el = el.parentElement;\n }\n if (el && el.classList.contains(Sidebar.classes.LAYOUT)) {\n return el;\n }\n return null;\n }\n\n const layouts = [container];\n let parent = nextSidebarParent(container);\n\n while (parent) {\n // Add parent to front of layouts array, so we sort outer -> inner\n layouts.unshift(parent);\n parent = nextSidebarParent(parent);\n }\n\n const count = { left: 0, right: 0 };\n layouts.forEach(function (x: HTMLElement, i: number): void {\n x.style.setProperty(\"--bslib-sidebar-counter\", i.toString());\n const isRight = x.classList.contains(\"sidebar-right\");\n const thisCount = isRight ? count.right++ : count.left++;\n x.style.setProperty(\n \"--bslib-sidebar-overlap-counter\",\n thisCount.toString()\n );\n });\n }\n\n /**\n * Initialize the sidebar's initial state when `open = \"desktop\"`.\n * @private\n */\n private _initDesktop(): void {\n const { container } = this.layout;\n // If sidebar is marked open='desktop'...\n if (container.dataset.bslibSidebarOpen?.trim() !== \"desktop\") {\n return;\n }\n\n // then close sidebar on mobile\n const initCollapsed = window\n .getComputedStyle(container)\n .getPropertyValue(\"--bslib-sidebar-js-init-collapsed\");\n\n if (initCollapsed.trim() === \"true\") {\n this.toggle(\"close\");\n }\n }\n\n /**\n * Toggle the sidebar's open/closed state.\n * @public\n * @param {SidebarToggleMethod | undefined} method Whether to `\"open\"`,\n * `\"close\"` or `\"toggle\"` the sidebar. If `.toggle()` is called without an\n * argument, it will toggle the sidebar's state.\n */\n public toggle(method: SidebarToggleMethod | undefined): void {\n if (typeof method === \"undefined\") {\n method = \"toggle\";\n }\n\n const { container, sidebar } = this.layout;\n const isClosed = this.isClosed;\n\n if ([\"open\", \"close\", \"toggle\"].indexOf(method) === -1) {\n throw new Error(`Unknown method ${method}`);\n }\n\n if (method === \"toggle\") {\n method = isClosed ? \"open\" : \"close\";\n }\n\n if ((isClosed && method === \"close\") || (!isClosed && method === \"open\")) {\n // nothing to do, sidebar is already in the desired state\n return;\n }\n\n if (method === \"open\") {\n // unhide sidebar immediately when opening,\n // otherwise the sidebar is hidden on transitionend\n sidebar.hidden = false;\n }\n\n // Add a transitioning class just before adding COLLAPSE_CLASS since we want\n // some of the transitioning styles to apply before the collapse state\n container.classList.add(Sidebar.classes.TRANSITIONING);\n container.classList.toggle(Sidebar.classes.COLLAPSE);\n }\n\n /**\n * When the sidebar open/close transition ends, finalize the sidebar's state.\n * @private\n */\n private _finalizeState(): void {\n const { container, sidebar, toggle } = this.layout;\n container.classList.remove(Sidebar.classes.TRANSITIONING);\n sidebar.hidden = this.isClosed;\n toggle.ariaExpanded = this.isClosed ? \"false\" : \"true\";\n\n // Send browser-native event with updated sidebar state\n const event = new CustomEvent(\"bslib.sidebar\", {\n bubbles: true,\n detail: { open: !this.isClosed },\n });\n sidebar.dispatchEvent(event);\n\n // Trigger Shiny input and output binding events\n $(sidebar).trigger(\"toggleCollapse.sidebarInputBinding\");\n $(sidebar).trigger(this.isClosed ? \"hidden\" : \"shown\");\n }\n}\n\n/**\n * A Shiny input binding for a sidebar.\n * @class SidebarInputBinding\n * @typedef {SidebarInputBinding}\n * @extends {InputBinding}\n */\nclass SidebarInputBinding extends InputBinding {\n find(scope: HTMLElement) {\n return $(scope).find(`.${Sidebar.classes.LAYOUT} > .bslib-sidebar-input`);\n }\n\n getValue(el: HTMLElement): boolean {\n const sb = Sidebar.getInstance(el.parentElement as HTMLElement);\n if (!sb) return false;\n return !sb.isClosed;\n }\n\n setValue(el: HTMLElement, value: boolean): void {\n const method = value ? \"open\" : \"close\";\n this.receiveMessage(el, { method });\n }\n\n subscribe(el: HTMLElement, callback: (x: boolean) => void) {\n $(el).on(\n \"toggleCollapse.sidebarInputBinding\",\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n function (event) {\n callback(true);\n }\n );\n }\n\n unsubscribe(el: HTMLElement) {\n $(el).off(\".sidebarInputBinding\");\n }\n\n receiveMessage(el: HTMLElement, data: SidebarMessageData) {\n const sb = Sidebar.getInstance(el.parentElement as HTMLElement);\n if (sb) sb.toggle(data.method);\n }\n}\n\nregisterBinding(SidebarInputBinding, \"sidebar\");\n\n// attach Sidebar class to window for global usage\n(window as any).bslib = (window as any).bslib || {};\n(window as any).bslib.Sidebar = Sidebar;\n"], - "mappings": "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", - "names": ["InputBinding", "registerBinding", "inputBindingClass", "name", "ShinyResizeObserver", "entries", "resizeEvent", "resized", "entry", "el", "binding", "onResize", "owner", "img", "idxEl", "_Sidebar", "container", "initScript", "el", "flushResizeObserver", "initSelector", "_a", "toggle", "ev", "selectorChildLayouts", "nextSidebarParent", "layouts", "parent", "count", "x", "i", "thisCount", "method", "sidebar", "isClosed", "event", "Sidebar", "ShinyResizeObserver", "SidebarInputBinding", "InputBinding", "scope", "sb", "value", "callback", "data", "registerBinding"] -} diff --git a/shiny/experimental/www/bslib/components/sidebar/sidebar.css b/shiny/experimental/www/bslib/components/sidebar/sidebar.css new file mode 100644 index 000000000..319af78f7 --- /dev/null +++ b/shiny/experimental/www/bslib/components/sidebar/sidebar.css @@ -0,0 +1 @@ +.bslib-sidebar-layout{--bslib-sidebar-transition-duration: 500ms;--bslib-sidebar-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, var(--bs-border-color-translucent));--bslib-sidebar-border-radius: var(--bs-border-radius);--bslib-sidebar-vert-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, var(--bs-border-color-translucent));--bslib-sidebar-bg: #f7f7f7;--bslib-sidebar-fg: #000;--bslib-sidebar-padding: calc(var(--bslib-spacer) * 1.5);--bslib-sidebar-icon-size: var(--bslib-spacer, 1rem);--bslib-collapse-toggle-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, var(--bs-border-color-translucent));--bslib-collapse-toggle-transform: 90deg;--bslib-collapse-toggle-right-transform: -90deg;--bslib-sidebar-column-main: minmax(0, 1fr);display:grid !important;grid-template-columns:Min(calc(100% - var(--bslib-sidebar-icon-size)), var(--bslib-sidebar-width, 250px)) var(--bslib-sidebar-column-main);position:relative;transition:grid-template-columns ease-in-out var(--bslib-sidebar-transition-duration);border:var(--bslib-sidebar-border);border-radius:var(--bslib-sidebar-border-radius)}@media (prefers-reduced-motion: reduce){.bslib-sidebar-layout{transition:none}}.bslib-sidebar-layout[data-bslib-sidebar-border="false"]{border:none}.bslib-sidebar-layout[data-bslib-sidebar-border-radius="false"]{border-radius:initial}.bslib-sidebar-layout>.main,.bslib-sidebar-layout>.sidebar{grid-row:1 / 2;border-radius:inherit;overflow:auto}.bslib-sidebar-layout>.main{grid-column:2 / 3;border-top-left-radius:0;border-bottom-left-radius:0;padding:var(--bslib-sidebar-padding)}.bslib-sidebar-layout>.sidebar{grid-column:1 / 2;width:100%;height:100%;border-right:var(--bslib-sidebar-vert-border);border-top-right-radius:0;border-bottom-right-radius:0;background-color:var(--bslib-sidebar-bg);color:var(--bslib-sidebar-fg)}.bslib-sidebar-layout>.sidebar>.sidebar-content{display:flex;flex-direction:column;padding:var(--bslib-sidebar-padding)}.bslib-sidebar-layout>.sidebar>.sidebar-content>:last-child:not(.sidebar-title){margin-bottom:0}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion{margin-left:calc(-1 * var(--bslib-sidebar-padding));margin-right:calc(-1 * var(--bslib-sidebar-padding))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:first-child{margin-top:calc(-1 * var(--bslib-sidebar-padding))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:last-child{margin-bottom:calc(-1 * var(--bslib-sidebar-padding))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:last-child){margin-bottom:1rem}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion .accordion-body{display:flex;flex-direction:column}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:first-child) .accordion-item:first-child{border-top:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.bslib-sidebar-layout>.sidebar>.sidebar-content>.accordion:not(:last-child) .accordion-item:last-child{border-bottom:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.bslib-sidebar-layout>.sidebar>.sidebar-content>.sidebar-title+.accordion{margin-top:calc(-1rem - var(--bs-card-border-width, 1px))}.bslib-sidebar-layout>.sidebar>.sidebar-content>.sidebar-title:has(+.accordion){border-bottom:none}.bslib-sidebar-layout>.sidebar .shiny-input-container{width:100%}.bslib-sidebar-layout>.collapse-toggle{grid-row:1 / 2;grid-column:1 / 2;display:inline-flex;align-items:center;position:absolute;right:calc(-1 * var(--bslib-sidebar-icon-size));bottom:calc(var(--bslib-sidebar-padding) + var(--bslib-sidebar-overlap-counter, 0) * calc(var(--bslib-sidebar-icon-size) + var(--bslib-sidebar-padding)));border:var(--bslib-collapse-toggle-border);border-left:none;border-radius:0 var(--bs-border-radius) var(--bs-border-radius) 0;padding:7px 0;background-color:var(--bslib-sidebar-bg);color:var(--bslib-sidebar-fg)}.bslib-sidebar-layout>.collapse-toggle>.collapse-icon{opacity:0.8;width:var(--bslib-sidebar-icon-size);height:var(--bslib-sidebar-icon-size);transform:rotate(var(--bslib-collapse-toggle-transform));transition:transform cubic-bezier(0.68, -0.55, 0.27, 1.55) var(--bslib-sidebar-transition-duration)}.bslib-sidebar-layout>.collapse-toggle:hover>.collapse-icon{opacity:1}.bslib-sidebar-layout .sidebar-title{font-size:1.25rem;line-height:1.25;margin-top:0;margin-bottom:1rem;padding-bottom:1rem;border-bottom:var(--bslib-sidebar-border)}.bslib-sidebar-layout.sidebar-right{grid-template-columns:var(--bslib-sidebar-column-main) Min(calc(100% - var(--bslib-sidebar-icon-size)), var(--bslib-sidebar-width, 250px))}.bslib-sidebar-layout.sidebar-right>.main{grid-column:1 / 2;border-top-right-radius:0;border-bottom-right-radius:0;border-top-left-radius:inherit;border-bottom-left-radius:inherit}.bslib-sidebar-layout.sidebar-right>.sidebar{grid-column:2 / 3;border-right:none;border-left:var(--bslib-sidebar-vert-border);border-top-left-radius:0;border-bottom-left-radius:0}.bslib-sidebar-layout.sidebar-right>.collapse-toggle{grid-column:2 / 3;left:calc(-1 * var(--bslib-sidebar-icon-size));right:unset;border-radius:var(--bs-border-radius) 0 0 var(--bs-border-radius);border-right:none;border-left:var(--bslib-collapse-toggle-border)}.bslib-sidebar-layout.sidebar-right>.collapse-toggle>.collapse-icon{transform:rotate(var(--bslib-collapse-toggle-right-transform))}.bslib-sidebar-layout.sidebar-collapsed{--bslib-collapse-toggle-transform: -90deg;--bslib-collapse-toggle-right-transform: 90deg;--bslib-sidebar-vert-border: none;grid-template-columns:0 minmax(0, 1fr)}.bslib-sidebar-layout.sidebar-collapsed.sidebar-right{grid-template-columns:minmax(0, 1fr) 0}.bslib-sidebar-layout.sidebar-collapsed:not(.transitioning)>.sidebar>*{display:none}.bslib-sidebar-layout.sidebar-collapsed>.main{border-radius:inherit}.bslib-sidebar-layout.sidebar-collapsed>.collapse-toggle{right:calc(-1 * var(--bslib-sidebar-icon-size) - var(--bs-card-border-width, 1px))}.bslib-sidebar-layout.sidebar-collapsed.sidebar-right>.collapse-toggle{left:calc(-1 * var(--bslib-sidebar-icon-size) - var(--bs-card-border-width, 1px));right:unset}@media (min-width: 576px){.bslib-sidebar-layout.transitioning>.sidebar>.sidebar-content{display:none}}@media (max-width: 575.98px){.bslib-sidebar-layout,.bslib-sidebar-layout.sidebar-right{--bslib-sidebar-vert-border: none;--bslib-sidebar-horiz-border: var(--bs-card-border-width, 1px) solid var(--bs-card-border-color, var(--bs-border-color-translucent));--bslib-collapse-toggle-transform: -180deg;--bslib-collapse-toggle-right-transform: -180deg;grid-template-columns:1fr !important;grid-template-rows:fit-content(var(--bslib-sidebar-max-height-mobile, auto)) minmax(0, 1fr)}.bslib-sidebar-layout[data-bslib-sidebar-open="desktop"],.bslib-sidebar-layout.sidebar-right[data-bslib-sidebar-open="desktop"]{--bslib-sidebar-js-init-collapsed: true}.bslib-sidebar-layout>.sidebar,.bslib-sidebar-layout.sidebar-right>.sidebar{grid-row:1 / 2;grid-column:1 / 2;width:100%;border:none;border-bottom:var(--bslib-sidebar-horiz-border);border-radius:0}.bslib-sidebar-layout>.main,.bslib-sidebar-layout.sidebar-right>.main{grid-row:2 / 3;grid-column:1 / 2;border-top-left-radius:0;border-top-right-radius:0;border-bottom-right-radius:inherit;border-bottom-left-radius:inherit}.bslib-sidebar-layout>.collapse-toggle,.bslib-sidebar-layout.sidebar-right>.collapse-toggle{grid-row:2 / 3;grid-column:1 / 2;border-top:none !important;border:var(--bslib-collapse-toggle-border);border-radius:0 0 var(--bs-border-radius) var(--bs-border-radius);padding:0 4px}.bslib-sidebar-layout>.collapse-toggle>.collapse-icon,.bslib-sidebar-layout.sidebar-right>.collapse-toggle>.collapse-icon{transition-duration:calc(var(--bslib-sidebar-transition-duration) * 0.33)}.bslib-sidebar-layout>.collapse-toggle,.bslib-sidebar-layout.sidebar-right>.collapse-toggle,.bslib-sidebar-layout.sidebar-right>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-right>.collapse-toggle{top:calc(-1 * var(--bs-card-border-width, 1px))}.bslib-sidebar-layout.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-right.sidebar-collapsed>.collapse-toggle{top:0}.bslib-sidebar-layout>.collapse-toggle,.bslib-sidebar-layout.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-right>.collapse-toggle,.bslib-sidebar-layout.sidebar-right.sidebar-right.sidebar-collapsed>.collapse-toggle{right:calc(var(--bslib-sidebar-padding) + var(--bslib-sidebar-counter, 0) * calc(var(--bslib-sidebar-icon-size) + var(--bslib-sidebar-padding)));bottom:initial;left:initial}.bslib-sidebar-layout.sidebar-collapsed,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed{--bslib-collapse-toggle-transform: 0deg;--bslib-collapse-toggle-right-transform: 0deg;grid-template-rows:0 minmax(0, 1fr)}.bslib-sidebar-layout.sidebar-collapsed>.main,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.main{border-top-left-radius:inherit;border-top-right-radius:inherit}.bslib-sidebar-layout.sidebar-collapsed>.sidebar,.bslib-sidebar-layout.sidebar-right.sidebar-collapsed>.sidebar{border-bottom:none}} diff --git a/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js b/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js new file mode 100644 index 000000000..025c9fe5e --- /dev/null +++ b/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js @@ -0,0 +1,3 @@ +/*! bslib 0.5.0.9000 | (c) 2012-2023 RStudio, PBC. | License: MIT + file LICENSE */ +"use strict";(()=>{var g=window.Shiny?Shiny.InputBinding:class{};function f(b,e){window.Shiny&&Shiny.inputBindings.register(new b,"bslib."+e)}var u=class{constructor(){this.resizeObserverEntries=[],this.resizeObserver=new ResizeObserver(e=>{let t=new Event("resize");if(window.dispatchEvent(t),!window.Shiny)return;let i=[];for(let r of e)r.target instanceof HTMLElement&&r.target.querySelector(".shiny-bound-output")&&r.target.querySelectorAll(".shiny-bound-output").forEach(o=>{if(i.includes(o))return;let{binding:l,onResize:d}=$(o).data("shinyOutputBinding");if(!l||!l.resize)return;let s=o.shinyResizeObserver;if(s&&s!==this||(s||(o.shinyResizeObserver=this),d(o),i.push(o),!o.classList.contains("shiny-plot-output")))return;let c=o.querySelector('img:not([width="100%"])');c&&c.setAttribute("width","100%")})})}observe(e){this.resizeObserver.observe(e),this.resizeObserverEntries.push(e)}unobserve(e){let t=this.resizeObserverEntries.indexOf(e);t<0||(this.resizeObserver.unobserve(e),this.resizeObserverEntries.splice(t,1))}flush(){this.resizeObserverEntries.forEach(e=>{document.body.contains(e)||this.unobserve(e)})}};var n=class{constructor(e){if(n.instanceMap.set(e,this),this.layout={container:e,main:e.querySelector(":scope > .main"),sidebar:e.querySelector(":scope > .sidebar"),toggle:e.querySelector(":scope > .collapse-toggle")},!this.layout.toggle)throw new Error("Tried to initialize a non-collapsible sidebar.");let t=this.layout.sidebar.querySelector(":scope > .sidebar-content > .accordion");t&&t.classList.add("accordion-flush"),this._initEventListeners(),this._initSidebarCounters(),this._initDesktop(),n.shinyResizeObserver.observe(this.layout.main),e.removeAttribute("data-bslib-sidebar-init");let i=e.querySelector(":scope > script[data-bslib-sidebar-init]");i&&e.removeChild(i)}get isClosed(){return this.layout.container.classList.contains(n.classes.COLLAPSE)}static getInstance(e){return n.instanceMap.get(e)}static initCollapsibleAll(e=!0){if(document.readyState==="loading"){n.onReadyScheduled||(n.onReadyScheduled=!0,document.addEventListener("DOMContentLoaded",()=>{n.initCollapsibleAll(!1)}));return}let t=`.${n.classes.LAYOUT}[data-bslib-sidebar-init]`;if(!document.querySelector(t))return;e&&n.shinyResizeObserver.flush(),document.querySelectorAll(t).forEach(r=>new n(r))}_initEventListeners(){var t;let{toggle:e}=this.layout;e.addEventListener("click",i=>{i.preventDefault(),this.toggle("toggle")}),(t=e.querySelector(".collapse-icon"))==null||t.addEventListener("transitionend",()=>this._finalizeState())}_initSidebarCounters(){let{container:e}=this.layout,t=`.${n.classes.LAYOUT}> .main > .${n.classes.LAYOUT}:not([data-bslib-sidebar-open="always"])`;if(!(e.querySelector(t)===null))return;function r(s){return s=s?s.parentElement:null,s&&s.classList.contains("main")&&(s=s.parentElement),s&&s.classList.contains(n.classes.LAYOUT)?s:null}let o=[e],l=r(e);for(;l;)o.unshift(l),l=r(l);let d={left:0,right:0};o.forEach(function(s,c){s.style.setProperty("--bslib-sidebar-counter",c.toString());let h=s.classList.contains("sidebar-right")?d.right++:d.left++;s.style.setProperty("--bslib-sidebar-overlap-counter",h.toString())})}_initDesktop(){var i;let{container:e}=this.layout;if(((i=e.dataset.bslibSidebarOpen)==null?void 0:i.trim())!=="desktop")return;window.getComputedStyle(e).getPropertyValue("--bslib-sidebar-js-init-collapsed").trim()==="true"&&this.toggle("close")}toggle(e){typeof e=="undefined"&&(e="toggle");let{container:t,sidebar:i}=this.layout,r=this.isClosed;if(["open","close","toggle"].indexOf(e)===-1)throw new Error(`Unknown method ${e}`);e==="toggle"&&(e=r?"open":"close"),!(r&&e==="close"||!r&&e==="open")&&(e==="open"&&(i.hidden=!1),t.classList.add(n.classes.TRANSITIONING),t.classList.toggle(n.classes.COLLAPSE))}_finalizeState(){let{container:e,sidebar:t,toggle:i}=this.layout;e.classList.remove(n.classes.TRANSITIONING),t.hidden=this.isClosed,i.ariaExpanded=this.isClosed?"false":"true";let r=new CustomEvent("bslib.sidebar",{bubbles:!0,detail:{open:!this.isClosed}});t.dispatchEvent(r),$(t).trigger("toggleCollapse.sidebarInputBinding"),$(t).trigger(this.isClosed?"hidden":"shown")}},a=n;a.shinyResizeObserver=new u,a.classes={LAYOUT:"bslib-sidebar-layout",COLLAPSE:"sidebar-collapsed",TRANSITIONING:"transitioning"},a.onReadyScheduled=!1,a.instanceMap=new WeakMap;var p=class extends g{find(e){return $(e).find(`.${a.classes.LAYOUT} > .bslib-sidebar-input`)}getValue(e){let t=a.getInstance(e.parentElement);return t?!t.isClosed:!1}setValue(e,t){let i=t?"open":"close";this.receiveMessage(e,{method:i})}subscribe(e,t){$(e).on("toggleCollapse.sidebarInputBinding",function(i){t(!0)})}unsubscribe(e){$(e).off(".sidebarInputBinding")}receiveMessage(e,t){let i=a.getInstance(e.parentElement);i&&i.toggle(t.method)}};f(p,"sidebar");window.bslib=window.bslib||{};window.bslib.Sidebar=a;})(); +//# sourceMappingURL=sidebar.min.js.map diff --git a/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js.map b/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js.map new file mode 100644 index 000000000..df29f8075 --- /dev/null +++ b/shiny/experimental/www/bslib/components/sidebar/sidebar.min.js.map @@ -0,0 +1,7 @@ +{ + "version": 3, + "sources": ["../../../../srcts/src/components/_utils.ts", "../../../../srcts/src/components/_shinyResizeObserver.ts", "../../../../srcts/src/components/sidebar.ts"], + "sourcesContent": ["import type { HtmlDep } from \"rstudio-shiny/srcts/types/src/shiny/render\";\n\nimport type { InputBinding as InputBindingType } from \"rstudio-shiny/srcts/types/src/bindings/input\";\n\n// Exclude undefined from T\ntype NotUndefined = T extends undefined ? never : T;\n\n// eslint-disable-next-line @typescript-eslint/naming-convention\nconst InputBinding = (\n window.Shiny ? Shiny.InputBinding : class {}\n) as typeof InputBindingType;\n\nfunction registerBinding(\n inputBindingClass: new () => InputBindingType,\n name: string\n): void {\n if (window.Shiny) {\n Shiny.inputBindings.register(new inputBindingClass(), \"bslib.\" + name);\n }\n}\n\n// Return true if the key exists on the object and the value is not undefined.\n//\n// This method is mainly used in input bindings' `receiveMessage` method.\n// Since we know that the values are sent by Shiny via `{jsonlite}`,\n// then we know that there are no `undefined` values. `null` is possible, but not `undefined`.\nfunction hasDefinedProperty<\n Prop extends keyof X,\n X extends { [key: string]: any }\n>(\n obj: X,\n prop: Prop\n): obj is X & { [key in NonNullable]: NotUndefined } {\n return (\n Object.prototype.hasOwnProperty.call(obj, prop) && obj[prop] !== undefined\n );\n}\n\n// TODO: Shiny should trigger resize events when the output\n// https://github.com/rstudio/shiny/pull/3682\nfunction doWindowResizeOnElementResize(el: HTMLElement): void {\n if ($(el).data(\"window-resize-observer\")) {\n return;\n }\n const resizeEvent = new Event(\"resize\");\n const ro = new ResizeObserver(() => {\n window.dispatchEvent(resizeEvent);\n });\n ro.observe(el);\n $(el).data(\"window-resize-observer\", ro);\n}\n\nfunction getAllFocusableChildren(el: HTMLElement): HTMLElement[] {\n // Cross-referenced with https://allyjs.io/data-tables/focusable.html\n const base = [\n \"a[href]\",\n \"area[href]\",\n \"button\",\n \"details summary\",\n \"input\",\n \"iframe\",\n \"select\",\n \"textarea\",\n '[contentEditable=\"\"]',\n '[contentEditable=\"true\"]',\n '[contentEditable=\"TRUE\"]',\n \"[tabindex]\",\n ];\n const modifiers = [':not([tabindex=\"-1\"])', \":not([disabled])\"];\n const selectors = base.map((b) => b + modifiers.join(\"\"));\n const focusable = el.querySelectorAll(selectors.join(\", \"));\n return Array.from(focusable) as HTMLElement[];\n}\n\nexport {\n InputBinding,\n registerBinding,\n hasDefinedProperty,\n doWindowResizeOnElementResize,\n getAllFocusableChildren,\n};\nexport type { HtmlDep };\n", "/**\n * A resize observer that ensures Shiny outputs resize during or just after\n * their parent container size changes. Useful, in particular, for sidebar\n * transitions or for full-screen card transitions.\n *\n * @class ShinyResizeObserver\n * @typedef {ShinyResizeObserver}\n */\nclass ShinyResizeObserver {\n /**\n * The actual ResizeObserver instance.\n * @private\n * @type {ResizeObserver}\n */\n private resizeObserver: ResizeObserver;\n /**\n * An array of elements that are currently being watched by the Resize\n * Observer.\n *\n * @details\n * We don't currently have lifecycle hooks that allow us to unobserve elements\n * when they are removed from the DOM. As a result, we need to manually check\n * that the elements we're watching still exist in the DOM. This array keeps\n * track of the elements we're watching so that we can check them later.\n * @private\n * @type {HTMLElement[]}\n */\n private resizeObserverEntries: HTMLElement[];\n\n /**\n * Watch containers for size changes and ensure that Shiny outputs and\n * htmlwidgets within resize appropriately.\n *\n * @details\n * The ShinyResizeObserver is used to watch the containers, such as Sidebars\n * and Cards for size changes, in particular when the sidebar state is toggled\n * or the card body is expanded full screen. It performs two primary tasks:\n *\n * 1. Dispatches a `resize` event on the window object. This is necessary to\n * ensure that Shiny outputs resize appropriately. In general, the window\n * resizing is throttled and the output update occurs when the transition\n * is complete.\n * 2. If an output with a resize method on the output binding is detected, we\n * directly call the `.onResize()` method of the binding. This ensures that\n * htmlwidgets transition smoothly. In static mode, htmlwidgets does this\n * already.\n *\n * @note\n * This resize observer also handles race conditions in some complex\n * fill-based layouts with multiple outputs (e.g., plotly), where shiny\n * initializes with the correct sizing, but in-between the 1st and last\n * renderValue(), the size of the output containers can change, meaning every\n * output but the 1st gets initialized with the wrong size during their\n * renderValue(). Then, after the render phase, shiny won't know to trigger a\n * resize since all the widgets will return to their original size (and thus,\n * Shiny thinks there isn't any resizing to do). The resize observer works\n * around this by ensuring that the output is resized whenever its container\n * size changes.\n * @constructor\n */\n constructor() {\n this.resizeObserverEntries = [];\n this.resizeObserver = new ResizeObserver((entries) => {\n const resizeEvent = new Event(\"resize\");\n window.dispatchEvent(resizeEvent);\n\n // the rest of this callback is only relevant in Shiny apps\n if (!window.Shiny) return;\n\n const resized = [] as HTMLElement[];\n\n for (const entry of entries) {\n if (!(entry.target instanceof HTMLElement)) continue;\n if (!entry.target.querySelector(\".shiny-bound-output\")) continue;\n\n entry.target\n .querySelectorAll(\".shiny-bound-output\")\n .forEach((el) => {\n if (resized.includes(el)) return;\n\n const { binding, onResize } = $(el).data(\"shinyOutputBinding\");\n if (!binding || !binding.resize) return;\n\n // if this output is owned by another observer, skip it\n const owner = (el as any).shinyResizeObserver;\n if (owner && owner !== this) return;\n // mark this output as owned by this shinyResizeObserver instance\n if (!owner) (el as any).shinyResizeObserver = this;\n\n // trigger immediate resizing of outputs with a resize method\n onResize(el);\n // only once per output and resize event\n resized.push(el);\n\n // set plot images to 100% width temporarily during the transition\n if (!el.classList.contains(\"shiny-plot-output\")) return;\n const img = el.querySelector(\n 'img:not([width=\"100%\"])'\n );\n if (img) img.setAttribute(\"width\", \"100%\");\n });\n }\n });\n }\n\n /**\n * Observe an element for size changes.\n * @param {HTMLElement} el - The element to observe.\n */\n observe(el: HTMLElement): void {\n this.resizeObserver.observe(el);\n this.resizeObserverEntries.push(el);\n }\n\n /**\n * Stop observing an element for size changes.\n * @param {HTMLElement} el - The element to stop observing.\n */\n unobserve(el: HTMLElement): void {\n const idxEl = this.resizeObserverEntries.indexOf(el);\n if (idxEl < 0) return;\n\n this.resizeObserver.unobserve(el);\n this.resizeObserverEntries.splice(idxEl, 1);\n }\n\n /**\n * This method checks that we're not continuing to watch elements that no\n * longer exist in the DOM. If any are found, we stop observing them and\n * remove them from our array of observed elements.\n *\n * @private\n * @static\n */\n flush(): void {\n this.resizeObserverEntries.forEach((el) => {\n if (!document.body.contains(el)) this.unobserve(el);\n });\n }\n}\n\nexport { ShinyResizeObserver };\n", "import { InputBinding, registerBinding } from \"./_utils\";\nimport { ShinyResizeObserver } from \"./_shinyResizeObserver\";\n\n/**\n * Methods for programmatically toggling the state of the sidebar. These methods\n * describe the desired state of the sidebar: `\"close\"` and `\"open\"` transition\n * the sidebar to the desired state, unless the sidebar is already in that\n * state. `\"toggle\"` transitions the sidebar to the state opposite of its\n * current state.\n * @typedef {SidebarToggleMethod}\n */\ntype SidebarToggleMethod = \"close\" | \"open\" | \"toggle\";\n\n/**\n * Data received by the input binding's `receiveMessage` method.\n * @typedef {SidebarMessageData}\n */\ntype SidebarMessageData = {\n method: SidebarToggleMethod;\n};\n\n/**\n * The DOM elements that make up the sidebar. `main`, `sidebar`, and `toggle`\n * are all direct children of `container` (in that order).\n * @interface SidebarComponents\n * @typedef {SidebarComponents}\n */\ninterface SidebarComponents {\n /**\n * The `layout_sidebar()` parent container, with class\n * `Sidebar.classes.LAYOUT`.\n * @type {HTMLElement}\n */\n container: HTMLElement;\n /**\n * The main content area of the sidebar layout.\n * @type {HTMLElement}\n */\n main: HTMLElement;\n /**\n * The sidebar container of the sidebar layout.\n * @type {HTMLElement}\n */\n sidebar: HTMLElement;\n /**\n * The toggle button that is used to toggle the sidebar state.\n * @type {HTMLElement}\n */\n toggle: HTMLElement;\n}\n\n/**\n * The bslib sidebar component class. This class is only used for collapsible\n * sidebars.\n *\n * @class Sidebar\n * @typedef {Sidebar}\n */\nclass Sidebar {\n /**\n * The DOM elements that make up the sidebar, see `SidebarComponents`.\n * @private\n * @type {SidebarComponents}\n */\n private layout: SidebarComponents;\n\n /**\n * A Shiny-specific resize observer that ensures Shiny outputs in the main\n * content areas of the sidebar resize appropriately.\n * @private\n * @type {ShinyResizeObserver}\n * @static\n */\n private static shinyResizeObserver = new ShinyResizeObserver();\n\n /**\n * Creates an instance of a collapsible bslib Sidebar.\n * @constructor\n * @param {HTMLElement} container\n */\n constructor(container: HTMLElement) {\n Sidebar.instanceMap.set(container, this);\n this.layout = {\n container,\n main: container.querySelector(\":scope > .main\") as HTMLElement,\n sidebar: container.querySelector(\":scope > .sidebar\") as HTMLElement,\n toggle: container.querySelector(\n \":scope > .collapse-toggle\"\n ) as HTMLElement,\n } as SidebarComponents;\n\n if (!this.layout.toggle) {\n throw new Error(\"Tried to initialize a non-collapsible sidebar.\");\n }\n\n const sideAccordion = this.layout.sidebar.querySelector(\n \":scope > .sidebar-content > .accordion\"\n );\n if (sideAccordion) sideAccordion.classList.add(\"accordion-flush\");\n\n this._initEventListeners();\n this._initSidebarCounters();\n this._initDesktop();\n\n // Start watching the main content area for size changes to ensure Shiny\n // outputs resize appropriately during sidebar transitions.\n Sidebar.shinyResizeObserver.observe(this.layout.main);\n\n container.removeAttribute(\"data-bslib-sidebar-init\");\n const initScript = container.querySelector(\n \":scope > script[data-bslib-sidebar-init]\"\n );\n if (initScript) {\n container.removeChild(initScript);\n }\n }\n\n /**\n * Read the current state of the sidebar. Note that, when calling this method,\n * the sidebar may be transitioning into the state returned by this method.\n *\n * @description\n * The sidebar state works as follows, starting from the open state. When the\n * sidebar is closed:\n * 1. We add both the `COLLAPSE` and `TRANSITIONING` classes to the sidebar.\n * 2. The sidebar collapse begins to animate. On desktop devices, and where it\n * is supported, we transition the `grid-template-columns` property of the\n * sidebar layout. On mobile, the sidebar is hidden immediately. In both\n * cases, the collapse icon rotates and we use this rotation to determine\n * when the transition is complete.\n * 3. If another sidebar state toggle is requested while closing the sidebar,\n * we remove the `COLLAPSE` class and the animation immediately starts to\n * reverse.\n * 4. When the `transition` is complete, we remove the `TRANSITIONING` class.\n * @readonly\n * @type {boolean}\n */\n get isClosed(): boolean {\n return this.layout.container.classList.contains(Sidebar.classes.COLLAPSE);\n }\n\n /**\n * Static classes related to the sidebar layout or state.\n * @public\n * @static\n * @readonly\n * @type {{ LAYOUT: string; COLLAPSE: string; TRANSITIONING: string; }}\n */\n public static readonly classes = {\n // eslint-disable-next-line @typescript-eslint/naming-convention\n LAYOUT: \"bslib-sidebar-layout\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n COLLAPSE: \"sidebar-collapsed\",\n // eslint-disable-next-line @typescript-eslint/naming-convention\n TRANSITIONING: \"transitioning\",\n };\n\n /**\n * If sidebars are initialized before the DOM is ready, we re-schedule the\n * initialization to occur on DOMContentLoaded.\n * @private\n * @static\n * @type {boolean}\n */\n private static onReadyScheduled = false;\n /**\n * A map of initialized sidebars to their respective Sidebar instances.\n * @private\n * @static\n * @type {WeakMap}\n */\n private static instanceMap: WeakMap = new WeakMap();\n\n /**\n * Given a sidebar container, return the Sidebar instance associated with it.\n * @public\n * @static\n * @param {HTMLElement} el\n * @returns {(Sidebar | undefined)}\n */\n public static getInstance(el: HTMLElement): Sidebar | undefined {\n return Sidebar.instanceMap.get(el);\n }\n\n /**\n * Initialize all collapsible sidebars on the page.\n * @public\n * @static\n * @param {boolean} [flushResizeObserver=true] When `true`, we remove\n * non-existent elements from the ResizeObserver. This is required\n * periodically to prevent memory leaks. To avoid over-checking, we only flush\n * the ResizeObserver when initializing sidebars after page load.\n */\n public static initCollapsibleAll(flushResizeObserver = true): void {\n if (document.readyState === \"loading\") {\n if (!Sidebar.onReadyScheduled) {\n Sidebar.onReadyScheduled = true;\n document.addEventListener(\"DOMContentLoaded\", () => {\n Sidebar.initCollapsibleAll(false);\n });\n }\n return;\n }\n\n const initSelector = `.${Sidebar.classes.LAYOUT}[data-bslib-sidebar-init]`;\n if (!document.querySelector(initSelector)) {\n // no sidebars to initialize\n return;\n }\n\n if (flushResizeObserver) Sidebar.shinyResizeObserver.flush();\n\n const containers = document.querySelectorAll(initSelector);\n containers.forEach((container) => new Sidebar(container as HTMLElement));\n }\n\n /**\n * Initialize event listeners for the sidebar toggle button.\n * @private\n */\n private _initEventListeners(): void {\n const { toggle } = this.layout;\n\n toggle.addEventListener(\"click\", (ev) => {\n ev.preventDefault();\n this.toggle(\"toggle\");\n });\n\n // Remove the transitioning class when the transition ends. We watch the\n // collapse toggle icon because it's guaranteed to transition, whereas the\n // sidebar doesn't animate on mobile (or in browsers where animating\n // grid-template-columns is not supported).\n toggle\n .querySelector(\".collapse-icon\")\n ?.addEventListener(\"transitionend\", () => this._finalizeState());\n }\n\n /**\n * Initialize nested sidebar counters.\n *\n * @description\n * This function walks up the DOM tree, adding CSS variables to each direct\n * parent sidebar layout that count the layout's position in the stack of\n * nested layouts. We use these counters to keep the collapse toggles from\n * overlapping. Note that always-open sidebars that don't have collapse\n * toggles break the chain of nesting.\n * @private\n */\n private _initSidebarCounters(): void {\n const { container } = this.layout;\n\n const selectorChildLayouts =\n `.${Sidebar.classes.LAYOUT}` +\n \"> .main > \" +\n `.${Sidebar.classes.LAYOUT}:not([data-bslib-sidebar-open=\"always\"])`;\n\n const isInnermostLayout =\n container.querySelector(selectorChildLayouts) === null;\n\n if (!isInnermostLayout) {\n // There are sidebar layouts nested within this layout; defer to children\n return;\n }\n\n function nextSidebarParent(el: HTMLElement | null): HTMLElement | null {\n el = el ? el.parentElement : null;\n if (el && el.classList.contains(\"main\")) {\n // .bslib-sidebar-layout > .main > .bslib-sidebar-layout\n el = el.parentElement;\n }\n if (el && el.classList.contains(Sidebar.classes.LAYOUT)) {\n return el;\n }\n return null;\n }\n\n const layouts = [container];\n let parent = nextSidebarParent(container);\n\n while (parent) {\n // Add parent to front of layouts array, so we sort outer -> inner\n layouts.unshift(parent);\n parent = nextSidebarParent(parent);\n }\n\n const count = { left: 0, right: 0 };\n layouts.forEach(function (x: HTMLElement, i: number): void {\n x.style.setProperty(\"--bslib-sidebar-counter\", i.toString());\n const isRight = x.classList.contains(\"sidebar-right\");\n const thisCount = isRight ? count.right++ : count.left++;\n x.style.setProperty(\n \"--bslib-sidebar-overlap-counter\",\n thisCount.toString()\n );\n });\n }\n\n /**\n * Initialize the sidebar's initial state when `open = \"desktop\"`.\n * @private\n */\n private _initDesktop(): void {\n const { container } = this.layout;\n // If sidebar is marked open='desktop'...\n if (container.dataset.bslibSidebarOpen?.trim() !== \"desktop\") {\n return;\n }\n\n // then close sidebar on mobile\n const initCollapsed = window\n .getComputedStyle(container)\n .getPropertyValue(\"--bslib-sidebar-js-init-collapsed\");\n\n if (initCollapsed.trim() === \"true\") {\n this.toggle(\"close\");\n }\n }\n\n /**\n * Toggle the sidebar's open/closed state.\n * @public\n * @param {SidebarToggleMethod | undefined} method Whether to `\"open\"`,\n * `\"close\"` or `\"toggle\"` the sidebar. If `.toggle()` is called without an\n * argument, it will toggle the sidebar's state.\n */\n public toggle(method: SidebarToggleMethod | undefined): void {\n if (typeof method === \"undefined\") {\n method = \"toggle\";\n }\n\n const { container, sidebar } = this.layout;\n const isClosed = this.isClosed;\n\n if ([\"open\", \"close\", \"toggle\"].indexOf(method) === -1) {\n throw new Error(`Unknown method ${method}`);\n }\n\n if (method === \"toggle\") {\n method = isClosed ? \"open\" : \"close\";\n }\n\n if ((isClosed && method === \"close\") || (!isClosed && method === \"open\")) {\n // nothing to do, sidebar is already in the desired state\n return;\n }\n\n if (method === \"open\") {\n // unhide sidebar immediately when opening,\n // otherwise the sidebar is hidden on transitionend\n sidebar.hidden = false;\n }\n\n // Add a transitioning class just before adding COLLAPSE_CLASS since we want\n // some of the transitioning styles to apply before the collapse state\n container.classList.add(Sidebar.classes.TRANSITIONING);\n container.classList.toggle(Sidebar.classes.COLLAPSE);\n }\n\n /**\n * When the sidebar open/close transition ends, finalize the sidebar's state.\n * @private\n */\n private _finalizeState(): void {\n const { container, sidebar, toggle } = this.layout;\n container.classList.remove(Sidebar.classes.TRANSITIONING);\n sidebar.hidden = this.isClosed;\n toggle.ariaExpanded = this.isClosed ? \"false\" : \"true\";\n\n // Send browser-native event with updated sidebar state\n const event = new CustomEvent(\"bslib.sidebar\", {\n bubbles: true,\n detail: { open: !this.isClosed },\n });\n sidebar.dispatchEvent(event);\n\n // Trigger Shiny input and output binding events\n $(sidebar).trigger(\"toggleCollapse.sidebarInputBinding\");\n $(sidebar).trigger(this.isClosed ? \"hidden\" : \"shown\");\n }\n}\n\n/**\n * A Shiny input binding for a sidebar.\n * @class SidebarInputBinding\n * @typedef {SidebarInputBinding}\n * @extends {InputBinding}\n */\nclass SidebarInputBinding extends InputBinding {\n find(scope: HTMLElement) {\n return $(scope).find(`.${Sidebar.classes.LAYOUT} > .bslib-sidebar-input`);\n }\n\n getValue(el: HTMLElement): boolean {\n const sb = Sidebar.getInstance(el.parentElement as HTMLElement);\n if (!sb) return false;\n return !sb.isClosed;\n }\n\n setValue(el: HTMLElement, value: boolean): void {\n const method = value ? \"open\" : \"close\";\n this.receiveMessage(el, { method });\n }\n\n subscribe(el: HTMLElement, callback: (x: boolean) => void) {\n $(el).on(\n \"toggleCollapse.sidebarInputBinding\",\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n function (event) {\n callback(true);\n }\n );\n }\n\n unsubscribe(el: HTMLElement) {\n $(el).off(\".sidebarInputBinding\");\n }\n\n receiveMessage(el: HTMLElement, data: SidebarMessageData) {\n const sb = Sidebar.getInstance(el.parentElement as HTMLElement);\n if (sb) sb.toggle(data.method);\n }\n}\n\nregisterBinding(SidebarInputBinding, \"sidebar\");\n\n// attach Sidebar class to window for global usage\n(window as any).bslib = (window as any).bslib || {};\n(window as any).bslib.Sidebar = Sidebar;\n"], + "mappings": ";mBAQA,IAAMA,EACJ,OAAO,MAAQ,MAAM,aAAe,KAAM,CAAC,EAG7C,SAASC,EACPC,EACAC,EACM,CACF,OAAO,OACT,MAAM,cAAc,SAAS,IAAID,EAAqB,SAAWC,CAAI,CAEzE,CCXA,IAAMC,EAAN,KAA0B,CAoDxB,aAAc,CACZ,KAAK,sBAAwB,CAAC,EAC9B,KAAK,eAAiB,IAAI,eAAgBC,GAAY,CACpD,IAAMC,EAAc,IAAI,MAAM,QAAQ,EAItC,GAHA,OAAO,cAAcA,CAAW,EAG5B,CAAC,OAAO,MAAO,OAEnB,IAAMC,EAAU,CAAC,EAEjB,QAAWC,KAASH,EACZG,EAAM,kBAAkB,aACzBA,EAAM,OAAO,cAAc,qBAAqB,GAErDA,EAAM,OACH,iBAA8B,qBAAqB,EACnD,QAASC,GAAO,CACf,GAAIF,EAAQ,SAASE,CAAE,EAAG,OAE1B,GAAM,CAAE,QAAAC,EAAS,SAAAC,CAAS,EAAI,EAAEF,CAAE,EAAE,KAAK,oBAAoB,EAC7D,GAAI,CAACC,GAAW,CAACA,EAAQ,OAAQ,OAGjC,IAAME,EAASH,EAAW,oBAW1B,GAVIG,GAASA,IAAU,OAElBA,IAAQH,EAAW,oBAAsB,MAG9CE,EAASF,CAAE,EAEXF,EAAQ,KAAKE,CAAE,EAGX,CAACA,EAAG,UAAU,SAAS,mBAAmB,GAAG,OACjD,IAAMI,EAAMJ,EAAG,cACb,yBACF,EACII,GAAKA,EAAI,aAAa,QAAS,MAAM,CAC3C,CAAC,CAEP,CAAC,CACH,CAMA,QAAQJ,EAAuB,CAC7B,KAAK,eAAe,QAAQA,CAAE,EAC9B,KAAK,sBAAsB,KAAKA,CAAE,CACpC,CAMA,UAAUA,EAAuB,CAC/B,IAAMK,EAAQ,KAAK,sBAAsB,QAAQL,CAAE,EAC/CK,EAAQ,IAEZ,KAAK,eAAe,UAAUL,CAAE,EAChC,KAAK,sBAAsB,OAAOK,EAAO,CAAC,EAC5C,CAUA,OAAc,CACZ,KAAK,sBAAsB,QAASL,GAAO,CACpC,SAAS,KAAK,SAASA,CAAE,GAAG,KAAK,UAAUA,CAAE,CACpD,CAAC,CACH,CACF,ECjFA,IAAMM,EAAN,KAAc,CAsBZ,YAAYC,EAAwB,CAWlC,GAVAD,EAAQ,YAAY,IAAIC,EAAW,IAAI,EACvC,KAAK,OAAS,CACZ,UAAAA,EACA,KAAMA,EAAU,cAAc,gBAAgB,EAC9C,QAASA,EAAU,cAAc,mBAAmB,EACpD,OAAQA,EAAU,cAChB,2BACF,CACF,EAEI,CAAC,KAAK,OAAO,OACf,MAAM,IAAI,MAAM,gDAAgD,EAGlE,IAAMC,EAAgB,KAAK,OAAO,QAAQ,cACxC,wCACF,EACIA,GAAeA,EAAc,UAAU,IAAI,iBAAiB,EAEhE,KAAK,oBAAoB,EACzB,KAAK,qBAAqB,EAC1B,KAAK,aAAa,EAIlBF,EAAQ,oBAAoB,QAAQ,KAAK,OAAO,IAAI,EAEpDC,EAAU,gBAAgB,yBAAyB,EACnD,IAAME,EAAaF,EAAU,cAC3B,0CACF,EACIE,GACFF,EAAU,YAAYE,CAAU,CAEpC,CAsBA,IAAI,UAAoB,CACtB,OAAO,KAAK,OAAO,UAAU,UAAU,SAASH,EAAQ,QAAQ,QAAQ,CAC1E,CAyCA,OAAc,YAAYI,EAAsC,CAC9D,OAAOJ,EAAQ,YAAY,IAAII,CAAE,CACnC,CAWA,OAAc,mBAAmBC,EAAsB,GAAY,CACjE,GAAI,SAAS,aAAe,UAAW,CAChCL,EAAQ,mBACXA,EAAQ,iBAAmB,GAC3B,SAAS,iBAAiB,mBAAoB,IAAM,CAClDA,EAAQ,mBAAmB,EAAK,CAClC,CAAC,GAEH,MACF,CAEA,IAAMM,EAAe,IAAIN,EAAQ,QAAQ,kCACzC,GAAI,CAAC,SAAS,cAAcM,CAAY,EAEtC,OAGED,GAAqBL,EAAQ,oBAAoB,MAAM,EAExC,SAAS,iBAAiBM,CAAY,EAC9C,QAASL,GAAc,IAAID,EAAQC,CAAwB,CAAC,CACzE,CAMQ,qBAA4B,CA5NtC,IAAAM,EA6NI,GAAM,CAAE,OAAAC,CAAO,EAAI,KAAK,OAExBA,EAAO,iBAAiB,QAAUC,GAAO,CACvCA,EAAG,eAAe,EAClB,KAAK,OAAO,QAAQ,CACtB,CAAC,GAMDF,EAAAC,EACG,cAAc,gBAAgB,IADjC,MAAAD,EAEI,iBAAiB,gBAAiB,IAAM,KAAK,eAAe,EAClE,CAaQ,sBAA6B,CACnC,GAAM,CAAE,UAAAN,CAAU,EAAI,KAAK,OAErBS,EACJ,IAAIV,EAAQ,QAAQ,oBAEhBA,EAAQ,QAAQ,iDAKtB,GAAI,EAFFC,EAAU,cAAcS,CAAoB,IAAM,MAIlD,OAGF,SAASC,EAAkBP,EAA4C,CAMrE,OALAA,EAAKA,EAAKA,EAAG,cAAgB,KACzBA,GAAMA,EAAG,UAAU,SAAS,MAAM,IAEpCA,EAAKA,EAAG,eAENA,GAAMA,EAAG,UAAU,SAASJ,EAAQ,QAAQ,MAAM,EAC7CI,EAEF,IACT,CAEA,IAAMQ,EAAU,CAACX,CAAS,EACtBY,EAASF,EAAkBV,CAAS,EAExC,KAAOY,GAELD,EAAQ,QAAQC,CAAM,EACtBA,EAASF,EAAkBE,CAAM,EAGnC,IAAMC,EAAQ,CAAE,KAAM,EAAG,MAAO,CAAE,EAClCF,EAAQ,QAAQ,SAAUG,EAAgBC,EAAiB,CACzDD,EAAE,MAAM,YAAY,0BAA2BC,EAAE,SAAS,CAAC,EAE3D,IAAMC,EADUF,EAAE,UAAU,SAAS,eAAe,EACxBD,EAAM,QAAUA,EAAM,OAClDC,EAAE,MAAM,YACN,kCACAE,EAAU,SAAS,CACrB,CACF,CAAC,CACH,CAMQ,cAAqB,CA7S/B,IAAAV,EA8SI,GAAM,CAAE,UAAAN,CAAU,EAAI,KAAK,OAE3B,KAAIM,EAAAN,EAAU,QAAQ,mBAAlB,YAAAM,EAAoC,UAAW,UACjD,OAIoB,OACnB,iBAAiBN,CAAS,EAC1B,iBAAiB,mCAAmC,EAErC,KAAK,IAAM,QAC3B,KAAK,OAAO,OAAO,CAEvB,CASO,OAAOiB,EAA+C,CACvD,OAAOA,GAAW,cACpBA,EAAS,UAGX,GAAM,CAAE,UAAAjB,EAAW,QAAAkB,CAAQ,EAAI,KAAK,OAC9BC,EAAW,KAAK,SAEtB,GAAI,CAAC,OAAQ,QAAS,QAAQ,EAAE,QAAQF,CAAM,IAAM,GAClD,MAAM,IAAI,MAAM,kBAAkBA,GAAQ,EAGxCA,IAAW,WACbA,EAASE,EAAW,OAAS,SAG1B,EAAAA,GAAYF,IAAW,SAAa,CAACE,GAAYF,IAAW,UAK7DA,IAAW,SAGbC,EAAQ,OAAS,IAKnBlB,EAAU,UAAU,IAAID,EAAQ,QAAQ,aAAa,EACrDC,EAAU,UAAU,OAAOD,EAAQ,QAAQ,QAAQ,EACrD,CAMQ,gBAAuB,CAC7B,GAAM,CAAE,UAAAC,EAAW,QAAAkB,EAAS,OAAAX,CAAO,EAAI,KAAK,OAC5CP,EAAU,UAAU,OAAOD,EAAQ,QAAQ,aAAa,EACxDmB,EAAQ,OAAS,KAAK,SACtBX,EAAO,aAAe,KAAK,SAAW,QAAU,OAGhD,IAAMa,EAAQ,IAAI,YAAY,gBAAiB,CAC7C,QAAS,GACT,OAAQ,CAAE,KAAM,CAAC,KAAK,QAAS,CACjC,CAAC,EACDF,EAAQ,cAAcE,CAAK,EAG3B,EAAEF,CAAO,EAAE,QAAQ,oCAAoC,EACvD,EAAEA,CAAO,EAAE,QAAQ,KAAK,SAAW,SAAW,OAAO,CACvD,CACF,EAjUMG,EAANtB,EAAMsB,EAeW,oBAAsB,IAAIC,EAfrCD,EA0FmB,QAAU,CAE/B,OAAQ,uBAER,SAAU,oBAEV,cAAe,eACjB,EAjGIA,EA0GW,iBAAmB,GA1G9BA,EAiHW,YAA6C,IAAI,QAwNlE,IAAME,EAAN,cAAkCC,CAAa,CAC7C,KAAKC,EAAoB,CACvB,OAAO,EAAEA,CAAK,EAAE,KAAK,IAAIJ,EAAQ,QAAQ,+BAA+B,CAC1E,CAEA,SAASlB,EAA0B,CACjC,IAAMuB,EAAKL,EAAQ,YAAYlB,EAAG,aAA4B,EAC9D,OAAKuB,EACE,CAACA,EAAG,SADK,EAElB,CAEA,SAASvB,EAAiBwB,EAAsB,CAC9C,IAAMV,EAASU,EAAQ,OAAS,QAChC,KAAK,eAAexB,EAAI,CAAE,OAAAc,CAAO,CAAC,CACpC,CAEA,UAAUd,EAAiByB,EAAgC,CACzD,EAAEzB,CAAE,EAAE,GACJ,qCAEA,SAAUiB,EAAO,CACfQ,EAAS,EAAI,CACf,CACF,CACF,CAEA,YAAYzB,EAAiB,CAC3B,EAAEA,CAAE,EAAE,IAAI,sBAAsB,CAClC,CAEA,eAAeA,EAAiB0B,EAA0B,CACxD,IAAMH,EAAKL,EAAQ,YAAYlB,EAAG,aAA4B,EAC1DuB,GAAIA,EAAG,OAAOG,EAAK,MAAM,CAC/B,CACF,EAEAC,EAAgBP,EAAqB,SAAS,EAG7C,OAAe,MAAS,OAAe,OAAS,CAAC,EACjD,OAAe,MAAM,QAAUF", + "names": ["InputBinding", "registerBinding", "inputBindingClass", "name", "ShinyResizeObserver", "entries", "resizeEvent", "resized", "entry", "el", "binding", "onResize", "owner", "img", "idxEl", "_Sidebar", "container", "sideAccordion", "initScript", "el", "flushResizeObserver", "initSelector", "_a", "toggle", "ev", "selectorChildLayouts", "nextSidebarParent", "layouts", "parent", "count", "x", "i", "thisCount", "method", "sidebar", "isClosed", "event", "Sidebar", "ShinyResizeObserver", "SidebarInputBinding", "InputBinding", "scope", "sb", "value", "callback", "data", "registerBinding"] +} diff --git a/shiny/experimental/www/bslib/components/value_box/value_box.css b/shiny/experimental/www/bslib/components/value_box/value_box.css new file mode 100644 index 000000000..1022f7123 --- /dev/null +++ b/shiny/experimental/www/bslib/components/value_box/value_box.css @@ -0,0 +1 @@ +.bslib-value-box{--bslib-value-box-separator-color: rgba(var(--bs-white-rgb, 255,255,255), 0.175)}.bslib-value-box .value-box-grid{grid-template-columns:var(--bslib-value-box-widths)}.bslib-value-box .value-box-showcase{align-items:center;justify-content:center;margin-top:auto;margin-bottom:auto;padding:1rem;overflow:hidden;max-height:var(--bslib-value-box-max-height)}.bslib-value-box .value-box-showcase .bi,.bslib-value-box .value-box-showcase .fa{opacity:.85;min-width:50px;max-width:125%}.bslib-value-box .value-box-showcase .bi{font-size:5rem}.bslib-value-box .value-box-showcase .fa{font-size:4rem}.bslib-value-box .value-box-showcase.showcase-top-right{align-items:end;padding-left:0;padding-bottom:0}.bslib-value-box .value-box-area{justify-content:center;padding:1.5rem 1rem;font-size:.9rem;font-weight:500}.bslib-value-box .value-box-area *{color:inherit;margin-bottom:0;margin-top:0}.bslib-value-box .value-box-area>:first-child{font-size:1rem;margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2;color:var(--bs-heading-color);color:inherit}.bslib-value-box .value-box-area>:first-child::after{content:'\00a0 '}.bslib-value-box .value-box-area>:nth-child(2){font-size:calc(1.325rem + .9vw);margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2;color:var(--bs-heading-color);color:inherit}@media (min-width: 1200px){.bslib-value-box .value-box-area>:nth-child(2){font-size:2rem}}.bslib-value-box .value-box-area>:nth-child(2)::after{content:'\00a0 '}.bslib-value-box .value-box-area.border-start{border-color:var(--bslib-value-box-separator-color) !important}.bslib-value-box.bslib-full-screen .value-box-grid{grid-template-columns:var(--bslib-value-box-widths-full-screen)}.bslib-value-box.bslib-full-screen .value-box-showcase{max-height:var(--bslib-value-box-max-height-full-screen)}.bslib-value-box:not(.bslib-full-screen) .value-box-showcase.showcase-top-right{margin-top:0}@media (max-width: 575.98px){.bslib-value-box .value-box-grid{grid-template-columns:var(--bslib-value-box-widths) !important}} diff --git a/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js b/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js new file mode 100644 index 000000000..eb701a901 --- /dev/null +++ b/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js @@ -0,0 +1,112 @@ +/*! bslib 0.5.0.9000 | (c) 2012-2023 RStudio, PBC. | License: MIT + file LICENSE */ +"use strict";(()=>{var dt=Object.defineProperty,Ot=Object.defineProperties,Nt=Object.getOwnPropertyDescriptor,Ut=Object.getOwnPropertyDescriptors;var at=Object.getOwnPropertySymbols;var Lt=Object.prototype.hasOwnProperty,It=Object.prototype.propertyIsEnumerable;var ht=(n,t,e)=>t in n?dt(n,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):n[t]=e,T=(n,t)=>{for(var e in t||(t={}))Lt.call(t,e)&&ht(n,e,t[e]);if(at)for(var e of at(t))It.call(t,e)&&ht(n,e,t[e]);return n},L=(n,t)=>Ot(n,Ut(t));var B=(n,t,e,i)=>{for(var s=i>1?void 0:i?Nt(t,e):t,o=n.length-1,r;o>=0;o--)(r=n[o])&&(s=(i?r(t,e,s):r(s))||s);return i&&s&&dt(t,e,s),s};var W=(n,t,e)=>new Promise((i,s)=>{var o=l=>{try{h(e.next(l))}catch(a){s(a)}},r=l=>{try{h(e.throw(l))}catch(a){s(a)}},h=l=>l.done?i(l.value):Promise.resolve(l.value).then(o,r);h((e=e.apply(n,t)).next())});var I=window,V=I.ShadowRoot&&(I.ShadyCSS===void 0||I.ShadyCSS.nativeShadow)&&"adoptedStyleSheets"in Document.prototype&&"replace"in CSSStyleSheet.prototype,pt=Symbol(),ct=new WeakMap,R=class{constructor(t,e,i){if(this._$cssResult$=!0,i!==pt)throw Error("CSSResult is not constructable. Use `unsafeCSS` or `css` instead.");this.cssText=t,this.t=e}get styleSheet(){let t=this.o,e=this.t;if(V&&t===void 0){let i=e!==void 0&&e.length===1;i&&(t=ct.get(e)),t===void 0&&((this.o=t=new CSSStyleSheet).replaceSync(this.cssText),i&&ct.set(e,t))}return t}toString(){return this.cssText}},ut=n=>new R(typeof n=="string"?n:n+"",void 0,pt);var F=(n,t)=>{V?n.adoptedStyleSheets=t.map(e=>e instanceof CSSStyleSheet?e:e.styleSheet):t.forEach(e=>{let i=document.createElement("style"),s=I.litNonce;s!==void 0&&i.setAttribute("nonce",s),i.textContent=e.cssText,n.appendChild(i)})},D=V?n=>n:n=>n instanceof CSSStyleSheet?(t=>{let e="";for(let i of t.cssRules)e+=i.cssText;return ut(e)})(n):n;var K,j=window,mt=j.trustedTypes,Rt=mt?mt.emptyScript:"",vt=j.reactiveElementPolyfillSupport,G={toAttribute(n,t){switch(t){case Boolean:n=n?Rt:null;break;case Object:case Array:n=n==null?n:JSON.stringify(n)}return n},fromAttribute(n,t){let e=n;switch(t){case Boolean:e=n!==null;break;case Number:e=n===null?null:Number(n);break;case Object:case Array:try{e=JSON.parse(n)}catch(i){e=null}}return e}},ft=(n,t)=>t!==n&&(t==t||n==n),J={attribute:!0,type:String,converter:G,reflect:!1,hasChanged:ft},Q="finalized",v=class extends HTMLElement{constructor(){super(),this._$Ei=new Map,this.isUpdatePending=!1,this.hasUpdated=!1,this._$El=null,this.u()}static addInitializer(t){var e;this.finalize(),((e=this.h)!==null&&e!==void 0?e:this.h=[]).push(t)}static get observedAttributes(){this.finalize();let t=[];return this.elementProperties.forEach((e,i)=>{let s=this._$Ep(i,e);s!==void 0&&(this._$Ev.set(s,i),t.push(s))}),t}static createProperty(t,e=J){if(e.state&&(e.attribute=!1),this.finalize(),this.elementProperties.set(t,e),!e.noAccessor&&!this.prototype.hasOwnProperty(t)){let i=typeof t=="symbol"?Symbol():"__"+t,s=this.getPropertyDescriptor(t,i,e);s!==void 0&&Object.defineProperty(this.prototype,t,s)}}static getPropertyDescriptor(t,e,i){return{get(){return this[e]},set(s){let o=this[t];this[e]=s,this.requestUpdate(t,o,i)},configurable:!0,enumerable:!0}}static getPropertyOptions(t){return this.elementProperties.get(t)||J}static finalize(){if(this.hasOwnProperty(Q))return!1;this[Q]=!0;let t=Object.getPrototypeOf(this);if(t.finalize(),t.h!==void 0&&(this.h=[...t.h]),this.elementProperties=new Map(t.elementProperties),this._$Ev=new Map,this.hasOwnProperty("properties")){let e=this.properties,i=[...Object.getOwnPropertyNames(e),...Object.getOwnPropertySymbols(e)];for(let s of i)this.createProperty(s,e[s])}return this.elementStyles=this.finalizeStyles(this.styles),!0}static finalizeStyles(t){let e=[];if(Array.isArray(t)){let i=new Set(t.flat(1/0).reverse());for(let s of i)e.unshift(D(s))}else t!==void 0&&e.push(D(t));return e}static _$Ep(t,e){let i=e.attribute;return i===!1?void 0:typeof i=="string"?i:typeof t=="string"?t.toLowerCase():void 0}u(){var t;this._$E_=new Promise(e=>this.enableUpdating=e),this._$AL=new Map,this._$Eg(),this.requestUpdate(),(t=this.constructor.h)===null||t===void 0||t.forEach(e=>e(this))}addController(t){var e,i;((e=this._$ES)!==null&&e!==void 0?e:this._$ES=[]).push(t),this.renderRoot!==void 0&&this.isConnected&&((i=t.hostConnected)===null||i===void 0||i.call(t))}removeController(t){var e;(e=this._$ES)===null||e===void 0||e.splice(this._$ES.indexOf(t)>>>0,1)}_$Eg(){this.constructor.elementProperties.forEach((t,e)=>{this.hasOwnProperty(e)&&(this._$Ei.set(e,this[e]),delete this[e])})}createRenderRoot(){var t;let e=(t=this.shadowRoot)!==null&&t!==void 0?t:this.attachShadow(this.constructor.shadowRootOptions);return F(e,this.constructor.elementStyles),e}connectedCallback(){var t;this.renderRoot===void 0&&(this.renderRoot=this.createRenderRoot()),this.enableUpdating(!0),(t=this._$ES)===null||t===void 0||t.forEach(e=>{var i;return(i=e.hostConnected)===null||i===void 0?void 0:i.call(e)})}enableUpdating(t){}disconnectedCallback(){var t;(t=this._$ES)===null||t===void 0||t.forEach(e=>{var i;return(i=e.hostDisconnected)===null||i===void 0?void 0:i.call(e)})}attributeChangedCallback(t,e,i){this._$AK(t,i)}_$EO(t,e,i=J){var s;let o=this.constructor._$Ep(t,i);if(o!==void 0&&i.reflect===!0){let r=(((s=i.converter)===null||s===void 0?void 0:s.toAttribute)!==void 0?i.converter:G).toAttribute(e,i.type);this._$El=t,r==null?this.removeAttribute(o):this.setAttribute(o,r),this._$El=null}}_$AK(t,e){var i;let s=this.constructor,o=s._$Ev.get(t);if(o!==void 0&&this._$El!==o){let r=s.getPropertyOptions(o),h=typeof r.converter=="function"?{fromAttribute:r.converter}:((i=r.converter)===null||i===void 0?void 0:i.fromAttribute)!==void 0?r.converter:G;this._$El=o,this[o]=h.fromAttribute(e,r.type),this._$El=null}}requestUpdate(t,e,i){let s=!0;t!==void 0&&(((i=i||this.constructor.getPropertyOptions(t)).hasChanged||ft)(this[t],e)?(this._$AL.has(t)||this._$AL.set(t,e),i.reflect===!0&&this._$El!==t&&(this._$EC===void 0&&(this._$EC=new Map),this._$EC.set(t,i))):s=!1),!this.isUpdatePending&&s&&(this._$E_=this._$Ej())}_$Ej(){return W(this,null,function*(){this.isUpdatePending=!0;try{yield this._$E_}catch(e){Promise.reject(e)}let t=this.scheduleUpdate();return t!=null&&(yield t),!this.isUpdatePending})}scheduleUpdate(){return this.performUpdate()}performUpdate(){var t;if(!this.isUpdatePending)return;this.hasUpdated,this._$Ei&&(this._$Ei.forEach((s,o)=>this[o]=s),this._$Ei=void 0);let e=!1,i=this._$AL;try{e=this.shouldUpdate(i),e?(this.willUpdate(i),(t=this._$ES)===null||t===void 0||t.forEach(s=>{var o;return(o=s.hostUpdate)===null||o===void 0?void 0:o.call(s)}),this.update(i)):this._$Ek()}catch(s){throw e=!1,this._$Ek(),s}e&&this._$AE(i)}willUpdate(t){}_$AE(t){var e;(e=this._$ES)===null||e===void 0||e.forEach(i=>{var s;return(s=i.hostUpdated)===null||s===void 0?void 0:s.call(i)}),this.hasUpdated||(this.hasUpdated=!0,this.firstUpdated(t)),this.updated(t)}_$Ek(){this._$AL=new Map,this.isUpdatePending=!1}get updateComplete(){return this.getUpdateComplete()}getUpdateComplete(){return this._$E_}shouldUpdate(t){return!0}update(t){this._$EC!==void 0&&(this._$EC.forEach((e,i)=>this._$EO(i,this[i],e)),this._$EC=void 0),this._$Ek()}updated(t){}firstUpdated(t){}};v[Q]=!0,v.elementProperties=new Map,v.elementStyles=[],v.shadowRootOptions={mode:"open"},vt==null||vt({ReactiveElement:v}),((K=j.reactiveElementVersions)!==null&&K!==void 0?K:j.reactiveElementVersions=[]).push("1.6.2");var Z,q=window,w=q.trustedTypes,gt=w?w.createPolicy("lit-html",{createHTML:n=>n}):void 0,Y="$lit$",f=`lit$${(Math.random()+"").slice(9)}$`,Ct="?"+f,Vt=`<${Ct}>`,A=document,H=()=>A.createComment(""),M=n=>n===null||typeof n!="object"&&typeof n!="function",St=Array.isArray,Dt=n=>St(n)||typeof(n==null?void 0:n[Symbol.iterator])=="function",X=`[ +\f\r]`,P=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,yt=/-->/g,_t=/>/g,_=RegExp(`>|${X}(?:([^\\s"'>=/]+)(${X}*=${X}*(?:[^ +\f\r"'\`<>=]|("|')|))|$)`,"g"),$t=/'/g,bt=/"/g,Tt=/^(?:script|style|textarea|title)$/i,wt=n=>(t,...e)=>({_$litType$:n,strings:t,values:e}),Yt=wt(1),te=wt(2),E=Symbol.for("lit-noChange"),p=Symbol.for("lit-nothing"),At=new WeakMap,b=A.createTreeWalker(A,129,null,!1);function xt(n,t){if(!Array.isArray(n)||!n.hasOwnProperty("raw"))throw Error("invalid template strings array");return gt!==void 0?gt.createHTML(t):t}var jt=(n,t)=>{let e=n.length-1,i=[],s,o=t===2?"":"",r=P;for(let h=0;h"?(r=s!=null?s:P,c=-1):d[1]===void 0?c=-2:(c=r.lastIndex-d[2].length,a=d[1],r=d[3]===void 0?_:d[3]==='"'?bt:$t):r===bt||r===$t?r=_:r===yt||r===_t?r=P:(r=_,s=void 0);let m=r===_&&n[h+1].startsWith("/>")?" ":"";o+=r===P?l+Vt:c>=0?(i.push(a),l.slice(0,c)+Y+l.slice(c)+f+m):l+f+(c===-2?(i.push(void 0),h):m)}return[xt(n,o+(n[e]||"")+(t===2?"":"")),i]},C=class{constructor({strings:t,_$litType$:e},i){let s;this.parts=[];let o=0,r=0,h=t.length-1,l=this.parts,[a,d]=jt(t,e);if(this.el=C.createElement(a,i),b.currentNode=this.el.content,e===2){let c=this.el.content,u=c.firstChild;u.remove(),c.append(...u.childNodes)}for(;(s=b.nextNode())!==null&&l.length0){s.textContent=w?w.emptyScript:"";for(let m=0;m2||i[0]!==""||i[1]!==""?(this._$AH=Array(i.length-1).fill(new String),this.strings=i):this._$AH=p}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(t,e=this,i,s){let o=this.strings,r=!1;if(o===void 0)t=x(this,t,e,0),r=!M(t)||t!==this._$AH&&t!==E,r&&(this._$AH=t);else{let h=t,l,a;for(t=o[0],l=0;l{var i,s;let o=(i=e==null?void 0:e.renderBefore)!==null&&i!==void 0?i:t,r=o._$litPart$;if(r===void 0){let h=(s=e==null?void 0:e.renderBefore)!==null&&s!==void 0?s:null;o._$litPart$=r=new S(t.insertBefore(H(),h),h,void 0,e!=null?e:{})}return r._$AI(n),r};var ot,rt;var g=class extends v{constructor(){super(...arguments),this.renderOptions={host:this},this._$Do=void 0}createRenderRoot(){var t,e;let i=super.createRenderRoot();return(t=(e=this.renderOptions).renderBefore)!==null&&t!==void 0||(e.renderBefore=i.firstChild),i}update(t){let e=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(t),this._$Do=kt(e,this.renderRoot,this.renderOptions)}connectedCallback(){var t;super.connectedCallback(),(t=this._$Do)===null||t===void 0||t.setConnected(!0)}disconnectedCallback(){var t;super.disconnectedCallback(),(t=this._$Do)===null||t===void 0||t.setConnected(!1)}render(){return E}};g.finalized=!0,g._$litElement$=!0,(ot=globalThis.litElementHydrateSupport)===null||ot===void 0||ot.call(globalThis,{LitElement:g});var Pt=globalThis.litElementPolyfillSupport;Pt==null||Pt({LitElement:g});((rt=globalThis.litElementVersions)!==null&&rt!==void 0?rt:globalThis.litElementVersions=[]).push("3.3.2");var zt=(n,t)=>t.kind==="method"&&t.descriptor&&!("value"in t.descriptor)?L(T({},t),{finisher(e){e.createProperty(t.key,n)}}):{kind:"field",key:Symbol(),placement:"own",descriptor:{},originalKey:t.key,initializer(){typeof t.initializer=="function"&&(this[t.key]=t.initializer.call(this))},finisher(e){e.createProperty(t.key,n)}},Bt=(n,t,e)=>{t.constructor.createProperty(e,n)};function z(n){return(t,e)=>e!==void 0?Bt(n,t,e):zt(n,t)}var lt,He=((lt=window.HTMLSlotElement)===null||lt===void 0?void 0:lt.prototype.assignedElements)!=null?(n,t)=>n.assignedElements(t):(n,t)=>n.assignedNodes(t).filter(e=>e.nodeType===Node.ELEMENT_NODE);var N=class extends g{constructor(){super(...arguments);this.elementChildren=[]}connectedCallback(){this.elementChildren=Array.from(this.childNodes),this.maybeCarryFill(),super.connectedCallback()}get slotElements(){return this.elementChildren}createRenderRoot(){return this}maybeCarryFill(){this.isFillCarrier?(this.classList.add("html-fill-container"),this.classList.add("html-fill-item")):(this.classList.remove("html-fill-container"),this.classList.remove("html-fill-item"))}get isFillCarrier(){if(!this.parentElement)return!1;let e=this.parentElement.classList.contains("html-fill-container"),i=Array.from(this.children).some(s=>s.classList.contains("html-fill-item"));return e&&i}};N.isShinyInput=!1;var Wt=window.bootstrap?window.bootstrap.Tooltip:class{},y=class extends N{constructor(){super();this.placement="auto";this.options="{}";this.visible=!1;this.onChangeCallback=e=>{};this._onShown=this._onShown.bind(this),this._onHidden=this._onHidden.bind(this),this.style.display="contents"}get allOptions(){let e=JSON.parse(this.options);return T({title:this.title,placement:this.placement,html:!0,sanitize:!0},e)}get title(){return this.children[0].innerHTML}connectedCallback(){super.connectedCallback(),this.triggerElement.setAttribute("data-bs-toggle","tooltip"),this._tooltip=new Wt(this.triggerElement,this.allOptions),this._observer=this._createVisibilityObserver(),this.triggerElement.addEventListener("shown.bs.tooltip",this._onShown),this.triggerElement.addEventListener("hidden.bs.tooltip",this._onHidden)}disconnectedCallback(){this.triggerElement.removeEventListener("shown.bs.tooltip",this._onShown),this.triggerElement.removeEventListener("hidden.bs.tooltip",this._onHidden),super.disconnectedCallback()}render(){return p}get triggerElement(){if(this.children.length>1){let e=this.children[this.children.length-1];return e.setAttribute("tabindex","0"),e}if(this.childNodes.length>1){let e=document.createElement("span");return e.setAttribute("tabindex","0"),e.append(this.childNodes[this.childNodes.length-1]),this.appendChild(e),e}return this}getValue(){return this.visible}_onShown(){this.visible=!0,this.onChangeCallback(!0),this._observer.observe(this.triggerElement)}_onHidden(){this.visible=!1,this.onChangeCallback(!0),this._observer.unobserve(this.triggerElement)}receiveMessage(e,i){let s=i.method;if(s==="toggle")this._toggle(i.value);else if(s==="update")this._updateTitle(i.title);else throw new Error(`Unknown method ${s}`)}_toggle(e){e==="toggle"&&(e=this.visible?"hide":"show"),e==="hide"&&this._tooltip.hide(),e==="show"&&this._show()}_show(){!this.visible&&this.visibleTrigger&&this._tooltip.show()}get visibleTrigger(){let e=this.triggerElement;return e&&e.offsetParent!==null}_updateTitle(e){e&&(Shiny.renderDependencies(e.deps),this._setContentCarefully(e.html))}_setContentCarefully(e){let{tip:i}=this._tooltip;if(i&&i.offsetParent!==null){let s=i.querySelector(".tooltip-inner");s&&(s.innerHTML=e),this._tooltip.update(),$(this).one("hidden.bs.tooltip",function(){this._setContent(e)})}else this._setContent(e)}_setContent(e){this._tooltip.setContent({".tooltip-inner":e})}_createVisibilityObserver(){let e=i=>{this.visible&&i.forEach(s=>{s.isIntersecting||this._tooltip.hide()})};return new IntersectionObserver(e)}};y.tagName="bslib-tooltip",y.isShinyInput=!0,B([z({type:String})],y.prototype,"placement",2),B([z({type:String})],y.prototype,"options",2);function Ht(n,{type:t=null}={}){if(!window.Shiny)return;class e extends Shiny.InputBinding{constructor(){super()}find(s){return $(s).find(n)}getValue(s){return"getValue"in s?s.getValue():s.value}getType(s){return t}subscribe(s,o){s.onChangeCallback=o}unsubscribe(s){s.onChangeCallback=o=>{}}receiveMessage(s,o){s.receiveMessage(s,o)}}Shiny.inputBindings.register(new e,`${n}-Binding`)}[y].forEach(n=>{customElements.define(n.tagName,n),n.isShinyInput&&Ht(n.tagName)});})(); +/*! Bundled license information: + +@lit/reactive-element/css-tag.js: + (** + * @license + * Copyright 2019 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/reactive-element.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +lit-html/lit-html.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +lit-element/lit-element.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +lit-html/is-server.js: + (** + * @license + * Copyright 2022 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/custom-element.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/property.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/state.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/base.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/event-options.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/query.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/query-all.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/query-async.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/query-assigned-elements.js: + (** + * @license + * Copyright 2021 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) + +@lit/reactive-element/decorators/query-assigned-nodes.js: + (** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + *) +*/ +//# sourceMappingURL=webComponents.min.js.map diff --git a/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js.map b/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js.map new file mode 100644 index 000000000..ae97810d0 --- /dev/null +++ b/shiny/experimental/www/bslib/components/webComponents/webComponents.min.js.map @@ -0,0 +1,7 @@ +{ + "version": 3, + "sources": ["../../../../node_modules/@lit/reactive-element/src/css-tag.ts", "../../../../node_modules/@lit/reactive-element/src/reactive-element.ts", "../../../../node_modules/lit-html/src/lit-html.ts", "../../../../node_modules/lit-element/src/lit-element.ts", "../../../../node_modules/@lit/reactive-element/src/decorators/property.ts", "../../../../node_modules/@lit/reactive-element/src/decorators/query-assigned-elements.ts", "../../../../srcts/src/components/webcomponents/_lightElement.ts", "../../../../srcts/src/components/tooltip.ts", "../../../../srcts/src/components/webcomponents/_makeInputBinding.ts", "../../../../srcts/src/components/webComponents.ts"], + "sourcesContent": ["/**\n * @license\n * Copyright 2019 Google LLC\n * SPDX-License-Identifier: BSD-3-Clause\n */\n\nconst NODE_MODE = false;\nconst global = NODE_MODE ? globalThis : window;\n\n/**\n * Whether the current browser supports `adoptedStyleSheets`.\n */\nexport const supportsAdoptingStyleSheets: boolean =\n global.ShadowRoot &&\n (global.ShadyCSS === undefined || global.ShadyCSS.nativeShadow) &&\n 'adoptedStyleSheets' in Document.prototype &&\n 'replace' in CSSStyleSheet.prototype;\n\n/**\n * A CSSResult or native CSSStyleSheet.\n *\n * In browsers that support constructible CSS style sheets, CSSStyleSheet\n * object can be used for styling along side CSSResult from the `css`\n * template tag.\n */\nexport type CSSResultOrNative = CSSResult | CSSStyleSheet;\n\nexport type CSSResultArray = Array;\n\n/**\n * A single CSSResult, CSSStyleSheet, or an array or nested arrays of those.\n */\nexport type CSSResultGroup = CSSResultOrNative | CSSResultArray;\n\nconst constructionToken = Symbol();\n\nconst cssTagCache = new WeakMap();\n\n/**\n * A container for a string of CSS text, that may be used to create a CSSStyleSheet.\n *\n * CSSResult is the return value of `css`-tagged template literals and\n * `unsafeCSS()`. In order to ensure that CSSResults are only created via the\n * `css` tag and `unsafeCSS()`, CSSResult cannot be constructed directly.\n */\nexport class CSSResult {\n // This property needs to remain unminified.\n ['_$cssResult$'] = true;\n readonly cssText: string;\n private _styleSheet?: CSSStyleSheet;\n private _strings: TemplateStringsArray | undefined;\n\n private constructor(\n cssText: string,\n strings: TemplateStringsArray | undefined,\n safeToken: symbol\n ) {\n if (safeToken !== constructionToken) {\n throw new Error(\n 'CSSResult is not constructable. Use `unsafeCSS` or `css` instead.'\n );\n }\n this.cssText = cssText;\n this._strings = strings;\n }\n\n // This is a getter so that it's lazy. In practice, this means stylesheets\n // are not created until the first element instance is made.\n get styleSheet(): CSSStyleSheet | undefined {\n // If `supportsAdoptingStyleSheets` is true then we assume CSSStyleSheet is\n // constructable.\n let styleSheet = this._styleSheet;\n const strings = this._strings;\n if (supportsAdoptingStyleSheets && styleSheet === undefined) {\n const cacheable = strings !== undefined && strings.length === 1;\n if (cacheable) {\n styleSheet = cssTagCache.get(strings);\n }\n if (styleSheet === undefined) {\n (this._styleSheet = styleSheet = new CSSStyleSheet()).replaceSync(\n this.cssText\n );\n if (cacheable) {\n cssTagCache.set(strings, styleSheet);\n }\n }\n }\n return styleSheet;\n }\n\n toString(): string {\n return this.cssText;\n }\n}\n\ntype ConstructableCSSResult = CSSResult & {\n new (\n cssText: string,\n strings: TemplateStringsArray | undefined,\n safeToken: symbol\n ): CSSResult;\n};\n\nconst textFromCSSResult = (value: CSSResultGroup | number) => {\n // This property needs to remain unminified.\n if ((value as CSSResult)['_$cssResult$'] === true) {\n return (value as CSSResult).cssText;\n } else if (typeof value === 'number') {\n return value;\n } else {\n throw new Error(\n `Value passed to 'css' function must be a 'css' function result: ` +\n `${value}. Use 'unsafeCSS' to pass non-literal values, but take care ` +\n `to ensure page security.`\n );\n }\n};\n\n/**\n * Wrap a value for interpolation in a {@linkcode css} tagged template literal.\n *\n * This is unsafe because untrusted CSS text can be used to phone home\n * or exfiltrate data to an attacker controlled site. Take care to only use\n * this with trusted input.\n */\nexport const unsafeCSS = (value: unknown) =>\n new (CSSResult as ConstructableCSSResult)(\n typeof value === 'string' ? value : String(value),\n undefined,\n constructionToken\n );\n\n/**\n * A template literal tag which can be used with LitElement's\n * {@linkcode LitElement.styles} property to set element styles.\n *\n * For security reasons, only literal string values and number may be used in\n * embedded expressions. To incorporate non-literal values {@linkcode unsafeCSS}\n * may be used inside an expression.\n */\nexport const css = (\n strings: TemplateStringsArray,\n ...values: (CSSResultGroup | number)[]\n): CSSResult => {\n const cssText =\n strings.length === 1\n ? strings[0]\n : values.reduce(\n (acc, v, idx) => acc + textFromCSSResult(v) + strings[idx + 1],\n strings[0]\n );\n return new (CSSResult as ConstructableCSSResult)(\n cssText,\n strings,\n constructionToken\n );\n};\n\n/**\n * Applies the given styles to a `shadowRoot`. When Shadow DOM is\n * available but `adoptedStyleSheets` is not, styles are appended to the\n * `shadowRoot` to [mimic spec behavior](https://wicg.github.io/construct-stylesheets/#using-constructed-stylesheets).\n * Note, when shimming is used, any styles that are subsequently placed into\n * the shadowRoot should be placed *before* any shimmed adopted styles. This\n * will match spec behavior that gives adopted sheets precedence over styles in\n * shadowRoot.\n */\nexport const adoptStyles = (\n renderRoot: ShadowRoot,\n styles: Array\n) => {\n if (supportsAdoptingStyleSheets) {\n (renderRoot as ShadowRoot).adoptedStyleSheets = styles.map((s) =>\n s instanceof CSSStyleSheet ? s : s.styleSheet!\n );\n } else {\n styles.forEach((s) => {\n const style = document.createElement('style');\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const nonce = (global as any)['litNonce'];\n if (nonce !== undefined) {\n style.setAttribute('nonce', nonce);\n }\n style.textContent = (s as CSSResult).cssText;\n renderRoot.appendChild(style);\n });\n }\n};\n\nconst cssResultFromStyleSheet = (sheet: CSSStyleSheet) => {\n let cssText = '';\n for (const rule of sheet.cssRules) {\n cssText += rule.cssText;\n }\n return unsafeCSS(cssText);\n};\n\nexport const getCompatibleStyle =\n supportsAdoptingStyleSheets ||\n (NODE_MODE && global.CSSStyleSheet === undefined)\n ? (s: CSSResultOrNative) => s\n : (s: CSSResultOrNative) =>\n s instanceof CSSStyleSheet ? cssResultFromStyleSheet(s) : s;\n", "/**\n * @license\n * Copyright 2017 Google LLC\n * SPDX-License-Identifier: BSD-3-Clause\n */\n\n/**\n * Use this module if you want to create your own base class extending\n * {@link ReactiveElement}.\n * @packageDocumentation\n */\n\nimport {\n getCompatibleStyle,\n adoptStyles,\n CSSResultGroup,\n CSSResultOrNative,\n} from './css-tag.js';\nimport type {\n ReactiveController,\n ReactiveControllerHost,\n} from './reactive-controller.js';\n\n// In the Node build, this import will be injected by Rollup:\n// import {HTMLElement, customElements} from '@lit-labs/ssr-dom-shim';\n\nexport * from './css-tag.js';\nexport type {\n ReactiveController,\n ReactiveControllerHost,\n} from './reactive-controller.js';\n\nconst NODE_MODE = false;\nconst global = NODE_MODE ? globalThis : window;\n\nif (NODE_MODE) {\n global.customElements ??= customElements;\n}\n\nconst DEV_MODE = true;\n\nlet requestUpdateThenable: (name: string) => {\n then: (\n onfulfilled?: (value: boolean) => void,\n _onrejected?: () => void\n ) => void;\n};\n\nlet issueWarning: (code: string, warning: string) => void;\n\nconst trustedTypes = (global as unknown as {trustedTypes?: {emptyScript: ''}})\n .trustedTypes;\n\n// Temporary workaround for https://crbug.com/993268\n// Currently, any attribute starting with \"on\" is considered to be a\n// TrustedScript source. Such boolean attributes must be set to the equivalent\n// trusted emptyScript value.\nconst emptyStringForBooleanAttribute = trustedTypes\n ? (trustedTypes.emptyScript as unknown as '')\n : '';\n\nconst polyfillSupport = DEV_MODE\n ? global.reactiveElementPolyfillSupportDevMode\n : global.reactiveElementPolyfillSupport;\n\nif (DEV_MODE) {\n // Ensure warnings are issued only 1x, even if multiple versions of Lit\n // are loaded.\n const issuedWarnings: Set = (global.litIssuedWarnings ??=\n new Set());\n\n // Issue a warning, if we haven't already.\n issueWarning = (code: string, warning: string) => {\n warning += ` See https://lit.dev/msg/${code} for more information.`;\n if (!issuedWarnings.has(warning)) {\n console.warn(warning);\n issuedWarnings.add(warning);\n }\n };\n\n issueWarning(\n 'dev-mode',\n `Lit is in dev mode. Not recommended for production!`\n );\n\n // Issue polyfill support warning.\n if (global.ShadyDOM?.inUse && polyfillSupport === undefined) {\n issueWarning(\n 'polyfill-support-missing',\n `Shadow DOM is being polyfilled via \\`ShadyDOM\\` but ` +\n `the \\`polyfill-support\\` module has not been loaded.`\n );\n }\n\n requestUpdateThenable = (name) => ({\n then: (\n onfulfilled?: (value: boolean) => void,\n _onrejected?: () => void\n ) => {\n issueWarning(\n 'request-update-promise',\n `The \\`requestUpdate\\` method should no longer return a Promise but ` +\n `does so on \\`${name}\\`. Use \\`updateComplete\\` instead.`\n );\n if (onfulfilled !== undefined) {\n onfulfilled(false);\n }\n },\n });\n}\n\n/**\n * Contains types that are part of the unstable debug API.\n *\n * Everything in this API is not stable and may change or be removed in the future,\n * even on patch releases.\n */\n// eslint-disable-next-line @typescript-eslint/no-namespace\nexport namespace ReactiveUnstable {\n /**\n * When Lit is running in dev mode and `window.emitLitDebugLogEvents` is true,\n * we will emit 'lit-debug' events to window, with live details about the update and render\n * lifecycle. These can be useful for writing debug tooling and visualizations.\n *\n * Please be aware that running with window.emitLitDebugLogEvents has performance overhead,\n * making certain operations that are normally very cheap (like a no-op render) much slower,\n * because we must copy data and dispatch events.\n */\n // eslint-disable-next-line @typescript-eslint/no-namespace\n export namespace DebugLog {\n export type Entry = Update;\n export interface Update {\n kind: 'update';\n }\n }\n}\n\ninterface DebugLoggingWindow {\n // Even in dev mode, we generally don't want to emit these events, as that's\n // another level of cost, so only emit them when DEV_MODE is true _and_ when\n // window.emitLitDebugEvents is true.\n emitLitDebugLogEvents?: boolean;\n}\n\n/**\n * Useful for visualizing and logging insights into what the Lit template system is doing.\n *\n * Compiled out of prod mode builds.\n */\nconst debugLogEvent = DEV_MODE\n ? (event: ReactiveUnstable.DebugLog.Entry) => {\n const shouldEmit = (global as unknown as DebugLoggingWindow)\n .emitLitDebugLogEvents;\n if (!shouldEmit) {\n return;\n }\n global.dispatchEvent(\n new CustomEvent('lit-debug', {\n detail: event,\n })\n );\n }\n : undefined;\n\n/*\n * When using Closure Compiler, JSCompiler_renameProperty(property, object) is\n * replaced at compile time by the munged name for object[property]. We cannot\n * alias this function, so we have to use a small shim that has the same\n * behavior when not compiling.\n */\n/*@__INLINE__*/\nconst JSCompiler_renameProperty =

(\n prop: P,\n _obj: unknown\n): P => prop;\n\n/**\n * Converts property values to and from attribute values.\n */\nexport interface ComplexAttributeConverter {\n /**\n * Called to convert an attribute value to a property\n * value.\n */\n fromAttribute?(value: string | null, type?: TypeHint): Type;\n\n /**\n * Called to convert a property value to an attribute\n * value.\n *\n * It returns unknown instead of string, to be compatible with\n * https://github.com/WICG/trusted-types (and similar efforts).\n */\n toAttribute?(value: Type, type?: TypeHint): unknown;\n}\n\ntype AttributeConverter =\n | ComplexAttributeConverter\n | ((value: string | null, type?: TypeHint) => Type);\n\n/**\n * Defines options for a property accessor.\n */\nexport interface PropertyDeclaration {\n /**\n * When set to `true`, indicates the property is internal private state. The\n * property should not be set by users. When using TypeScript, this property\n * should be marked as `private` or `protected`, and it is also a common\n * practice to use a leading `_` in the name. The property is not added to\n * `observedAttributes`.\n */\n readonly state?: boolean;\n\n /**\n * Indicates how and whether the property becomes an observed attribute.\n * If the value is `false`, the property is not added to `observedAttributes`.\n * If true or absent, the lowercased property name is observed (e.g. `fooBar`\n * becomes `foobar`). If a string, the string value is observed (e.g\n * `attribute: 'foo-bar'`).\n */\n readonly attribute?: boolean | string;\n\n /**\n * Indicates the type of the property. This is used only as a hint for the\n * `converter` to determine how to convert the attribute\n * to/from a property.\n */\n readonly type?: TypeHint;\n\n /**\n * Indicates how to convert the attribute to/from a property. If this value\n * is a function, it is used to convert the attribute value a the property\n * value. If it's an object, it can have keys for `fromAttribute` and\n * `toAttribute`. If no `toAttribute` function is provided and\n * `reflect` is set to `true`, the property value is set directly to the\n * attribute. A default `converter` is used if none is provided; it supports\n * `Boolean`, `String`, `Number`, `Object`, and `Array`. Note,\n * when a property changes and the converter is used to update the attribute,\n * the property is never updated again as a result of the attribute changing,\n * and vice versa.\n */\n readonly converter?: AttributeConverter;\n\n /**\n * Indicates if the property should reflect to an attribute.\n * If `true`, when the property is set, the attribute is set using the\n * attribute name determined according to the rules for the `attribute`\n * property option and the value of the property converted using the rules\n * from the `converter` property option.\n */\n readonly reflect?: boolean;\n\n /**\n * A function that indicates if a property should be considered changed when\n * it is set. The function should take the `newValue` and `oldValue` and\n * return `true` if an update should be requested.\n */\n hasChanged?(value: Type, oldValue: Type): boolean;\n\n /**\n * Indicates whether an accessor will be created for this property. By\n * default, an accessor will be generated for this property that requests an\n * update when set. If this flag is `true`, no accessor will be created, and\n * it will be the user's responsibility to call\n * `this.requestUpdate(propertyName, oldValue)` to request an update when\n * the property changes.\n */\n readonly noAccessor?: boolean;\n}\n\n/**\n * Map of properties to PropertyDeclaration options. For each property an\n * accessor is made, and the property is processed according to the\n * PropertyDeclaration options.\n */\nexport interface PropertyDeclarations {\n readonly [key: string]: PropertyDeclaration;\n}\n\ntype PropertyDeclarationMap = Map;\n\ntype AttributeMap = Map;\n\n/**\n * A Map of property keys to values.\n *\n * Takes an optional type parameter T, which when specified as a non-any,\n * non-unknown type, will make the Map more strongly-typed, associating the map\n * keys with their corresponding value type on T.\n *\n * Use `PropertyValues` when overriding ReactiveElement.update() and\n * other lifecycle methods in order to get stronger type-checking on keys\n * and values.\n */\n// This type is conditional so that if the parameter T is not specified, or\n// is `any`, the type will include `Map`. Since T is not\n// given in the uses of PropertyValues in this file, all uses here fallback to\n// meaning `Map`, but if a developer uses\n// `PropertyValues` (or any other value for T) they will get a\n// strongly-typed Map type.\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type PropertyValues = T extends object\n ? PropertyValueMap\n : Map;\n\n/**\n * Do not use, instead prefer {@linkcode PropertyValues}.\n */\n// This type must be exported such that JavaScript generated by the Google\n// Closure Compiler can import a type reference.\nexport interface PropertyValueMap extends Map {\n get(k: K): T[K];\n set(key: K, value: T[K]): this;\n has(k: K): boolean;\n delete(k: K): boolean;\n}\n\nexport const defaultConverter: ComplexAttributeConverter = {\n toAttribute(value: unknown, type?: unknown): unknown {\n switch (type) {\n case Boolean:\n value = value ? emptyStringForBooleanAttribute : null;\n break;\n case Object:\n case Array:\n // if the value is `null` or `undefined` pass this through\n // to allow removing/no change behavior.\n value = value == null ? value : JSON.stringify(value);\n break;\n }\n return value;\n },\n\n fromAttribute(value: string | null, type?: unknown) {\n let fromValue: unknown = value;\n switch (type) {\n case Boolean:\n fromValue = value !== null;\n break;\n case Number:\n fromValue = value === null ? null : Number(value);\n break;\n case Object:\n case Array:\n // Do *not* generate exception when invalid JSON is set as elements\n // don't normally complain on being mis-configured.\n // TODO(sorvell): Do generate exception in *dev mode*.\n try {\n // Assert to adhere to Bazel's \"must type assert JSON parse\" rule.\n fromValue = JSON.parse(value!) as unknown;\n } catch (e) {\n fromValue = null;\n }\n break;\n }\n return fromValue;\n },\n};\n\nexport interface HasChanged {\n (value: unknown, old: unknown): boolean;\n}\n\n/**\n * Change function that returns true if `value` is different from `oldValue`.\n * This method is used as the default for a property's `hasChanged` function.\n */\nexport const notEqual: HasChanged = (value: unknown, old: unknown): boolean => {\n // This ensures (old==NaN, value==NaN) always returns false\n return old !== value && (old === old || value === value);\n};\n\nconst defaultPropertyDeclaration: PropertyDeclaration = {\n attribute: true,\n type: String,\n converter: defaultConverter,\n reflect: false,\n hasChanged: notEqual,\n};\n\n/**\n * The Closure JS Compiler doesn't currently have good support for static\n * property semantics where \"this\" is dynamic (e.g.\n * https://github.com/google/closure-compiler/issues/3177 and others) so we use\n * this hack to bypass any rewriting by the compiler.\n */\nconst finalized = 'finalized';\n\n/**\n * A string representing one of the supported dev mode warning categories.\n */\nexport type WarningKind = 'change-in-update' | 'migration';\n\nexport type Initializer = (element: ReactiveElement) => void;\n\n/**\n * Base element class which manages element properties and attributes. When\n * properties change, the `update` method is asynchronously called. This method\n * should be supplied by subclassers to render updates as desired.\n * @noInheritDoc\n */\nexport abstract class ReactiveElement\n // In the Node build, this `extends` clause will be substituted with\n // `(globalThis.HTMLElement ?? HTMLElement)`.\n //\n // This way, we will first prefer any global `HTMLElement` polyfill that the\n // user has assigned, and then fall back to the `HTMLElement` shim which has\n // been imported (see note at the top of this file about how this import is\n // generated by Rollup). Note that the `HTMLElement` variable has been\n // shadowed by this import, so it no longer refers to the global.\n extends HTMLElement\n implements ReactiveControllerHost\n{\n // Note: these are patched in only in DEV_MODE.\n /**\n * Read or set all the enabled warning categories for this class.\n *\n * This property is only used in development builds.\n *\n * @nocollapse\n * @category dev-mode\n */\n static enabledWarnings?: WarningKind[];\n\n /**\n * Enable the given warning category for this class.\n *\n * This method only exists in development builds, so it should be accessed\n * with a guard like:\n *\n * ```ts\n * // Enable for all ReactiveElement subclasses\n * ReactiveElement.enableWarning?.('migration');\n *\n * // Enable for only MyElement and subclasses\n * MyElement.enableWarning?.('migration');\n * ```\n *\n * @nocollapse\n * @category dev-mode\n */\n static enableWarning?: (warningKind: WarningKind) => void;\n\n /**\n * Disable the given warning category for this class.\n *\n * This method only exists in development builds, so it should be accessed\n * with a guard like:\n *\n * ```ts\n * // Disable for all ReactiveElement subclasses\n * ReactiveElement.disableWarning?.('migration');\n *\n * // Disable for only MyElement and subclasses\n * MyElement.disableWarning?.('migration');\n * ```\n *\n * @nocollapse\n * @category dev-mode\n */\n static disableWarning?: (warningKind: WarningKind) => void;\n\n /**\n * Adds an initializer function to the class that is called during instance\n * construction.\n *\n * This is useful for code that runs against a `ReactiveElement`\n * subclass, such as a decorator, that needs to do work for each\n * instance, such as setting up a `ReactiveController`.\n *\n * ```ts\n * const myDecorator = (target: typeof ReactiveElement, key: string) => {\n * target.addInitializer((instance: ReactiveElement) => {\n * // This is run during construction of the element\n * new MyController(instance);\n * });\n * }\n * ```\n *\n * Decorating a field will then cause each instance to run an initializer\n * that adds a controller:\n *\n * ```ts\n * class MyElement extends LitElement {\n * @myDecorator foo;\n * }\n * ```\n *\n * Initializers are stored per-constructor. Adding an initializer to a\n * subclass does not add it to a superclass. Since initializers are run in\n * constructors, initializers will run in order of the class hierarchy,\n * starting with superclasses and progressing to the instance's class.\n *\n * @nocollapse\n */\n static addInitializer(initializer: Initializer) {\n this.finalize();\n (this._initializers ??= []).push(initializer);\n }\n\n static _initializers?: Initializer[];\n\n /*\n * Due to closure compiler ES6 compilation bugs, @nocollapse is required on\n * all static methods and properties with initializers. Reference:\n * - https://github.com/google/closure-compiler/issues/1776\n */\n\n /**\n * Maps attribute names to properties; for example `foobar` attribute to\n * `fooBar` property. Created lazily on user subclasses when finalizing the\n * class.\n * @nocollapse\n */\n private static __attributeToPropertyMap: AttributeMap;\n\n /**\n * Marks class as having finished creating properties.\n */\n protected static [finalized] = true;\n\n /**\n * Memoized list of all element properties, including any superclass properties.\n * Created lazily on user subclasses when finalizing the class.\n * @nocollapse\n * @category properties\n */\n static elementProperties: PropertyDeclarationMap = new Map();\n\n /**\n * User-supplied object that maps property names to `PropertyDeclaration`\n * objects containing options for configuring reactive properties. When\n * a reactive property is set the element will update and render.\n *\n * By default properties are public fields, and as such, they should be\n * considered as primarily settable by element users, either via attribute or\n * the property itself.\n *\n * Generally, properties that are changed by the element should be private or\n * protected fields and should use the `state: true` option. Properties\n * marked as `state` do not reflect from the corresponding attribute\n *\n * However, sometimes element code does need to set a public property. This\n * should typically only be done in response to user interaction, and an event\n * should be fired informing the user; for example, a checkbox sets its\n * `checked` property when clicked and fires a `changed` event. Mutating\n * public properties should typically not be done for non-primitive (object or\n * array) properties. In other cases when an element needs to manage state, a\n * private property set with the `state: true` option should be used. When\n * needed, state properties can be initialized via public properties to\n * facilitate complex interactions.\n * @nocollapse\n * @category properties\n */\n static properties: PropertyDeclarations;\n\n /**\n * Memoized list of all element styles.\n * Created lazily on user subclasses when finalizing the class.\n * @nocollapse\n * @category styles\n */\n static elementStyles: Array = [];\n\n /**\n * Array of styles to apply to the element. The styles should be defined\n * using the {@linkcode css} tag function, via constructible stylesheets, or\n * imported from native CSS module scripts.\n *\n * Note on Content Security Policy:\n *\n * Element styles are implemented with `