R/listing.R

Defines functions absolute_preview_url resolve_preview_url as_collection_name as_collection_dir articles_info category_hash categories_listing_html subscription_html custom_html html_for_author html_for_categories_metadata html_for_articles article_listing_html generate_listing resolve_yaml_listing resolve_listing

resolve_listing <- function(input_file, site_config, metadata) {

  # alias collection
  collection <- metadata$listing

  # get articles
  articles <- articles_info(dirname(input_file), collection)

  # generate listing
  generate_listing(
    input_file,
    metadata,
    site_config,
    collection,
    articles
  )
}

resolve_yaml_listing <- function(input_file, site_config, metadata, yaml_listing) {

  site_dir <- find_site_dir(input_file)

  listing_articles <- list()

  categories <- TRUE
  categories_metadata <- TRUE
  authors_metadata <- TRUE

  for (collection in names(yaml_listing)) {


    collection <- site_collections(site_dir, site_config)[[collection]]

    if (identical(collection[["categories"]], FALSE))
      categories <- FALSE

    if (identical(collection[["categories_metadata"]], FALSE))
      categories_metadata <- FALSE

    if (identical(collection[["authors_metadata"]], FALSE))
      authors_metadata <- FALSE


    articles <- yaml_listing[[collection$name]]

    articles_file <- file.path(as_collection_dir(dirname(input_file), collection$name),
                               file_with_ext(collection$name, "json"))

    all_articles <- read_articles_json(articles_file, site_dir, site_config, collection)

    articles <- lapply(articles, function(article) {
      path <- paste0(collection$name, "/", article, "/")
      for (article in all_articles) {
        if (identical(path, article$path))
          return(article)
      }
      NULL
    })
    articles[sapply(articles, is.null)] <- NULL

    listing_articles <- append(listing_articles, articles)
  }

  # generate html
  listing_html <- html_for_articles(listing_articles,
                                    caption = metadata$title,
                                    categories = categories,
                                    categories_metadata = categories_metadata,
                                    authors_metadata = authors_metadata)

  listing <- list(
    html = html_file(listing_html)
  )
}

generate_listing <- function(input_file,
                             metadata,
                             site_config,
                             collection,
                             articles) {

  # validate that the collection exists
  site_dir <- dirname(input_file)
  as_collection_dir(site_dir, collection)

  # get the collection metadata
  collection <- site_collections(site_dir, site_config)[[as_collection_name(collection)]]

  # check for and enforce a limit on feed items (defaults to 20)
  feed_articles <- articles
  feed_items_max <- not_null(collection[["feed_items_max"]], 20)
  if (is.integer(feed_items_max) && (length(articles) > feed_items_max)) {
    feed_articles <- feed_articles[1:feed_items_max]
  }

  # generate feed and write it
  feed_xml <- file_with_ext(input_file, "xml")
  feed_xml <- write_feed_xml(feed_xml, site_config, collection, feed_articles)

  # generate html
  listing_html <- article_listing_html(site_dir, metadata, collection, articles)
  html <- html_file(listing_html)

  # return feed and listing html
  list(
    feed = feed_xml,
    html = html
  )
}


article_listing_html <- function(site_dir, metadata, collection, articles) {

  # detect whether we are showing categories (sidebar and inline)
  categories <- not_null(collection[["categories"]], TRUE)
  categories_metadata <- not_null(collection[["categories_metadata"]], TRUE)
  authors_metadata <- not_null(collection[["authors_metadata"]], TRUE)

  # check for subscription
  subscription_html <- subscription_html(site_dir, collection)
  if (!is.null(subscription_html)) {
    subscription_html <- tags$div(class = "sidebar-section subscribe",
      tags$h3("Subscribe"),
      subscription_html
    )
  }

  # check for custom HTML
  custom_html <- custom_html(site_dir, collection)
  if (!is.null(custom_html)) {
    # Bring your own header and styling. CSS class "sidebar-section.custom" is available if desired.
    custom_html <- tags$div(custom_html)
  }

  # generate html
  html_for_articles(articles,
                    caption = metadata$title,
                    categories = categories,
                    categories_metadata = categories_metadata,
                    authors_metadata = authors_metadata,
                    subscription_html = subscription_html,
                    custom_html = custom_html)
}

html_for_articles <- function(articles, caption = NULL, categories = FALSE, categories_metadata = FALSE, authors_metadata = FALSE, subscription_html = NULL, custom_html = NULL) {

  # generate categories listing
  categories_html <- if (categories) categories_listing_html(articles)

  # generate html
  article_index <- 0
  articles_html <- lapply(articles, function(article) {

    # bump article index
    article_index <<- article_index + 1

    # metadata
    metadata <- list()
    metadata$categories <- as.character(article[["categories"]])

    if (!is.null(article$preview))
      if (article_index <= 5)
        preview <- img(src = article$preview)
      else
        preview <- img(`data-src` = article$preview)
    else
      preview <- NULL

    if (categories_metadata)
      categories_metadata_html <- html_for_categories_metadata(article)
    else
      categories_metadata_html <- NULL

    a(href = article$path, class = "post-preview",
      tags$script(class = "post-metadata", type = "text/json",
                  HTML(jsonlite::toJSON(metadata))),
      div(class = "metadata",
        div(class = "publishedDate", date_as_abbrev(article$date)),
        if (authors_metadata) html_for_author(article$author)
      ),
      div(class = "thumbnail", preview),
      div(class = "description",
        h2(article$title),
        categories_metadata_html,
        p(article$description)
      )
    )
  })

  articles_html <- htmltools::tagList(
    h1(class = "posts-list-caption", `data-caption` = caption, caption),
    articles_html
  )

  # more posts link
  more_posts <- div(class = "posts-more",
    tags$a(href = "#", HTML("More articles &raquo;"))
  )

  # do we have a sidebar
  sidebar <- !is.null(custom_html) || !is.null(subscription_html) || !is.null(categories_html)

  # required JS and CSS
  listing_js_css <- html_from_file(
    system.file("rmarkdown/templates/distill_article/resources/listing.html",
                 package = "distill")
  )

  # wrap in a div
  if (sidebar) {
    placeholder_html("article_listing", tagList(
      listing_js_css,
      div(class = "posts-container posts-with-sidebar posts-apply-limit l-screen-inset",
        div(class = "posts-list", articles_html),
        div(class = "posts-sidebar", custom_html, subscription_html, categories_html),
        more_posts
      )
    ))
  } else {
    placeholder_html("article_listing", tagList(
      listing_js_css,
      div(class = "posts-container posts-apply-limit l-page",
        div(class = "posts-list", custom_html, subscription_html, articles_html),
        more_posts
      )
    ))
  }
}

html_for_categories_metadata <- function(article) {
  if (is.list(article[["categories"]])) {
    tags <- lapply(article[["categories"]], function(category) {
      div(class = "dt-tag", category)
    })
    div(class = "dt-tags", tags)
  } else {
    NULL
  }
}


html_for_author <- function(author) {
  author <- fixup_author(author)
  if (!is.null(author)) {
    tags <- lapply(author, function(auth) {
      div(class = "dt-author", auth$name)
    })
    div(class = "dt-authors", tags)
  } else {
    NULL
  }
}

custom_html <- function(site_dir, collection) {

  # check for custom HTML entry
  custom <- collection[["custom"]]
  if (!is.null(custom)) {
    custom <- file.path(site_dir, custom)
    if (!file.exists(custom))
      stop("Specified custom file '", custom, "' does not exist", call. = FALSE)
    html_from_file(custom)
  } else {
    NULL
  }

}

subscription_html <- function(site_dir, collection) {

  # check for subscribe entry
  subscribe <- collection[["subscribe"]]
  if (!is.null(subscribe)) {
    subscribe <- file.path(site_dir, subscribe)
    if (!file.exists(subscribe))
      stop("Specified subscribe file '", subscribe, "' does not exist", call. = FALSE)
    html_from_file(subscribe)
  } else {
    NULL
  }

}

categories_listing_html <- function(articles) {

  # count categories
  categories <- list()
  for (article in articles) {
    for (category in article[["categories"]])
      categories[[category]] <- not_null(categories[[category]], 0) + 1
  }

  if (length(categories) > 0) {

    # sort alphabetically
    indexes <- order(names(categories))
    categories <- categories[indexes]

    # caption for 'all articles'
    all_articles <- "Articles"
    if (identical(names(categories), tolower(names(categories))))
      all_articles <- tolower(all_articles)

    # generate html
    tags$div(class = "sidebar-section categories",
      tags$h3("Categories"),
      tags$ul(
        tags$li(
          tags$a(href = category_hash(all_articles), all_articles),
          tags$span(class = "category-count", sprintf("(%d)", length(articles)))
        ),
        lapply(names(categories), function(name) {
          tags$li(
            tags$a(href = category_hash(name), name),
            tags$span(class = "category-count", sprintf("(%d)", categories[[name]]))
          )
        })
      )
    )

  } else {
    NULL
  }
}


category_hash <- function(category) {
  paste0("#category:",gsub(" ", "_", category))
}


articles_info <- function(site_dir, collection) {
  collection <- as_collection_name(collection)
  collection_dir <- as_collection_dir(site_dir, collection)
  articles_json <- file.path(site_dir, collection_dir, file_with_ext(collection, "json"))
  if (!file.exists(articles_json))
    stop("The collection '", collection, "' does not have an article listing.\n",
         "(try running render_site() to generate the listing)", call. = FALSE)
  read_json(articles_json)
}

as_collection_dir <- function(site_dir, collection) {
  collection <- as_collection_name(collection)
  collection_dir <- paste0("_", collection)
  if (!dir_exists(file.path(site_dir, collection_dir)))
    stop("The collection '", collection, "' does not exist", call. = FALSE)
  collection_dir
}

as_collection_name <- function(collection) {
  if (is.list(collection))
    collection$name
  else
    collection
}

resolve_preview_url <- function(preview, path) {
  if (!is.null(preview) && !is_url(preview))
    preview <- url_path(path, preview)
  preview
}

absolute_preview_url <- function(preview_url, base_url) {
  if (!is.null(preview_url) && !is_url(preview_url))
    url_path(base_url, preview_url)
  else
    preview_url
}
jjallaire/radix documentation built on Nov. 28, 2023, 9:23 p.m.