embed_* functions: embed_cat_prop_plot(), embed_cat_table(), and embed_chr_table_html() have been removed. These were soft-deprecated since v1.1.0. Use makeme() instead (e.g., makeme(type = "cat_prop_plot_html"), makeme(type = "cat_table_html"), makeme(type = "chr_table_html")).cat_plot_docx now uses girafe global settings for colors: The colour_palette parameter has been removed from cat_plot_docx. Instead, colors are now controlled exclusively through global_settings_set(fn_name = "girafe") for consistency with HTML plots. This enables unified color management across all output formats (HTML and DOCX). Checkbox plot support (checked, not_checked, colour_2nd_binary_cat) is now also available for DOCX plots, with automatic legend hiding and label suppression for the unchecked category. Users should migrate from makeme(..., colour_palette = c(...)) to global_settings_set(fn_name = "girafe", new = list(palette_codes = list(c(...)))).crowd_plots_as_tabset() and crowd_tables_as_tabset() now insert Quarto {{< pagebreak >}} shortcodes between items when rendering to paginated formats (docx, typst, PDF). HTML-based formats (including revealjs and slidy) are detected via knitr::is_html_output() and are never page-broken. A new pagebreak parameter ("auto", "always", "never") controls this behaviourcheck_quarto_website_index() function to detect folders in a Quarto website project that contain .qmd files but are missing an index.qmd. Missing index files often cause broken navigation menus. Issues a cli warning listing the offending folders.quarto_pdf_post_render() function for use as a Quarto post-render script. For each rendered PDF, it checks for a matching DOCX file, extracts the document title from the DOCX metadata, sets it as the PDF metadata title (via Ghostscript), and updates the link text in the corresponding index.html. Defaults to reading output files from the QUARTO_PROJECT_OUTPUT_FILES environment variable set by Quarto during project render.make_file_links() function for dynamically creating markdown lists with links to files. Extracts document titles from DOCX, PPTX, and PDF file metadata and generates formatted markdown lists. Ideal for creating navigation links in Quarto/RMarkdown documents that point to generated reports in a folder. Supports glob patterns, recursive search, and customizable list formatting (unordered or numbered). It appends a file extension suffix to link labels when multiple file types are present for the same base name, making it clear which format each link points to.ggsaver() to automatically apply colour palettes from girafe() global settings when saving plots. This ensures saved PNG/PDF images match the appearance of interactive plots displayed with girafe(). Palette settings can be configured via global_settings_set(fn_name = "girafe", new = list(palette_codes = ...)) and will be automatically applied when saving plots through get_fig_title_suffix_from_ggplot() or direct ggsaver() callsfolder and file_prefix parameters to get_fig_title_suffix_from_ggplot() for controlling where files are saved and what prefix to use for filenamesget_fig_title_suffix_from_ggplot() to support global settings inheritance via global_settings_set(), consistent with other saros functions like make_link() and makeme()cat_prop_plot_docx and cat_freq_plot_docx into a single cat_plot_docx type, bringing it in line with the cat_plot_html API. The freq parameter now controls whether to create stacked proportion/percentage plots (freq=FALSE, default) or dodged frequency plots (freq=TRUE)cat_table_docx and chr_table_docx output types for creating Word document tables. These types use officer::body_add_table() to embed tables in Word documents. Like HTML table types, they return data.frames when docx_return_object=TRUE, enabling flexible table compositiondocx_return_object parameter to makeme() to control return type for DOCX outputs. When TRUE, returns the underlying object (mschart for plots, data.frame for tables) instead of embedding it in an rdocx document, enabling more flexible composition of Word documentsfig_height_h_barchart2() to S3 generic: Now supports both ggplot2 objects (from type="cat_plot_html") and mschart objects (from type="cat_plot_docx", docx_return_object=TRUE). This enables automatic height calculation for Word charts, making it easier to size charts consistently across output formatsn_range2() to S3 generic: Now supports both ggplot2 objects and mschart objects, similar to fig_height_h_barchart2(). This enables consistent sample size reporting across both HTML and Word chart outputs, making it easier to annotate charts with N ranges regardless of output formatmake_link() tests to use dynamic rlang::hash() values instead of hardcoded hashes, preventing fragile failures when hash representations change across environments or versions (#587)crowd_plots_as_tabset() now gives an informative error when passed table output (e.g. from type = "cat_table_html") instead of plot output, directing users to crowd_tables_as_tabset() (#515)txt_from_cat_mesos_plots() now gives an informative error when passed table output instead of plot output, instead of failing with a cryptic .category_order column-not-found error (#515)girafe() to return a static ggplot object instead of a girafe object when rendering in non-HTML knitr documents (e.g. PDF, Word). Previously the interactive widget was returned unconditionally, causing rendering failures in non-HTML output formats (#551)cat_plot_html data labels being left-aligned instead of centered. The hjust value for geom_label was corrected from 0 to 0.5 (#553)txt_from_cat_mesos_plots() for checkbox variables to compare on the checked category (or the first non-NA category as fallback) rather than always using the first category. This ensures the correct proportion is highlighted when the variable uses checked/not_checked encoding (#556)txt_from_cat_mesos_plots() color legend inconsistency across mesos groups. Category color assignment now only computes full factor levels for factor/ordered variables, preventing spurious extra colors from appearing in legends for non-factor variables (#552, #555)txt_from_cat_mesos_plots() showing zero proportions due to .variable_label_original being overwritten by the N-suffix mutation in add_n_to_label() before it could be used for joining. The original column is now preserved before any label mutation (#554)int_plot_html to always hide the fill legend (previously shown for single dep without indep), and to correctly respect hide_axis_text_if_single_variable (previously ignored for int_plot_html, only applied to cat_plot_html) (#557)cat_plot_docx percentage plots to show consistent axis breaks at 0%, 25%, 50%, 75%, and 100% by setting axis limits (0-1) and major unit (0.25; experimental) for the y-axis. This improves readability and consistency across Word document chartscat_plot_docx: increased label_font_size and main_font_size defaults from 6 to 9 for better readability in Word documentshide_axis_text_if_single_variable parameter not being respected in cat_plot_docx output. The function now properly hides axis labels for single-variable plots when this parameter is TRUE, matching the behavior of cat_plot_htmlcolour_palette parameter not being applied in cat_plot_docx output. The mschart library requires colour palettes to be named vectors where names match category levels, and the length must match the number of categories. The function now: (1) names the colour palette vector with category levels, (2) expands the palette with generated colours if user provides fewer colours than categories, and (3) trims the palette if user provides more colours than neededset_pdf_metadata_title() corrupting non-ASCII characters (e.g. æøå) in PDF metadata titles. The pdfmark title is now encoded as a UTF-16BE hex string with BOM instead of a plain PostScript literal, avoiding locale/codepage-dependent byte interpretation by Ghostscriptfig_height_h_barchart2() where hide_axis_text_if_single_variable=TRUE was paradoxically giving larger heights than FALSE. Now correctly detects when axis text has been hidden (via .variable_label_original column) and applies a reduction factor (default 0.6 via multiplier_hide_axis_single_var parameter) plus reduced minimum height, resulting in appropriately smaller plots when axis text is hiddenfig_height_h_barchart2() for int_plot_html plots: previously always returned max (12) regardless of plot complexity, and errored with "only supports a single indep variable" when an indep variable was present (because .value was falsely detected as a second indep column). Height now scales properly with n_y and n_cats_x via fig_height_h_barchart() with n_cats_y = 1 and n_legend_lines = 0n_range2() for int_plot_html plots to report N range per dependent variable instead of total count across all variables. Now correctly calculates sample size separately for each variable and reports the range (e.g., [250-299] when variables have different amounts of missing data)guess_legend_ncols() in girafe() to properly handle fill aesthetics using expressions like fill = factor(cyl). The function now uses get_fill_levels() to evaluate expressions in data context instead of direct column access, preventing "no non-missing arguments to max; returning -Inf" warningsmakeme() now gives an informative error message when the requested output type is incompatible with the dependent variable types (e.g., using type = "cat_table_html" with numeric variables, or type = "int_plot_html" with factor variables). Previously this produced a cryptic error about setting an attribute on NULL.fig_height_h_barchart2() ggplot height estimation: The function now extracts information directly from the ggplot object for more accurate height calculations:
base_size) instead of using a fixed default, so heights scale correctly when users change the global ggplot2 theme (e.g., theme_gray(base_size = 14))"right", "left") or hidden ("none"), no vertical space is allocated for it, producing more compact figurescount_max_wrapped_lines()) to count the true number of lines labels wrap to, instead of estimating from character countsfig_height_h_barchart() strip angle condition to use abs(strip_angle), so negative angles (e.g., -90° from default ggplot2 strip themes) are handled identically to their positive counterpartsresolve_category_colors() helper, removing duplication across make_content.cat_plot_html(), make_content.cat_plot_docx(), and related functions (#548)validate_* prefix for better discoverability via auto-complete (e.g., validate_bool(), validate_integerish(), validate_double(), validate_string())validate_params() helper function that consolidates multiple parameter validations into a single declarative call, reducing code duplication from 30+ lines to ~5 lines in fig_height_h_barchart()validate_integerish_rule(), validate_double_rule(), validate_bool_rule(), validate_string_rule()) for more expressive and maintainable validation specifications in validate_makeme_options()check_* aliases for existing codeis_int_plot_html() helper function to centralize detection logic for int_plot_html data structures, improving code maintainability and consistency across fig_height_h_barchart2() and n_rng2()quiet parameter to global_settings_reset() to optionally suppress informational messages when resetting global settings to package defaultsmakeme() now defaults to type = "auto" which intelligently detects the appropriate output type based on dependent variable classes:
int_plot_htmlchr_table_htmlcat_plot_htmlcrowd_plots_as_tabset() with save = NULL produced cryptic "object 'caption' not found" error. Added proper validation to ensure save parameter is a single logical value (TRUE or FALSE)txt_from_cat_mesos_plots() where second group (others) proportions incorrectly became zero. Refactored to process each variable separately, ensuring both groups' proportions are correctly calculated per variabletxt_from_cat_mesos_plots() where n_highest_categories=2 with binary (2-category) variables summed all categories to 1.0, producing uninformative results. Now only applies n_highest_categories when the variable has more categories than the threshold, otherwise uses only the single highest/lowest categoryx_axis_label_width parameter had no effect in int_plot_html when no independent variable was present. The apply_label_wrapping() function now correctly wraps .variable_label when indep_length == 0makeme() with multiple crowds produced identical plots instead of crowd-specific filtered data. The process_crowd_data() function now correctly passes filtered subset_data to make_content() for each crowd, ensuring each plot displays statistics computed from only that crowd's data subsettxt_from_cat_mesos_plots() to handle cases where .category_order contains NA values by using na.rm = TRUE when calculating max values, preventing "NA/NaN argument" errorsn_rng2() to correctly calculate sample sizes for int_plot_html plots by only checking complete cases on relevant variables (.value and independent variables) instead of all columns in the dataset. Also modified make_content.int_plot_html() to only include necessary columns in the plot object, reducing memory footprintmakeme() returns an empty data.frame instead of NULL when no plot or table can be created, simplifying downstream code (e.g. gt::gt() fails if served NULL)descend parameter now works correctly with ordered factors while preserving their inherent level ordering. Ordered factors maintain their natural order as the base, but descend can reverse the display ordercustom_palette() to allow priority_palette_codes to supplement base palette when insufficient colors are availableshowNA = "always"makeme(type = "int_plot_html") for interactive interval plots with violin and box plotstxt_from_cat_mesos_plots() to generate textual summaries from two categorical mesos-scale plots, highlighting significant differences in selected categories between groups. Supports global settings via global_settings_set() (closes #506)crowd_plots_as_tabset() to convert a list of ggplot2 objects (typically from makeme(crowd = ...)) into Quarto tabsets with automatic height calculation and optional download links. Supports both categorical and interval plots with intelligent auto-detection based on plot layersget_fig_title_suffix_from_ggplot() generates figure title suffixes with N range and optional download links for plotsdata_label = "mean" and data_label = "median" in makeme() for type = "cat_*_html" outputs (addresses issue #460)data_label_position argument to makeme() allowing data labels to be positioned at "center", "bottom", "top", or "above" bars in categorical plots (addresses issue #365)chr_table_html to support multiple independent variables for displaying background context with open-ended text responsesna_colour parameter to hex_bw() function to allow customization of text color for NA fills (default: "#ffffff")colour_2nd_binary_cat parameter to girafe() for checkbox plots - when set with checked/not_checked, reverses category order so the second category receives the specified colordescend_indep now consistently reverses indep order across tables and plots; plots use centralized .indep_order when indep is on the x-axis; ordered indep factors take precedence over sort_indep_by (reversed only when descend_indep=TRUE); legends preserve unused response levels in .category.count_total_indep). This prevents accidental sorting on arbitrary or missing columns and clarifies supported behaviorsort_indep_by now explicitly defaults to ".factor_order" and accepts NULL (treated as ".factor_order"). When no independent variable is provided (indep = NULL), specifying sort_indep_by or descend_indep no longer errors; they are simply ignored.dep_order now key by .variable_name (unique) instead of .variable_label, preventing many-to-one join errors in edge cases while preserving displayed labelsmakeme() where combinations of valid factor variables with all-NA factor variables incorrectly threw "mix of categorical and continuous variables" error. Variable type checking now uses filtered variable lists instead of original lists, preventing premature type validation errorsmakeme() failures for sigtest_table when dep and indep variables overlap. Now automatically excludes indep variables from dep selection to prevent conflictsget_fig_title_suffix_from_ggplot() to prevent subscript-out-of-bounds errors when vectorized parameters have mismatched lengthsn_rng2() now gracefully handles plots not created by makeme() by calculating N from complete cases instead of aborting, and properly handles all-NA and empty .count_per_indep_group values by returning "0" with clear warnings instead of producing infinite valuescrowd_plots_as_tabset() now validates each plot individually in the loop, preventing invalid objects from causing failuresis.data.frame() and nrow() checks instead of length() for more reliable validationint_plot_html where label_separator = NULL with independent variables caused violin/boxplot and label geoms to appear in separate facets due to inconsistent string wrappinglabel_separator = NULL, as having different main questions is expected behaviorcheck_bool() function - removed redundant condition that made validation always pass for NA valuesglue_together_range() to prevent edge case failures with empty or invalid data rangescheck_no_duplicated_label_suffix() to handle empty data frames and missing columns gracefullycheck_sort_by() validation to properly handle empty character vectors with clear error messageskeep_subitem() to handle character inputs and use factor levels for better NA handlingarrange_table_data() sorting logic for better reliabilityggplot2::theme_set() to ggplot2::set_theme() due to ggplot2 4.0.0custom_palette() in girafe() to properly handle unnamed elements in priority_palette_codes as colors for "NA" category. Previously, unnamed elements were filtered out, causing NA categories to receive no color assignment (appearing as white fill with white text)makeme() examples for 73.8% faster execution (6.6s → 1.7s total). Reduced variable counts and crowd configurations while maintaining educational valuefig_height_h_barchart2() example for consistency with optimized examples.spread algorithm in subset_vector() for better spread maximization using evenly spaced positionsmakeme() into focused helper functions (argument setup, crowd processing, output assembly, validation). Improves readability, testability, and robustness without changing public API (closes #368)post_process_makeme_data() into focused single-responsibility functions: process_indep_factor_levels() for general factor reversal and process_binary_category_colors() for cat_plot_html-specific binary category processingprocess_binary_category_colors() logic from make_content.cat_plot_html() to girafe(), consolidating all color-related logic in one placegirafe() to integrate colour_2nd_binary_cat parameter with convert_to_checkbox_plot() functionpretty_tabular.R to tabular_write.R and extracted tabular_read() function into separate filezzz.R under .saros.env for easier maintenanceapply_label_wrapping() helper function, improving maintainability and consistencymake_content.int_plot_html() for clarity.onUnload() function to clean up global options when package is unloaded, preventing option pollution in user's R sessionair for consistent code formattinggirafe() and makeme() functions to meet CRAN's character limit per line requirementmakeme() helper functions with full roxygen2 documentation and @keywords internal annotation for internal API clarityint_plot_html using vdiffr snapshot tests covering various scenariostxt_from_cat_mesos_plots() with proper cleanup using withr::defer()survey package to Suggests for enhanced testing capabilitiesmakeme now handles arguments in parent frame scopes correctly.get_common_levels now works with no provided col_pos.".percentage" as valid (sorting) columncat_table_html returns NULL if nothing to displayscale_x_reordered now only sorts when neededcat_table_html takes showNA properlycat_plot_htmlmakeme() useful for creating plots comparing a group with all other groups combined, or everyone.fig_height_h_barchart() finally works adequately for 90 percent of figure combinations.makeme(), make_link(), girafe(), n_range(), n_range2()makeme(type="int_plot_html") for violin plotsmakeme(type="cat_plot_html") no longer shows a line one the y axis for single variable plots.crosstable.data.frame() and crosstable.survey() methods.embed_*-functions are replaced by makeme(type="cat_plot") etc
makeme() takes S3-methods which eases expansions.embed_* are thus lifecycle soft-deprecated.n_range() gives the n_range for a given dataset, dependent and independent variables.n_range2() same as n_range() but takes a makeme()-object for convenience.make_link() for generating downloadable figures, tables, data or anything else, on the spot.ggsaver() is a minor wrapper to ease using make_link with ggplot2::ggsave().fig_height_h_barchart() estimates the optimal figure height for a horizontal barchart.fig_height_h_barchart2() same as above, but takes a makeme()-object for convenience.makeme(type="cat_plot_html") allows sorting within each dependent variable-by-independent variable (facet-sort)-feat: Removed colour_palette (#328)
-fix: Argument chapter_qmd_start_section_filepath now has effect. (#321)
-fix: simplified create_email_credentials() (#305)
-fix: pkgdown for recode_checkbox_sets (#304)
-fix: cat_*_plot now displays keys for unused categories in legend (#301)
-fix: single_y_bivariates_if_deps_above() no longer affects univariates (#300)
-fix: More robust estimate_plot_height if dep only has one NA category (#299)
-fix: Significance test now works in edge cases (#297)
-feat: chr_table now ignores NA and empty strings. (#296)
-fix: serialized_format now actually supports {qs}, if installed. (#293)
-feat: Added default to replace_heading_for_group (#291)
-feat: Removed flexi-app (#289)
-feat: Removed unnecessary rendering-assistance tools. (#287)
-feat: gen_qmd_index has now been refactored (#286)
gen_qmd_file, which better handles index and reportcrosstable3() now handles character vectors. (#278)attach_qualtrics_labels is now more flexible.sanitize_labels.setup_access_restrictions now accepts NULL password input.setup_access_restrictions now can warn instead of error if rel_parent_path does not exist.sort_by now correctly sorts with descend=TRUE for ".upper", etc.list_available_element_typesdraft_report to only run on maintainer's local computer.sanitize_labels now has option to translate ASCII HTML-escaped characters to unicode.