Title: | Ranking Teams by Elo Rating and Comparable Methods |
---|---|
Description: | A flexible framework for calculating Elo ratings and resulting rankings of any two-team-per-matchup system (chess, sports leagues, 'Go', etc.). This implementation is capable of evaluating a variety of matchups, Elo rating updates, and win probabilities, all based on the basic Elo rating system. It also includes methods to benchmark performance, including logistic regression and Markov chain models. |
Authors: | Ethan Heinzen [aut, cre] |
Maintainer: | Ethan Heinzen <[email protected]> |
License: | GPL (>= 2) |
Version: | 3.0.2.9000 |
Built: | 2025-01-08 04:27:44 UTC |
Source: | https://github.com/eheinzen/elo |
elo.run
objectCalculate AUC on an elo.run
object
## S3 method for class 'elo.run' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.glm' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.running' auc(object, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.winpct' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.colley' auc(object, ..., subset = TRUE)
## S3 method for class 'elo.run' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.glm' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.running' auc(object, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.winpct' auc(object, ..., subset = TRUE) ## S3 method for class 'elo.colley' auc(object, ..., subset = TRUE)
object |
An object of class |
... |
Other arguments (not used at this time). |
subset |
(optional) A vector of indices on which to calculate |
running |
logical, denoting whether to use the running predicted values. |
discard.skipped |
Logical, denoting whether to ignore the skipped observations in the calculation |
The AUC of the predicted Elo probabilities and the actual win results.
Adapted from code here: https://stat.ethz.ch/pipermail/r-help/2005-September/079872.html
An implementation of Elo ratings for general use in 'R'.
Listed below are the most useful functions available in elo
:
elo.prob
: Calculate the probability that team A beats team B.
elo.update
: Calculate the update value for a given Elo matchup.
elo.calc
: Calculate post-update Elo values.
elo.run
: Calculate Elos for a series of matches.
score
: Create a 1/0/0.5 win "indicator" based on two teams' scores.
tournament
: Mock data for examples.
Elo, A. E. 1978. The Rating of Chess Players, Past and Present. New York: Arco.
library(elo)
library(elo)
Calculate post-update Elo values. This is vectorized.
elo.calc(wins.A, ...) ## Default S3 method: elo.calc(wins.A, elo.A, elo.B, k, ..., adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.calc(formula, data, na.action, subset, k = NULL, ...)
elo.calc(wins.A, ...) ## Default S3 method: elo.calc(wins.A, elo.A, elo.B, k, ..., adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.calc(formula, data, na.action, subset, k = NULL, ...)
wins.A |
Numeric vector of wins by team A. |
... |
Other arguments (not in use at this time). |
elo.A , elo.B
|
Numeric vectors of elo scores. |
k |
A constant k-value (or a vector, where appropriate). |
adjust.A , adjust.B
|
Numeric vectors to adjust |
formula |
A formula. See the help page for formulas for details. |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
A data.frame with two columns, giving the new Elo values after each update.
elo.prob
, elo.update
,
elo.model.frame
elo.calc(c(1, 0), c(1500, 1500), c(1500, 1600), k = 20) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.calc(wins.A ~ elo.A + elo.B + k(k), data = dat)
elo.calc(c(1, 0), c(1500, 1500), c(1500, 1600), k = 20) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.calc(wins.A ~ elo.A + elo.B + k(k), data = dat)
Compute a Colley matrix model for a matchup.
elo.colley( formula, data, family = "binomial", weights, na.action, subset, k = 1, ..., running = FALSE, skip = 0 )
elo.colley( formula, data, family = "binomial", weights, na.action, subset, k = 1, ..., running = FALSE, skip = 0 )
formula |
A formula. See the help page for formulas for details. |
data |
A |
family |
Argument passed to |
weights |
A vector of weights. Note that these weights are used in the Colley matrix creation, but not the regression. |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
k |
The fraction of a win to be assigned to the winning team. See "details". |
... |
Argument passed to |
running |
Logical, denoting whether to calculate "running" projected probabilities. If true, a model is fit for
group 1 on its own to predict group 2, then groups 1 and 2 to predict 3, then groups 1 through 3 to predict 4, etc.
Groups are determined in |
skip |
Integer, denoting how many groups to skip before fitting the running models. This is helpful if groups are small, where glm would have trouble converging for the first few groups. The predicted values are then set to 0.5 for the skipped groups. |
See the vignette for details on this method.
The differences in assigned scores (from the coefficients of the Colley matrix regression) are fed into a logistic
regression model to predict wins or (usually) a linear model to predict margin of victory.
In this setting, 'k' indicates the fraction of a win to be assigned to the winning team
(and the fraction of a loss to be assigned to the losing team); setting k = 1
(the default)
emits the "Bias Free" ranking method presented by Colley.
It is also possible to adjust the regression by setting the second argument of
adjust()
. As in elo.glm
,
the intercept represents the home-field advantage. Neutral fields can be indicated
using the neutral()
function, which sets the intercept to 0.
Colley W.N. Colley's Bias Free College Football Ranking Method: The Colley Matrix Explained. 2002.
glm
, summary.elo.colley
, score
,
mov
, elo.model.frame
elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor)
elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor)
Compute a (usually logistic) regression model for a series of matches.
elo.glm( formula, data, family = "binomial", weights, na.action, subset, ..., running = FALSE, skip = 0 )
elo.glm( formula, data, family = "binomial", weights, na.action, subset, ..., running = FALSE, skip = 0 )
formula |
A formula. See the help page for formulas for details. |
data |
A |
family |
Argument passed to |
weights |
Argument passed to |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
... |
Argument passed to |
running |
Logical, denoting whether to calculate "running" projected probabilities. If true, a model is fit for
group 1 on its own to predict group 2, then groups 1 and 2 to predict 3, then groups 1 through 3 to predict 4, etc.
Groups are determined in |
skip |
Integer, denoting how many groups to skip before fitting the running models. This is helpful if groups are small, where glm would have trouble converging for the first few groups. The predicted values are then set to 0.5 for the skipped groups. |
The formula syntax is the same as other elo
functions. A data.frame
of indicator variables is built, where an entry is 1 if a team is home, 0 if
a team didn't play, and -1 if a team is a visitor. Anything passed to adjust()
in
formula
is also put in the data.frame. A glm
model is then
run to predict wins or margin of victory.
With this setup, the intercept represents the home-field advantage. Neutral fields can be indicated
using the neutral()
function, which sets the intercept to 0.
Note that any weights specified in players()
will be ignored.
This is essentially the Bradley-Terry model.
An object of class c("elo.glm", "glm")
. If running==TRUE
, the class "elo.glm.running"
is prepended.
https://en.wikipedia.org/wiki/Bradley
glm
, summary.elo.glm
, score
,
mov
, elo.model.frame
data(tournament) elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) elo.glm(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, family = "gaussian")
data(tournament) elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) elo.glm(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, family = "gaussian")
Compute a Markov chain model for a series of matches.
elo.markovchain( formula, data, family = "binomial", weights, na.action, subset, k = NULL, ..., running = FALSE, skip = 0 )
elo.markovchain( formula, data, family = "binomial", weights, na.action, subset, k = NULL, ..., running = FALSE, skip = 0 )
formula |
A formula. See the help page for formulas for details. |
data |
A |
family |
Argument passed to |
weights |
A vector of weights. Note that these weights are used in the Markov Chain model, but not the regression. |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
k |
The probability that the winning team is better given that they won. See details. |
... |
Argument passed to |
running |
Logical, denoting whether to calculate "running" projected probabilities. If true, a model is fit for
group 1 on its own to predict group 2, then groups 1 and 2 to predict 3, then groups 1 through 3 to predict 4, etc.
Groups are determined in |
skip |
Integer, denoting how many groups to skip before fitting the running models. This is helpful if groups are small, where glm would have trouble converging for the first few groups. The predicted values are then set to 0.5 for the skipped groups. |
See the vignette for details on this method. The probabilities we call 'k' purely for convenience.
The differences in assigned scores (from the stationary distribution pi) are fed into a logistic
regression model to predict wins or (usually) a linear model to predict margin of victory.
It is also possible to adjust the regression by setting the second argument of
adjust()
. As in elo.glm
,
the intercept represents the home-field advantage. Neutral fields can be indicated
using the neutral()
function, which sets the intercept to 0.
Note that by assigning probabilities in the right way, this function emits the Logistic Regression Markov Chain model (LRMC).
Kvam, P. and Sokol, J.S. A logistic regression/Markov chain model for NCAA basketball. Naval Research Logistics. 2006. 53; 788-803.
glm
, summary.elo.markovchain
, score
,
mov
, elo.model.frame
elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) elo.markovchain(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, family = "gaussian", data = tournament, k = 0.7)
elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) elo.markovchain(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, family = "gaussian", data = tournament, k = 0.7)
elo
functionsA helper function to create the model.frame
for many elo
functions.
elo.model.frame( formula, data, na.action, subset, k = NULL, ..., required.vars = "elos", warn.k = TRUE, ncol.k = 1, ncol.elos = 2 )
elo.model.frame( formula, data, na.action, subset, k = NULL, ..., required.vars = "elos", warn.k = TRUE, ncol.k = 1, ncol.elos = 2 )
formula |
A formula. See the help page for formulas for details. |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
k |
A constant k-value (or a vector, where appropriate). |
... |
Other arguments (not in use at this time). |
required.vars |
One or more of |
warn.k |
Should a warning be issued if |
ncol.k |
How many columns ( |
ncol.elos |
How many Elo columns are expected? |
elo.run
, elo.calc
, elo.update
, elo.prob
Create a "margin of victory" based on two teams' scores
mov(score.A, score.B = 0)
mov(score.A, score.B = 0)
score.A |
Numeric; the score of the first team. Alternatively, this can be a pre-computed margin of victory which will get compared to 0. |
score.B |
Numeric; the score of the second team; default is 0, in case
|
An object with class "elo.mov"
, denoting score.A
= score.B
.
mov(12, 10) mov(10, 10) mov(10, 12)
mov(12, 10) mov(10, 10) mov(10, 12)
Calculate the mean square error (Brier score) for a model.
mse(object, ..., subset = TRUE) brier(object, ..., subset = TRUE) ## S3 method for class 'elo.run' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.glm' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.running' mse(object, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.winpct' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.colley' mse(object, ..., subset = TRUE)
mse(object, ..., subset = TRUE) brier(object, ..., subset = TRUE) ## S3 method for class 'elo.run' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.glm' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.running' mse(object, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.winpct' mse(object, ..., subset = TRUE) ## S3 method for class 'elo.colley' mse(object, ..., subset = TRUE)
object |
An object |
... |
Other arguments (not used at this time). |
subset |
(optional) A vector of indices on which to calculate |
running |
logical, denoting whether to use the running predicted values. |
discard.skipped |
Logical, denoting whether to ignore the skipped observations in the calculation |
Even though logistic regressions don't use the MSE on the y=0/1 scale, it can still be informative.
Note that the S3 method is mse
.
Calculate the probability that team A beats team B. This is vectorized.
elo.prob(elo.A, ...) ## Default S3 method: elo.prob(elo.A, elo.B, ..., elos = NULL, adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.prob(formula, data, na.action, subset, ..., elos = NULL) ## S3 method for class 'elo.multiteam.matrix' elo.prob(elo.A, ..., elos = NULL)
elo.prob(elo.A, ...) ## Default S3 method: elo.prob(elo.A, elo.B, ..., elos = NULL, adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.prob(formula, data, na.action, subset, ..., elos = NULL) ## S3 method for class 'elo.multiteam.matrix' elo.prob(elo.A, ..., elos = NULL)
elo.A , elo.B
|
Numeric vectors of elo scores, or else vectors of teams. |
... |
Other arguments (not in use at this time). |
elos |
An optional named vector containing Elo ratings for all teams in |
adjust.A , adjust.B
|
Numeric vectors to adjust |
formula |
A formula. See the help page for formulas for details. |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
Note that formula
can be missing the wins.A
component. If
present, it's ignored by elo.model.frame
.
A vector of Elo probabilities.
elo.update
, elo.calc
,
elo.model.frame
elo.prob(1500, 1500) elo.prob(c(1500, 1500), c(1500, 1600)) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.prob(~ elo.A + elo.B, data = dat) ## Also works to include the wins and k: elo.prob(wins.A ~ elo.A + elo.B + k(k), data = dat) ## Also allows teams elo.prob(c("A", "B"), c("C", "C"), elos = c(A = 1500, B = 1600, C = 1500))
elo.prob(1500, 1500) elo.prob(c(1500, 1500), c(1500, 1600)) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.prob(~ elo.A + elo.B, data = dat) ## Also works to include the wins and k: elo.prob(wins.A ~ elo.A + elo.B + k(k), data = dat) ## Also allows teams elo.prob(c("A", "B"), c("C", "C"), elos = c(A = 1500, B = 1600, C = 1500))
Calculate running Elos for a series of matches.
elo.run( formula, data, na.action, subset, k = NULL, initial.elos = NULL, ..., prob.fun = elo.prob, update.fun = elo.update, verbose = TRUE )
elo.run( formula, data, na.action, subset, k = NULL, initial.elos = NULL, ..., prob.fun = elo.prob, update.fun = elo.update, verbose = TRUE )
formula |
A formula. See the help page for formulas for details. |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
k |
A constant k-value (or a vector, where appropriate). |
initial.elos |
An optional named vector containing initial Elo ratings for all teams in |
... |
Other arguments (not used at this time). |
prob.fun |
A function with at least 4 arguments: elo.A, elo.B, adjust.A, and adjust.B. It should return a predicted probability that team A wins. The values passed in will be scalars, and a scalar is expected as output. |
update.fun |
A function with at least 6 arguments: the same as |
verbose |
Should a message be issued when R is used (over C++)? |
elo.run
is run two different ways: the first (default) uses C++ and may be up to 50 times faster,
while the second (when prob.fun
or update.fun
are specified) uses R but also supports custom update functions.
Prefer the first unless you really need a custom update function.
An object of class "elo.run"
or class "elo.run.regressed"
.
score
, elo.run.helperselo.run helpers, elo.calc
,
elo.update
, elo.prob
, elo.model.frame
.
data(tournament) elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, k = 20) # Create non-constant 'k' elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor + k(20*log(abs(points.Home - points.Visitor) + 1)), data = tournament) # Adjust Elo for, e.g., home-field advantage elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, 30) + team.Visitor, data = tournament, k = 20) tournament$home.field <- 30 elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, home.field) + team.Visitor, data = tournament, k = 20) # Regress the Elos back toward 1500 at the end of the half-season elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, 30) + team.Visitor + regress(half, 1500, 0.2), data = tournament, k = 20)
data(tournament) elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, k = 20) # Create non-constant 'k' elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor + k(20*log(abs(points.Home - points.Visitor) + 1)), data = tournament) # Adjust Elo for, e.g., home-field advantage elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, 30) + team.Visitor, data = tournament, k = 20) tournament$home.field <- 30 elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, home.field) + team.Visitor, data = tournament, k = 20) # Regress the Elos back toward 1500 at the end of the half-season elo.run(score(points.Home, points.Visitor) ~ adjust(team.Home, 30) + team.Visitor + regress(half, 1500, 0.2), data = tournament, k = 20)
elo.run
as.matrix
converts an Elo object into a matrix of running Elos. These are the Elos at the time of grouping,
but before any regression takes place.
## S3 method for class 'elo.run' as.matrix(x, ...) ## S3 method for class 'elo.run.regressed' as.matrix(x, ...) ## S3 method for class 'elo.run' as.data.frame(x, ...) final.elos(x, ...) ## S3 method for class 'elo.run' final.elos(x, ...) ## S3 method for class 'elo.run.regressed' final.elos(x, regressed = FALSE, ...)
## S3 method for class 'elo.run' as.matrix(x, ...) ## S3 method for class 'elo.run.regressed' as.matrix(x, ...) ## S3 method for class 'elo.run' as.data.frame(x, ...) final.elos(x, ...) ## S3 method for class 'elo.run' final.elos(x, ...) ## S3 method for class 'elo.run.regressed' final.elos(x, regressed = FALSE, ...)
x |
An object of class |
... |
Other arguments (Not in use at this time). |
regressed |
Logical, denoting whether to use the post-regressed ( |
as.data.frame
converts the "elos"
component of an object
from elo.run
into a data.frame.
final.elos
is a generic function to extract the last Elo per team.
A matrix, a data.frame, or a named vector.
e <- elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor + group(week), data = tournament, k = 20) head(as.matrix(e)) str(as.data.frame(e)) final.elos(e)
e <- elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor + group(week), data = tournament, k = 20) head(as.matrix(e)) str(as.data.frame(e)) final.elos(e)
Calculate running Elos for a series of multi-team matches.
elo.run.multiteam( formula, data, na.action, subset, k = NULL, initial.elos = NULL, ... )
elo.run.multiteam( formula, data, na.action, subset, k = NULL, initial.elos = NULL, ... )
formula |
A one-sided formula with a |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
k |
A constant k-value (or a vector, where appropriate). |
initial.elos |
An optional named vector containing initial Elo ratings for all teams in |
... |
Other arguments (not used at this time). |
This is like elo.run
(and in fact it runs elo.run
in the background).
The formula takes a multiteam()
object, which assumes that teams "win"
in a well-ordered ranking. It assumes that the first place team beats all other teams,
that the second place team loses to the first but beats the others, etc. In that regard,
elo.run.multiteam
reduces to elo.run
when the number of teams (ncol(multiteam())
) is 2
However, this is less flexible than elo.run
, because (1) there cannot be ties; (2) it does not accept
adjustments; and (3) k is constant within a "game"
data(tournament.multiteam) elo.run.multiteam(~ multiteam(Place_1, Place_2, Place_3, Place_4), data = tournament.multiteam, subset = -28, k = 20)
data(tournament.multiteam) elo.run.multiteam(~ multiteam(Place_1, Place_2, Place_3, Place_4), data = tournament.multiteam, subset = -28, k = 20)
Calculate the update value for a given Elo matchup. This is used in
elo.calc
, which reports the post-update Elo values. This is vectorized.
elo.update(wins.A, ...) ## Default S3 method: elo.update(wins.A, elo.A, elo.B, k, ..., adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.update(formula, data, na.action, subset, k = NULL, ...)
elo.update(wins.A, ...) ## Default S3 method: elo.update(wins.A, elo.A, elo.B, k, ..., adjust.A = 0, adjust.B = 0) ## S3 method for class 'formula' elo.update(formula, data, na.action, subset, k = NULL, ...)
wins.A |
Numeric vector of wins by team A. |
... |
Other arguments (not in use at this time). |
elo.A , elo.B
|
Numeric vectors of elo scores. |
k |
A constant k-value (or a vector, where appropriate). |
adjust.A , adjust.B
|
Numeric vectors to adjust |
formula |
A formula. See the help page for formulas for details. |
data |
A |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
A vector of Elo updates.
elo.prob
, elo.calc
,
elo.model.frame
elo.update(c(1, 0), c(1500, 1500), c(1500, 1600), k = 20) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.update(wins.A ~ elo.A + elo.B + k(k), data = dat)
elo.update(c(1, 0), c(1500, 1500), c(1500, 1600), k = 20) dat <- data.frame(wins.A = c(1, 0), elo.A = c(1500, 1500), elo.B = c(1500, 1600), k = c(20, 20)) elo.update(wins.A ~ elo.A + elo.B + k(k), data = dat)
Compute a (usually logistic) regression based on win percentage for a series of matches.
elo.winpct( formula, data, family = "binomial", weights, na.action, subset, ..., running = FALSE, skip = 0 )
elo.winpct( formula, data, family = "binomial", weights, na.action, subset, ..., running = FALSE, skip = 0 )
formula |
A formula. See the help page for formulas for details. |
data |
A |
family |
Argument passed to |
weights |
A vector of weights. Note that these are used in calculating wins and losses but not in the regression. |
na.action |
A function which indicates what should happen when the data contain NAs. |
subset |
An optional vector specifying a subset of observations. |
... |
Argument passed to |
running |
Logical, denoting whether to calculate "running" projected probabilities. If true, a model is fit for
group 1 on its own to predict group 2, then groups 1 and 2 to predict 3, then groups 1 through 3 to predict 4, etc.
Groups are determined in |
skip |
Integer, denoting how many groups to skip before fitting the running models. This is helpful if groups are small, where glm would have trouble converging for the first few groups. The predicted values are then set to 0.5 for the skipped groups. |
Win percentages are first calculated. Anything passed to adjust()
in
formula
is also put in the data.frame. A glm
model is then
run to predict wins or margin of victory.
With this setup, the intercept represents the home-field advantage. Neutral fields can be indicated
using the neutral()
function, which sets the intercept to 0.
glm
, summary.elo.winpct
, score
,
mov
, elo.model.frame
elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) elo.winpct(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, family = "gaussian")
elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) elo.winpct(mov(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, family = "gaussian")
Classify teams that are favored to win
favored(x, ..., subset = TRUE) ## S3 method for class 'elo.run' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.glm' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.running' favored(x, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.winpct' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.colley' favored(x, ..., subset = TRUE) ## Default S3 method: favored(x, p.A, ...)
favored(x, ..., subset = TRUE) ## S3 method for class 'elo.run' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.glm' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.running' favored(x, running = TRUE, discard.skipped = FALSE, ..., subset = TRUE) ## S3 method for class 'elo.markovchain' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.winpct' favored(x, ..., subset = TRUE) ## S3 method for class 'elo.colley' favored(x, ..., subset = TRUE) ## Default S3 method: favored(x, p.A, ...)
x |
An object from |
... |
Other arguments (not used at this time). |
subset |
(optional) A vector of indices on which to calculate |
running |
logical, denoting whether to use the running predicted values. |
discard.skipped |
Logical, denoting whether to ignore the skipped observations in the calculation |
p.A |
A vector of predicted win probabilities. |
Extract model values from elo
functions.
## S3 method for class 'elo.run' fitted(object, ...) ## S3 method for class 'elo.run' residuals(object, ...) ## S3 method for class 'elo.running' fitted(object, running = TRUE, ...) ## S3 method for class 'elo.glm' fitted(object, ...) ## S3 method for class 'elo.markovchain' fitted(object, ...) ## S3 method for class 'elo.winpct' fitted(object, ...) ## S3 method for class 'elo.colley' fitted(object, ...)
## S3 method for class 'elo.run' fitted(object, ...) ## S3 method for class 'elo.run' residuals(object, ...) ## S3 method for class 'elo.running' fitted(object, running = TRUE, ...) ## S3 method for class 'elo.glm' fitted(object, ...) ## S3 method for class 'elo.markovchain' fitted(object, ...) ## S3 method for class 'elo.winpct' fitted(object, ...) ## S3 method for class 'elo.colley' fitted(object, ...)
object |
An object. |
... |
Other arguments |
running |
logical, denoting whether to use the running predicted values. |
A vector of fitted values. For running values, it has an additional attribute denoting to which group (i.e., which model) the prediction belongs
elo
formulas and the specials thereinDetails on elo
functions and the special functions allowed in them to change functions' behaviors.
players(..., weights = NULL) multiteam(...) k(x, y = NULL) adjust(x, adjustment) regress(x, to, by, regress.unused = TRUE) group(x) neutral(x)
players(..., weights = NULL) multiteam(...) k(x, y = NULL) adjust(x, adjustment) regress(x, to, by, regress.unused = TRUE) group(x) neutral(x)
... |
Vectors to be coerced to character, which comprise of the players of a team. |
weights |
A vector giving the weights of Elo updates for the players in |
x , y
|
A vector. |
adjustment |
A single value or a vector of the same length as |
to |
Numeric: what Elo to regress to. Can be a single value or named vector the same length as the number of teams. |
by |
Numeric: by how much should Elos be regressed toward |
regress.unused |
Logical: whether to continue regressing teams which have stopped playing. |
In the functions in this package, formula
is usually of the form wins.A ~ elo.A + elo.B
,
where elo.A
and elo.B
are vectors of Elos, and wins.A
is between 0 and 1,
denoting whether team A (Elo A) won or lost (or something between). elo.prob
also allows
elo.A
and elo.B
to be character or factors, denoting which team(s) played. elo.run
requires elo.A
to be a vector of teams or a players matrix from players()
(sometimes denoted by "team.A"
), but elo.B
can be either a vector of teams or
players matrix ("team.B"
) or else a numeric column (denoting a fixed-Elo opponent).
elo.glm
requires both to be a vector of teams or players matrix. elo.markovchain
requires both to be a vector of teams.
formula
accepts six special functions in it:
k()
allows for complicated Elo updates. For
constant Elo updates, use the k =
argument instead of this special function.
Note that elo.markovchain
uses this function (or argument) as a convenient
way of specifying transition probabilities. elo.colley
uses this to indicate
the fraction of a win to be assigned to the winning team.
adjust()
allows for Elos to be adjusted for, e.g., home-field advantage. The second argument
to this function can be a scalar or vector of appropriate length. This can also be used in
elo.glm
and elo.markovchain
as an adjuster to the logistic regressions.
regress()
can be used to regress Elos back to a fixed value
after certain matches. Giving a logical vector identifies these matches after which to
regress back to the mean. Giving any other kind of vector regresses after the appropriate
groupings (see, e.g., duplicated(..., fromLast = TRUE)
). The other three arguments determine
what Elo to regress to (to =
), by how much to regress toward that value
(by =
), and whether to continue regressing teams which have stopped playing (regress.unused
,
default = TRUE
).
group()
is used to group matches (by, e.g., week). For elo.run
, Elos are not updated until
the group changes. It is also fed to as.matrix.elo.run
, giving the number of rows to return.
to produce only certain rows of matrix output. It also determines how many models to run (and on what data)
for elo.glm
and elo.markovchain
when running=TRUE
.
neutral()
is used in elo.glm
and elo.markovchain
to determine the intercept.
In short, the intercept is 1 - neutral()
, denoting home-field advantage. Therefore, the column
passed should be 0 (denoting home-field advantange) or 1 (denoting a neutral game). If omitted, all matches
are assumed to have home field advantage.
players()
is used for multiple players on a team contributing to an overall Elo. The Elo updates
are then assigned based on the specified weights. The weights are ignored in elo.glm
.
multiteam()
is used for matchups consisting of multiple teams and is only valid in elo.run.multiteam
.
elo
ObjectMake Predictions on an elo
Object
## S3 method for class 'elo.run' predict(object, newdata, ...) ## S3 method for class 'elo.run.regressed' predict(object, newdata, regressed = FALSE, ...) ## S3 method for class 'elo.run.multiteam' predict(object, newdata, ...) ## S3 method for class 'elo.glm' predict(object, newdata, type = "response", ...) ## S3 method for class 'elo.running' predict(object, newdata, running = TRUE, ...) ## S3 method for class 'elo.markovchain' predict(object, newdata, ...) ## S3 method for class 'elo.colley' predict(object, newdata, ...) ## S3 method for class 'elo.winpct' predict(object, newdata, ...)
## S3 method for class 'elo.run' predict(object, newdata, ...) ## S3 method for class 'elo.run.regressed' predict(object, newdata, regressed = FALSE, ...) ## S3 method for class 'elo.run.multiteam' predict(object, newdata, ...) ## S3 method for class 'elo.glm' predict(object, newdata, type = "response", ...) ## S3 method for class 'elo.running' predict(object, newdata, running = TRUE, ...) ## S3 method for class 'elo.markovchain' predict(object, newdata, ...) ## S3 method for class 'elo.colley' predict(object, newdata, ...) ## S3 method for class 'elo.winpct' predict(object, newdata, ...)
object |
An model from which to get predictions. |
newdata |
A new dataset containing the same variables as the call
that made |
... |
Other arguments. |
regressed |
See the note on |
type |
See |
running |
logical, denoting whether to use the running predicted values. Only makes
sense if |
Note that the "elo.glm.running"
objects will use a model fit on all the data to predict.
A vector of win probabilities.
data(tournament) t1 <- head(tournament, -3) t2 <- tail(tournament, 3) results <- elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, k = 20) predict(results) predict(results, newdata = t2) results <- elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor) predict(results) predict(results, newdata = t2) results <- elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor, k = 0.7) predict(results) predict(results, newdata = t2) results <- elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor) predict(results) predict(results, newdata = t2) results <- elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor, k = 0.7) predict(results) predict(results, newdata = t2)
data(tournament) t1 <- head(tournament, -3) t2 <- tail(tournament, 3) results <- elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, k = 20) predict(results) predict(results, newdata = t2) results <- elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor) predict(results) predict(results, newdata = t2) results <- elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor, k = 0.7) predict(results) predict(results, newdata = t2) results <- elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor) predict(results) predict(results, newdata = t2) results <- elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = t1, subset = points.Home != points.Visitor, k = 0.7) predict(results) predict(results, newdata = t2)
Extract the rankings from Elo objects.
rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.run' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.run.regressed' rank.teams(object, ties.method = "min", regressed = FALSE, ...) ## S3 method for class 'elo.glm' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.markovchain' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.winpct' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.colley' rank.teams(object, ties.method = "min", ...)
rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.run' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.run.regressed' rank.teams(object, ties.method = "min", regressed = FALSE, ...) ## S3 method for class 'elo.glm' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.markovchain' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.winpct' rank.teams(object, ties.method = "min", ...) ## S3 method for class 'elo.colley' rank.teams(object, ties.method = "min", ...)
object |
An object. |
ties.method |
Passed to |
... |
Other arguments |
regressed |
Passed to |
Create a 1/0/0.5 win "indicator" based on two teams' scores, and test for "score-ness".
score(score.A, score.B) is.score(x)
score(score.A, score.B) is.score(x)
score.A |
Numeric; the score of the first team (whose wins are to be denoted by 1). |
score.B |
Numeric; the score of the second team (whose wins are to be denoted by 0). |
x |
An R object. |
For score
, a vector containing 0, 1, and 0.5 (for ties). For
is.score
, TRUE
or FALSE
depending on whether all values of
x
are between 0 and 1 (inclusive).
score(12, 10) score(10, 10) score(10, 12)
score(12, 10) score(10, 10) score(10, 12)
elo
ObjectSummarize an elo
Object
## S3 method for class 'elo.run' summary(object, ...) ## S3 method for class 'elo.glm' summary(object, ...) ## S3 method for class 'elo.markovchain' summary(object, ...) ## S3 method for class 'elo.colley' summary(object, ...) ## S3 method for class 'elo.winpct' summary(object, ...)
## S3 method for class 'elo.run' summary(object, ...) ## S3 method for class 'elo.glm' summary(object, ...) ## S3 method for class 'elo.markovchain' summary(object, ...) ## S3 method for class 'elo.colley' summary(object, ...) ## S3 method for class 'elo.winpct' summary(object, ...)
object |
An object to summarize. |
... |
Other arguments |
A summary of object
.
summary(elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, k = 20)) summary(elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament)) mc <- elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) summary(mc) co <- elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) summary(co) wp <- elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) summary(wp)
summary(elo.run(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, k = 20)) summary(elo.glm(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament)) mc <- elo.markovchain(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) summary(mc) co <- elo.colley(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor) summary(co) wp <- elo.winpct(score(points.Home, points.Visitor) ~ team.Home + team.Visitor, data = tournament, subset = points.Home != points.Visitor, k = 0.7) summary(wp)
tournament
: Mock data for examplesA fake dataset containing results from "animal-ball" matches.
A data frame with 56 observations on the following 4 variables:
team.Home
The home team for the match
team.Visitor
The visiting team for the match
points.Home
Number of points scored by the home team
points.Visitor
Number of points scored by the visiting team
week
Week Number
half
The half of the season in which the match was played
data(tournament) str(tournament)
data(tournament) str(tournament)
tournament.multiteam
: Mock data for examplesA fake dataset containing results from "animal-ball" matches.
A data frame with 56 observations on the following 4 variables:
week
Week Number
half
The half of the season in which the match was played
Place_1
The first-place team
Place_2
The second-place team
Place_3
The third-place team
Place_4
The fourth-place team