R/calc_scores.yahtzee.R

Defines functions calc_scores.yahtzee apply_bonuses.yahtzee

# Ruleset for the official game
# Calculate available scores, given the current scorecard, and the dice rolls made
# Uses three steps
# 1 - Calculate available sores for all score types (including those already scored)
# 2 - Apply any additional "joker" rules for subsequent yahtzees after first yahtzee has been chosen
# 3 - NA any additional scores which have already been taken

# Input = a score table
calc_scores.yahtzee <- function(game) {

  t <- game$table

  ### ---- first calc possible scores for all scoring patterns ----

  t[t$section == "1s","score.available"] <- sum(game$dice == 1) * 1
  t[t$section == "2s","score.available"] <- sum(game$dice == 2) * 2
  t[t$section == "3s","score.available"] <- sum(game$dice == 3) * 3
  t[t$section == "4s","score.available"] <- sum(game$dice == 4) * 4
  t[t$section == "5s","score.available"] <- sum(game$dice == 5) * 5
  t[t$section == "6s","score.available"] <- sum(game$dice == 6) * 6
  t[t$section == "ch","score.available"] <- sum(game$dice)

  # for the "of a kind" type scores, calculate a table of frequencies, then utilise that
  x <- sort(table(game$dice),decreasing=T)
  t[t$section == "3k","score.available"] <- (x[1] >= 3)*sum(game$dice)
  t[t$section == "4k","score.available"] <- (x[1] >= 4)*sum(game$dice)
  t[t$section == "yz","score.available"] <- (x[1] == 5)*50
  t[t$section == "fh","score.available"] <- (x[1] == 3)*(x[2] == 2)*25

  #and then the staights..
  #calculate ordered string of distinct die values from the frequency table
  x2 <- paste(sort(names(x)),sep="",collapse="")
  t[t$section == "ss","score.available"] <- (grepl("1234",x2) == 1 | grepl("2345",x2) | grepl("3456",x2))*30
  t[t$section == "ls","score.available"] <- (x2 %in% c("12345","23456"))*40

  ### ---- Apply Joker rules ----
  # As per Forced Joker rules, as stated on wikipedia page

  if (x[1] == 5 & !is.na(t[t$section == "yz","score"])) {

    # (1) yahtzee score += 100 if have already positively scored for a previous yahtzee
    if (t[t$section == "yz","score"] > 0)
      t[t$section == "yz","score"] <- t[t$section == "yz","score"] + 100

    # (2a) if the corresponding upper section is availabel - that must be selected
    if (is.na(t[t$section == paste0(names(x)[1],"s"),"score"])) {
      t[t$section != paste0(names(x)[1],"s"),"score.available"] <- NA

    # (2b) Else if lower options are avilable, must pick one of those
    # Plus fh, ls and ss are available as jokers
    } else if (sum(is.na(t[t$half == 2, "score"])) > 0) {
      t[t$half == 1, "score.available"] <- NA
      t[t$section == "fh", "score.available"] <- 25
      t[t$section == "ss", "score.available"] <- 30
      t[t$section == "ls", "score.available"] <- 40

    }
    # (2c) If no lower options available, stuck with choosing an upper

  }

  ### ---- NA any options which have already been taken ----
  t$score.available[!is.na(t$score)] <- NA

  return(t)
}

## ---------------------------------------------------------------

apply_bonuses.yahtzee <- function(game) {

  table <- game$table

  ## calculate upper bonus
  if (sum(table[table$section %in% c("1s","2s","3s","4s","5s","6s"),"score"],na.rm=T) >= 63) {
    table[table$section=="ub", "score"] <- 35
  } else {
    if (sum(!is.na(table[table$section %in% c("1s","2s","3s","4s","5s","6s"),"score"])) == 6)
      table[table$section=="ub", "score"] <- 0
  }

  game$table <- table

  return(game)

}
JerBoon/yahtzR documentation built on Oct. 30, 2019, 7:39 p.m.